staticalyze object instanciation

This commit is contained in:
Jean-Marie 'Histausse' Mineau 2025-02-04 19:01:07 +01:00
parent 6b15bbf748
commit 5fdeb25682
Signed by: histausse
GPG key ID: B66AEEDA9B645AD2
3 changed files with 518 additions and 168 deletions

View file

@ -1,3 +1,4 @@
use androscalpel::SmaliName;
use androscalpel::{IdMethod, Instruction, Method}; use androscalpel::{IdMethod, Instruction, Method};
use anyhow::{bail, Context, Result}; use anyhow::{bail, Context, Result};
use std::sync::LazyLock; use std::sync::LazyLock;
@ -10,13 +11,21 @@ pub mod get_apk;
// does. // does.
/// Structure storing the runtime information of a reflection call. /// Structure storing the runtime information of a reflection call.
pub struct ReflectionData { pub struct ReflectionInvokeData {
pub method: IdMethod, pub method: IdMethod,
// TODO: variable number of args? // TODO: variable number of args?
// TODO: type of invoke? // TODO: type of invoke?
} }
struct RegistersInfo { pub struct ReflectionClassNewInstData {
pub constructor: IdMethod,
}
pub struct ReflectionCnstrNewInstData {
pub constructor: IdMethod,
}
pub struct RegistersInfo {
pub array_index: u8, pub array_index: u8,
//pub array: u8, //pub array: u8,
pub array_val: u8, pub array_val: u8,
@ -50,26 +59,44 @@ static MTH_GET_PARAMS_TY: LazyLock<IdMethod> = LazyLock::new(|| {
static MTH_GET_RET_TY: LazyLock<IdMethod> = LazyLock::new(|| { static MTH_GET_RET_TY: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali("Ljava/lang/reflect/Method;->getReturnType()Ljava/lang/Class;").unwrap() IdMethod::from_smali("Ljava/lang/reflect/Method;->getReturnType()Ljava/lang/Class;").unwrap()
}); });
static MTH_GET_DEC_TY: LazyLock<IdMethod> = LazyLock::new(|| { static MTH_GET_DEC_CLS: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali("Ljava/lang/reflect/Method;->getDeclaringClass()Ljava/lang/Class;") IdMethod::from_smali("Ljava/lang/reflect/Method;->getDeclaringClass()Ljava/lang/Class;")
.unwrap() .unwrap()
}); });
static STR_EQ: LazyLock<IdMethod> = LazyLock::new(|| { static STR_EQ: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali("Ljava/lang/String;->equals(Ljava/lang/Object;)Z").unwrap() IdMethod::from_smali("Ljava/lang/String;->equals(Ljava/lang/Object;)Z").unwrap()
}); });
static CLASS_NEW_INST: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali("Ljava/lang/Class;->newInstance()Ljava/lang/Object;").unwrap()
});
static CNSTR_NEW_INST: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali(
"Ljava/lang/reflect/Constructor;->newInstance([Ljava/lang/Object;)Ljava/lang/Object;",
)
.unwrap()
});
static CNSTR_GET_PARAMS_TY: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali("Ljava/lang/reflect/Constructor;->getParameterTypes()[Ljava/lang/Class;")
.unwrap()
});
static CNSTR_GET_DEC_CLS: LazyLock<IdMethod> = LazyLock::new(|| {
IdMethod::from_smali("Ljava/lang/reflect/Constructor;->getDeclaringClass()Ljava/lang/Class;")
.unwrap()
});
// Interesting stuff: https://cs.android.com/android/platform/superproject/main/+/main:art/runtime/verifier/reg_type.h;drc=83db0626fad8c6e0508754fffcbbd58e539d14a5;l=94 // Interesting stuff: https://cs.android.com/android/platform/superproject/main/+/main:art/runtime/verifier/reg_type.h;drc=83db0626fad8c6e0508754fffcbbd58e539d14a5;l=94
// https://cs.android.com/android/platform/superproject/main/+/main:art/runtime/verifier/method_verifier.cc;drc=83db0626fad8c6e0508754fffcbbd58e539d14a5;l=5328 // https://cs.android.com/android/platform/superproject/main/+/main:art/runtime/verifier/method_verifier.cc;drc=83db0626fad8c6e0508754fffcbbd58e539d14a5;l=5328
pub fn transform_method(meth: &mut Method, ref_data: &ReflectionData) -> Result<()> { pub fn transform_method(
meth: &mut Method,
ref_invoke_data: &ReflectionInvokeData,
ref_class_new_inst_data: &ReflectionClassNewInstData,
ref_cnstr_new_inst_data: &ReflectionCnstrNewInstData,
) -> Result<()> {
// checking meth.annotations might be usefull at some point // checking meth.annotations might be usefull at some point
let code = meth let code = meth
.code .code
.as_mut() .as_mut()
.with_context(|| format!("Code not found in {}", meth.descriptor.__str__()))?; .with_context(|| format!("Code not found in {}", meth.descriptor.__str__()))?;
println!(
"registers_size: {}\nins_size: {}\nouts_size: {}",
code.registers_size, code.ins_size, code.outs_size,
);
// TODO // TODO
if code.registers_size + RegistersInfo::NB_U8_REG > u8::MAX as u16 { if code.registers_size + RegistersInfo::NB_U8_REG > u8::MAX as u16 {
bail!("To many registers"); bail!("To many registers");
@ -83,43 +110,67 @@ pub fn transform_method(meth: &mut Method, ref_data: &ReflectionData) -> Result<
nb_arg_reg: 0, nb_arg_reg: 0,
}; };
let mut new_insns = vec![]; let mut new_insns = vec![];
let mut iter = code.insns.iter().peekable(); let mut iter = code.insns.iter();
while let Some(ins) = iter.next() { while let Some(ins) = iter.next() {
match ins { match ins {
Instruction::InvokeVirtual { method, args } if method == &*MTH_INVOKE => { Instruction::InvokeVirtual { method, args }
let move_ret = match iter.peek() { if method == &*MTH_INVOKE
Some(Instruction::MoveResult { .. }) || method == &*CLASS_NEW_INST
| Some(Instruction::MoveResultWide { .. }) || method == &*CNSTR_NEW_INST =>
| Some(Instruction::MoveResultObject { .. }) => iter.next().cloned(), {
_ => None, let (pseudo_insns, move_ret) = get_move_result(iter.clone());
if move_ret.is_some() {
while move_ret.as_ref() != iter.next() {}
}
let end_label = if method == &*MTH_INVOKE {
format!("end_reflection_call_at_{}", "TODO_ADDR")
} else if method == &*CLASS_NEW_INST || method == &*CNSTR_NEW_INST {
format!("end_reflection_instanciation_at_{}", "TODO_ADDR")
} else {
panic!("Should not happen!")
}; };
// TODO: rever from get_invoke_block failure // TODO: recover from failure
let label: String = "TODO_NAME_THIS".into(); let ins_block = if method == &*MTH_INVOKE {
for ins in get_invoke_block( get_invoke_block(
ref_data, ref_invoke_data,
args.as_slice(), args.as_slice(),
&mut register_info, &mut register_info,
&label, &end_label,
move_ret.clone(), move_ret.clone(),
)? )?
.into_iter() } else if method == &*CLASS_NEW_INST {
{ get_class_new_inst_block(
println!(" \x1b[92m{}\x1b[0m", ins.__str__()); ref_class_new_inst_data,
args.as_slice(),
&mut register_info,
&end_label,
move_ret.clone(),
)?
} else if method == &*CNSTR_NEW_INST {
get_cnstr_new_inst_block(
ref_cnstr_new_inst_data,
args.as_slice(),
&mut register_info,
&end_label,
move_ret.clone(),
)?
} else {
panic!("Should not happen!")
};
for ins in ins_block.into_iter() {
new_insns.push(ins); new_insns.push(ins);
} }
new_insns.push(ins.clone()); new_insns.push(ins.clone());
if let Some(move_ret) = move_ret { if let Some(move_ret) = move_ret {
for ins in pseudo_insns.into_iter() {
new_insns.push(ins);
}
new_insns.push(move_ret); new_insns.push(move_ret);
} }
println!(" \x1b[91m{}\x1b[0m", ins.__str__()); let end_label = Instruction::Label { name: end_label };
let lab = Instruction::Label { new_insns.push(end_label.clone());
name: format!("{label}_END"),
};
new_insns.push(lab.clone());
println!(" \x1b[91m{}\x1b[0m", lab.__str__());
} }
ins => { ins => {
println!(" {}", ins.__str__());
new_insns.push(ins.clone()); new_insns.push(ins.clone());
} }
} }
@ -137,18 +188,30 @@ pub fn transform_method(meth: &mut Method, ref_data: &ReflectionData) -> Result<
code.insns.append(&mut new_insns); code.insns.append(&mut new_insns);
code.registers_size += register_info.get_nb_added_reg(); code.registers_size += register_info.get_nb_added_reg();
println!("\nnew code:\n");
for ins in &code.insns {
println!(" {}", ins.__str__());
}
Ok(()) Ok(())
} }
/// Return the MoveResult{,Wide,Object} associated to the last instruction of the iterator.
/// TODO: return the list of pseudo instruction between the last instruction and the move result.
fn get_move_result<'a>(
mut iter: impl Iterator<Item = &'a Instruction>,
) -> (Vec<Instruction>, Option<Instruction>) {
if let Some(ins) = iter.next() {
match ins {
Instruction::MoveResult { .. }
| Instruction::MoveResultWide { .. }
| Instruction::MoveResultObject { .. } => return (vec![], Some(ins.clone())),
_ => (), // break,
}
}
(vec![], None)
}
fn get_invoke_block( fn get_invoke_block(
ref_data: &ReflectionData, ref_data: &ReflectionInvokeData,
invoke_arg: &[u16], invoke_arg: &[u16],
reg_inf: &mut RegistersInfo, reg_inf: &mut RegistersInfo,
label: &str, end_label: &str,
move_result: Option<Instruction>, move_result: Option<Instruction>,
) -> Result<Vec<Instruction>> { ) -> Result<Vec<Instruction>> {
let (method_obj, obj_inst, arg_arr) = if let &[a, b, c] = invoke_arg { let (method_obj, obj_inst, arg_arr) = if let &[a, b, c] = invoke_arg {
@ -167,117 +230,32 @@ fn get_invoke_block(
if reg_inf.nb_arg_reg < nb_args as u16 + 1 { if reg_inf.nb_arg_reg < nb_args as u16 + 1 {
reg_inf.nb_arg_reg = nb_args as u16 + 1; reg_inf.nb_arg_reg = nb_args as u16 + 1;
} }
let mut insns = vec![
// Check the runtime method is the right one let abort_label = format!(
// Check Name "end_static_call_to_{}_at_{}",
Instruction::InvokeVirtual { ref_data.method.try_to_smali()?,
method: MTH_GET_NAME.clone(), "TODO_ADDR"
args: vec![method_obj], );
}, let mut insns = test_method(
Instruction::MoveResultObject { method_obj,
to: reg_inf.array_index, // wrong name, but available for tmp val ref_data.method.clone(),
}, abort_label.clone(),
Instruction::ConstString { reg_inf,
reg: reg_inf.array_val, // wrong name, but available for tmp val );
lit: ref_data.method.name.clone(),
},
Instruction::InvokeVirtual {
method: STR_EQ.clone(),
args: vec![reg_inf.array_index as u16, reg_inf.array_val as u16],
},
Instruction::MoveResult {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::IfEqZ {
a: reg_inf.array_index,
label: format!("{label}_END_OF_CALL_1"), // TODO: rename 1
},
// Check Return Type
Instruction::InvokeVirtual {
method: MTH_GET_RET_TY.clone(),
args: vec![method_obj],
},
Instruction::MoveResultObject {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::ConstClass {
reg: reg_inf.array_val, // wrong name, but available for tmp val
lit: ref_data.method.proto.get_return_type(),
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: format!("{label}_END_OF_CALL_1"), // TODO: rename 1
},
// Check Declaring Type
Instruction::InvokeVirtual {
method: MTH_GET_DEC_TY.clone(),
args: vec![method_obj],
},
Instruction::MoveResultObject {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::ConstClass {
reg: reg_inf.array_val, // wrong name, but available for tmp val
lit: ref_data.method.class_.clone(),
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: format!("{label}_END_OF_CALL_1"), // TODO: rename 1
},
];
// Check for arg type
insns.push(Instruction::InvokeVirtual {
method: MTH_GET_PARAMS_TY.clone(),
args: vec![method_obj],
});
insns.push(Instruction::MoveResultObject { to: reg_inf.array });
// First check the number of args
insns.push(Instruction::ArrayLength {
dest: reg_inf.array_index,
arr: reg_inf.array,
});
insns.push(Instruction::Const {
reg: reg_inf.array_val,
lit: ref_data.method.proto.get_parameters().len() as i32,
});
insns.push(Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: format!("{label}_END_OF_CALL_1"), // TODO: rename 1
});
// then the type of each arg
for (i, param) in ref_data
.method
.proto
.get_parameters()
.into_iter()
.enumerate()
{
insns.push(Instruction::Const {
reg: reg_inf.array_index,
lit: i as i32,
});
insns.push(Instruction::AGetObject {
dest: reg_inf.array_val,
arr: reg_inf.array,
idx: reg_inf.array_index,
});
insns.push(Instruction::ConstClass {
reg: reg_inf.array_index, // wrong name, but available for tmp val
lit: param,
});
insns.push(Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: format!("{label}_END_OF_CALL_1"), // TODO: rename 1
})
}
// Move 'this' to fist arg // Move 'this' to fist arg
// We do a small detour to `reg_inf.array_val` because we need a u8 reg to down cast the
// Object reference to the right Class
insns.push(Instruction::MoveObject { insns.push(Instruction::MoveObject {
from: obj_inst, from: obj_inst,
to: reg_inf.array_val as u16,
});
insns.push(Instruction::CheckCast {
reg: reg_inf.array_val,
lit: ref_data.method.class_.clone(),
});
insns.push(Instruction::MoveObject {
from: reg_inf.array_val as u16,
to: reg_inf.first_arg, to: reg_inf.first_arg,
}); });
for (i, param) in ref_data.method.proto.get_parameters().iter().enumerate() { for (i, param) in ref_data.method.proto.get_parameters().iter().enumerate() {
@ -307,11 +285,354 @@ fn get_invoke_block(
insns.push(move_result); insns.push(move_result);
} }
insns.push(Instruction::Goto { insns.push(Instruction::Goto {
label: format!("{label}_END"), label: end_label.to_string(),
});
insns.push(Instruction::Label {
name: format!("{label}_END_OF_CALL_1"),
}); });
insns.push(Instruction::Label { name: abort_label });
// We need a few u8 regs here. For now, we assumes we work with less than 256 reg. // We need a few u8 regs here. For now, we assumes we work with less than 256 reg.
Ok(insns) Ok(insns)
} }
/// Generate bytecode that test if a `java.lang.reflect.Method` is equal to an [`IdMethod`]
///
/// - `method_obj_reg`: the register containing the `java.lang.reflect.Method`
/// - `id_method`: the expected [`IdMethod`].
/// - `abort_label`: the label where to jump if the method does not match `id_method`.
pub fn test_method(
method_obj_reg: u16,
id_method: IdMethod,
abort_label: String,
reg_inf: &mut RegistersInfo,
) -> Vec<Instruction> {
// Check for arg type
let mut insns = vec![
Instruction::InvokeVirtual {
method: MTH_GET_PARAMS_TY.clone(),
args: vec![method_obj_reg],
},
Instruction::MoveResultObject { to: reg_inf.array },
// First check the number of args
Instruction::ArrayLength {
dest: reg_inf.array_index,
arr: reg_inf.array,
},
Instruction::Const {
reg: reg_inf.array_val,
lit: id_method.proto.get_parameters().len() as i32,
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
},
];
// then the type of each arg
for (i, param) in id_method.proto.get_parameters().into_iter().enumerate() {
insns.push(Instruction::Const {
reg: reg_inf.array_index,
lit: i as i32,
});
insns.push(Instruction::AGetObject {
dest: reg_inf.array_val,
arr: reg_inf.array,
idx: reg_inf.array_index,
});
insns.push(Instruction::ConstClass {
reg: reg_inf.array_index, // wrong name, but available for tmp val
lit: param,
});
insns.push(Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
})
}
insns.append(&mut vec![
// Check the runtime method is the right one
// Check Name
Instruction::InvokeVirtual {
method: MTH_GET_NAME.clone(),
args: vec![method_obj_reg],
},
Instruction::MoveResultObject {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::ConstString {
reg: reg_inf.array_val, // wrong name, but available for tmp val
lit: id_method.name.clone(),
},
Instruction::InvokeVirtual {
method: STR_EQ.clone(),
args: vec![reg_inf.array_index as u16, reg_inf.array_val as u16],
},
Instruction::MoveResult {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::IfEqZ {
a: reg_inf.array_index,
label: abort_label.clone(),
},
// Check Return Type
Instruction::InvokeVirtual {
method: MTH_GET_RET_TY.clone(),
args: vec![method_obj_reg],
},
Instruction::MoveResultObject {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::ConstClass {
reg: reg_inf.array_val, // wrong name, but available for tmp val
lit: id_method.proto.get_return_type(),
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
},
// Check Declaring Type
Instruction::InvokeVirtual {
method: MTH_GET_DEC_CLS.clone(),
args: vec![method_obj_reg],
},
Instruction::MoveResultObject {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::ConstClass {
reg: reg_inf.array_val, // wrong name, but available for tmp val
lit: id_method.class_.clone(),
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
},
]);
insns
}
fn get_cnstr_new_inst_block(
ref_data: &ReflectionCnstrNewInstData,
invoke_arg: &[u16],
reg_inf: &mut RegistersInfo,
end_label: &str,
move_result: Option<Instruction>,
) -> Result<Vec<Instruction>> {
let (cnst_reg, arg_arr) = if let &[a, b] = invoke_arg {
(a, b)
} else {
bail!(
"Method;->invoke arg should have exactrly 2 arguments, found {}",
invoke_arg.len()
);
};
if cnst_reg > u8::MAX as u16 {
// TODO
bail!("Cannot transform instantiation calls to a class stored in 16 bits register");
}
if reg_inf.first_arg > u8::MAX as u16 {
// TODO
bail!("Cannot transform instantiation calls to a class with first argument register greater than 255.");
}
//let cnst_reg = cnst_reg as u8;
let nb_args = ref_data.constructor.proto.get_parameters().len();
if reg_inf.nb_arg_reg < nb_args as u16 + 1 {
reg_inf.nb_arg_reg = nb_args as u16 + 1;
}
let abort_label = format!(
"end_static_instance_with_{}_at_{}",
ref_data.constructor.try_to_smali()?,
"TODO_ADDR"
);
let mut insns = test_cnstr(
cnst_reg,
ref_data.constructor.clone(),
abort_label.clone(),
reg_inf,
);
for (i, param) in ref_data
.constructor
.proto
.get_parameters()
.iter()
.enumerate()
{
insns.push(Instruction::Const {
reg: reg_inf.array_index,
lit: i as i32,
});
insns.push(Instruction::AGetObject {
dest: reg_inf.array_val,
arr: arg_arr as u8, // TODO
idx: reg_inf.array_index,
});
insns.push(Instruction::CheckCast {
reg: reg_inf.array_val,
lit: param.clone(),
});
insns.push(Instruction::MoveObject {
from: reg_inf.array_val as u16,
to: reg_inf.first_arg + i as u16 + 1,
});
}
insns.push(Instruction::NewInstance {
reg: reg_inf.first_arg as u8,
lit: ref_data.constructor.class_.clone(),
});
insns.push(Instruction::InvokeDirect {
method: ref_data.constructor.clone(),
args: (reg_inf.first_arg..reg_inf.first_arg + nb_args as u16 + 1).collect(),
});
if let Some(Instruction::MoveResultObject { to }) = move_result {
insns.push(Instruction::MoveObject {
from: reg_inf.first_arg,
to: to as u16,
});
}
insns.push(Instruction::Goto {
label: end_label.to_string(),
});
insns.push(Instruction::Label { name: abort_label });
Ok(insns)
}
/// Generate bytecode that test if a `java.lang.reflect.Constructor` is equal to an [`IdMethod`]
///
/// - `method_obj_reg`: the register containing the `java.lang.reflect.Method`
/// - `id_method`: the expected [`IdMethod`].
/// - `abort_label`: the label where to jump if the method does not match `id_method`.
pub fn test_cnstr(
cnst_reg: u16,
id_method: IdMethod,
abort_label: String,
reg_inf: &mut RegistersInfo,
) -> Vec<Instruction> {
// Check for arg type
let mut insns = vec![
Instruction::InvokeVirtual {
method: CNSTR_GET_PARAMS_TY.clone(),
args: vec![cnst_reg],
},
Instruction::MoveResultObject { to: reg_inf.array },
// First check the number of args
Instruction::ArrayLength {
dest: reg_inf.array_index,
arr: reg_inf.array,
},
Instruction::Const {
reg: reg_inf.array_val,
lit: id_method.proto.get_parameters().len() as i32,
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
},
];
// then the type of each arg
for (i, param) in id_method.proto.get_parameters().into_iter().enumerate() {
insns.push(Instruction::Const {
reg: reg_inf.array_index,
lit: i as i32,
});
insns.push(Instruction::AGetObject {
dest: reg_inf.array_val,
arr: reg_inf.array,
idx: reg_inf.array_index,
});
insns.push(Instruction::ConstClass {
reg: reg_inf.array_index, // wrong name, but available for tmp val
lit: param,
});
insns.push(Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
})
}
insns.append(&mut vec![
// Check Declaring Type
Instruction::InvokeVirtual {
method: CNSTR_GET_DEC_CLS.clone(),
args: vec![cnst_reg],
},
Instruction::MoveResultObject {
to: reg_inf.array_index, // wrong name, but available for tmp val
},
Instruction::ConstClass {
reg: reg_inf.array_val, // wrong name, but available for tmp val
lit: id_method.class_.clone(),
},
Instruction::IfNe {
a: reg_inf.array_index,
b: reg_inf.array_val,
label: abort_label.clone(),
},
]);
insns
}
fn get_class_new_inst_block(
ref_data: &ReflectionClassNewInstData,
invoke_arg: &[u16],
reg_inf: &mut RegistersInfo,
end_label: &str,
move_result: Option<Instruction>,
) -> Result<Vec<Instruction>> {
let class_reg = if let &[a] = invoke_arg {
a
} else {
bail!(
"Method;->invoke arg should have exactrly 3 arguments, found {}",
invoke_arg.len()
);
};
if !ref_data.constructor.proto.get_parameters().is_empty() {
bail!(
"Class.newInstance can only initialize instance with zero args constructor, found {}",
ref_data.constructor.__str__()
);
}
if class_reg > u8::MAX as u16 {
// TODO
bail!("Cannot transform instantiation calls to a class stored in 16 bits register");
}
let class_reg = class_reg as u8;
let abort_label = format!(
"end_static_instance_with_{}_at_{}",
ref_data.constructor.try_to_smali()?,
"TODO_ADDR"
);
let obj_reg = match move_result {
Some(Instruction::MoveResultObject { to }) => to,
_ => reg_inf.array_index,
};
Ok(vec![
Instruction::ConstClass {
reg: reg_inf.array_index, // wrong name, but available for tmp val
lit: ref_data.constructor.class_.clone(),
},
Instruction::IfNe {
a: reg_inf.array_index,
b: class_reg,
label: abort_label.clone(),
},
Instruction::NewInstance {
reg: obj_reg,
lit: ref_data.constructor.class_.clone(),
},
Instruction::InvokeDirect {
method: ref_data.constructor.clone(),
args: vec![obj_reg as u16],
},
Instruction::Goto {
label: end_label.to_string(),
},
Instruction::Label { name: abort_label },
])
}

View file

@ -5,7 +5,9 @@ use std::path::PathBuf;
use androscalpel::{IdMethod, IdType}; use androscalpel::{IdMethod, IdType};
use patcher::get_apk::{get_apk, ApkLocation}; use patcher::get_apk::{get_apk, ApkLocation};
use patcher::{transform_method, ReflectionData}; use patcher::{
transform_method, ReflectionClassNewInstData, ReflectionCnstrNewInstData, ReflectionInvokeData,
};
use clap::Parser; use clap::Parser;
@ -36,25 +38,38 @@ fn main() {
.unwrap(); .unwrap();
//println!("{:#?}", class.direct_methods.keys()); //println!("{:#?}", class.direct_methods.keys());
//println!("{:#?}", class.virtual_methods.keys()); //println!("{:#?}", class.virtual_methods.keys());
let method = class for m in [
.virtual_methods "Lcom/example/theseus/reflection/MainActivity;->callVirtualMethodReflectCall()V",
.get_mut( "Lcom/example/theseus/reflection/MainActivity;->callConstructorVirtualMethodReflectConstr()V",
&IdMethod::from_smali( "Lcom/example/theseus/reflection/MainActivity;->callVirtualMethodReflectOldConst()V",
"Lcom/example/theseus/reflection/MainActivity;->callVirtualMethodReflectCall()V", ] {
) let method = class
.unwrap(), .virtual_methods
) .get_mut(&IdMethod::from_smali(m).unwrap())
.unwrap(); .unwrap();
transform_method( transform_method(
method, method,
&ReflectionData { &ReflectionInvokeData {
method: IdMethod::from_smali( method: IdMethod::from_smali(
"Lcom/example/theseus/reflection/Reflectee;->transfer(Ljava/lang/String;)Ljava/lang/String;", "Lcom/example/theseus/reflection/Reflectee;->transfer(Ljava/lang/String;)Ljava/lang/String;",
) )
.unwrap(), .unwrap(),
}, },
&ReflectionClassNewInstData {
constructor: IdMethod::from_smali(
"Lcom/example/theseus/reflection/Reflectee;-><init>()V",
)
.unwrap(),
},
&ReflectionCnstrNewInstData{
constructor: IdMethod::from_smali(
"Lcom/example/theseus/reflection/Reflectee;-><init>(Ljava/lang/String;)V",
)
.unwrap(),
},
) )
.unwrap(); .unwrap();
}
let mut dex_files = vec![]; let mut dex_files = vec![];
let mut files = apk.gen_raw_dex().unwrap(); let mut files = apk.gen_raw_dex().unwrap();
let mut i = 0; let mut i = 0;

View file

@ -65,10 +65,11 @@ public class MainActivity extends Activity {
Class[] params = mth.getParameterTypes(); Class[] params = mth.getParameterTypes();
if ( if (
mth.getName().equals("transfer") && mth.getName().equals("transfer") &&
ret == String.class && mth.getReturnType() == String.class &&
dec == Reflectee.class && mth.getDeclaringClass() == Reflectee.class &&
params.length == 1 && params.length == 1 &&
params[0] == String.class { params[0] == String.class
) {
Log.e("[TEST]", "OK"); Log.e("[TEST]", "OK");
} }
String newData = (String) mth.invoke(r, data); String newData = (String) mth.invoke(r, data);
@ -88,7 +89,13 @@ public class MainActivity extends Activity {
ClassLoader cl = MainActivity.class.getClassLoader(); ClassLoader cl = MainActivity.class.getClassLoader();
Class clz = cl.loadClass("com.example.theseus.reflection.Reflectee"); Class clz = cl.loadClass("com.example.theseus.reflection.Reflectee");
Constructor cst = clz.getDeclaredConstructor(String.class); Constructor cst = clz.getDeclaredConstructor(String.class);
Object r = cst.newInstance(data); Object r;
Class[] args = cst.getParameterTypes();
if (args.length == 1 && args[0] == String.class && cst.getDeclaringClass() == Reflectee.class) {
r = new Reflectee(data);
} else {
r = cst.newInstance(data);
}
Method mth = clz.getMethod("transfer", String.class); Method mth = clz.getMethod("transfer", String.class);
String newData = (String) mth.invoke(r, ""); String newData = (String) mth.invoke(r, "");
Utils.sink(this, newData); Utils.sink(this, newData);
@ -106,7 +113,14 @@ public class MainActivity extends Activity {
String data = Utils.source("no reflect constr"); String data = Utils.source("no reflect constr");
ClassLoader cl = MainActivity.class.getClassLoader(); ClassLoader cl = MainActivity.class.getClassLoader();
Class clz = cl.loadClass("com.example.theseus.reflection.Reflectee"); Class clz = cl.loadClass("com.example.theseus.reflection.Reflectee");
Object r = clz.newInstance(); Object r;
if (
clz == Reflectee.class
) {
r = new Reflectee();
} else {
r = clz.newInstance();
}
Method mth = clz.getMethod("transfer", String.class); Method mth = clz.getMethod("transfer", String.class);
String newData = (String) mth.invoke(r, data); String newData = (String) mth.invoke(r, data);
Utils.sink(this, newData); Utils.sink(this, newData);