use binaryninjacore_sys::*;
use crate::{
architecture::{Architecture, CoreArchitecture},
binaryview::{BinaryView, BinaryViewExt},
callingconvention::CallingConvention,
filemetadata::FileMetadata,
function::Function,
mlil::MediumLevelILFunction,
rc::*,
string::{raw_to_string, BnStrCompatible, BnString},
symbol::Symbol,
};
use lazy_static::lazy_static;
use std::ptr::null_mut;
use std::{
borrow::{Borrow, Cow},
collections::{HashMap, HashSet},
ffi::CStr,
fmt::{self, Debug, Display, Formatter},
hash::{Hash, Hasher},
iter::{zip, IntoIterator},
mem::{self, ManuallyDrop},
ops::Range,
os::raw::c_char,
ptr, result, slice,
sync::Mutex,
};
pub type Result<R> = result::Result<R, ()>;
pub type ReferenceType = BNReferenceType;
pub type TypeClass = BNTypeClass;
pub type NamedTypeReferenceClass = BNNamedTypeReferenceClass;
pub type MemberAccess = BNMemberAccess;
pub type MemberScope = BNMemberScope;
pub type ILBranchDependence = BNILBranchDependence;
pub type DataFlowQueryOption = BNDataFlowQueryOption;
pub type VariableSourceType = BNVariableSourceType;
pub struct Conf<T> {
pub contents: T,
pub confidence: u8,
}
pub trait ConfMergable<T, O> {
type Result;
fn merge(self, other: O) -> Self::Result;
}
impl<T> Conf<T> {
pub fn new(contents: T, confidence: u8) -> Self {
Self {
contents,
confidence,
}
}
pub fn map<U, F>(self, f: F) -> Conf<U>
where
F: FnOnce(T) -> U,
{
Conf::new(f(self.contents), self.confidence)
}
pub fn as_ref<U>(&self) -> Conf<&U>
where
T: AsRef<U>,
{
Conf::new(self.contents.as_ref(), self.confidence)
}
}
impl<T> ConfMergable<T, Conf<T>> for Conf<T> {
type Result = Conf<T>;
fn merge(self, other: Conf<T>) -> Conf<T> {
if other.confidence > self.confidence {
other
} else {
self
}
}
}
impl<T> ConfMergable<T, Option<Conf<T>>> for Conf<T> {
type Result = Conf<T>;
fn merge(self, other: Option<Conf<T>>) -> Conf<T> {
match other {
Some(c @ Conf { confidence, .. }) if confidence > self.confidence => c,
_ => self,
}
}
}
impl<T> ConfMergable<T, Conf<T>> for Option<Conf<T>> {
type Result = Conf<T>;
fn merge(self, other: Conf<T>) -> Conf<T> {
match self {
Some(c @ Conf { confidence, .. }) if confidence >= other.confidence => c,
_ => other,
}
}
}
impl<T> ConfMergable<T, Option<Conf<T>>> for Option<Conf<T>> {
type Result = Option<Conf<T>>;
fn merge(self, other: Option<Conf<T>>) -> Option<Conf<T>> {
match (self, other) {
(
Some(
this @ Conf {
confidence: this_confidence,
..
},
),
Some(
other @ Conf {
confidence: other_confidence,
..
},
),
) => {
if this_confidence >= other_confidence {
Some(this)
} else {
Some(other)
}
}
(None, Some(c)) => Some(c),
(Some(c), None) => Some(c),
(None, None) => None,
}
}
}
impl<T: Debug> Debug for Conf<T> {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{:?} ({} confidence)", self.contents, self.confidence)
}
}
impl<T: Display> Display for Conf<T> {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{} ({} confidence)", self.contents, self.confidence)
}
}
impl<T: PartialEq> PartialEq for Conf<T> {
fn eq(&self, other: &Self) -> bool {
self.contents.eq(&other.contents)
}
}
impl<T: Eq> Eq for Conf<T> {}
impl<T: Hash> Hash for Conf<T> {
fn hash<H: Hasher>(&self, state: &mut H) {
self.contents.hash(state);
}
}
impl<'a, T> From<&'a Conf<T>> for Conf<&'a T> {
fn from(c: &'a Conf<T>) -> Self {
Conf::new(&c.contents, c.confidence)
}
}
impl<'a, T: RefCountable> From<&'a Conf<Ref<T>>> for Conf<&'a T> {
fn from(c: &'a Conf<Ref<T>>) -> Self {
Conf::new(c.contents.as_ref(), c.confidence)
}
}
impl<'a, T: RefCountable> From<&'a Ref<T>> for Conf<&'a T> {
fn from(r: &'a Ref<T>) -> Self {
r.as_ref().into()
}
}
#[inline]
pub fn min_confidence() -> u8 {
u8::MIN
}
#[inline]
pub fn max_confidence() -> u8 {
u8::MAX
}
impl<T: Clone> Clone for Conf<T> {
fn clone(&self) -> Self {
Self {
contents: self.contents.clone(),
confidence: self.confidence,
}
}
}
impl<T: Copy> Copy for Conf<T> {}
impl<T> From<T> for Conf<T> {
fn from(contents: T) -> Self {
Self::new(contents, max_confidence())
}
}
impl From<BNTypeWithConfidence> for Conf<Ref<Type>> {
fn from(type_with_confidence: BNTypeWithConfidence) -> Self {
Self::new(
unsafe { Type::ref_from_raw(type_with_confidence.type_) },
type_with_confidence.confidence,
)
}
}
impl From<BNBoolWithConfidence> for Conf<bool> {
fn from(bool_with_confidence: BNBoolWithConfidence) -> Self {
Self::new(bool_with_confidence.value, bool_with_confidence.confidence)
}
}
impl From<BNCallingConventionWithConfidence> for Conf<Ref<CallingConvention<CoreArchitecture>>> {
fn from(cc_with_confidence: BNCallingConventionWithConfidence) -> Self {
Self::new(
unsafe {
CallingConvention::ref_from_raw(
cc_with_confidence.convention,
CoreArchitecture::from_raw(BNGetCallingConventionArchitecture(
cc_with_confidence.convention,
)),
)
},
cc_with_confidence.confidence,
)
}
}
impl From<BNOffsetWithConfidence> for Conf<i64> {
fn from(offset_with_confidence: BNOffsetWithConfidence) -> Self {
Self::new(
offset_with_confidence.value,
offset_with_confidence.confidence,
)
}
}
impl From<Conf<&Type>> for BNTypeWithConfidence {
fn from(conf: Conf<&Type>) -> Self {
Self {
type_: conf.contents.handle,
confidence: conf.confidence,
}
}
}
impl From<Conf<bool>> for BNBoolWithConfidence {
fn from(conf: Conf<bool>) -> Self {
Self {
value: conf.contents,
confidence: conf.confidence,
}
}
}
impl<A: Architecture> From<Conf<&CallingConvention<A>>> for BNCallingConventionWithConfidence {
fn from(conf: Conf<&CallingConvention<A>>) -> Self {
Self {
convention: conf.contents.handle,
confidence: conf.confidence,
}
}
}
impl From<Conf<i64>> for BNOffsetWithConfidence {
fn from(conf: Conf<i64>) -> Self {
Self {
value: conf.contents,
confidence: conf.confidence,
}
}
}
#[derive(PartialEq, Eq, Hash)]
pub struct TypeBuilder {
pub(crate) handle: *mut BNTypeBuilder,
}
impl TypeBuilder {
pub fn new(t: &Type) -> Self {
unsafe { Self::from_raw(BNCreateTypeBuilderFromType(t.handle)) }
}
pub(crate) unsafe fn from_raw(handle: *mut BNTypeBuilder) -> Self {
debug_assert!(!handle.is_null());
Self { handle }
}
pub fn finalize(&self) -> Ref<Type> {
unsafe { Type::ref_from_raw(BNFinalizeTypeBuilder(self.handle)) }
}
pub fn set_can_return<T: Into<Conf<bool>>>(&self, value: T) -> &Self {
let mut bool_with_confidence = value.into().into();
unsafe { BNSetFunctionTypeBuilderCanReturn(self.handle, &mut bool_with_confidence) };
self
}
pub fn set_pure<T: Into<Conf<bool>>>(&self, value: T) -> &Self {
let mut bool_with_confidence = value.into().into();
unsafe { BNSetTypeBuilderPure(self.handle, &mut bool_with_confidence) };
self
}
pub fn set_const<T: Into<Conf<bool>>>(&self, value: T) -> &Self {
let mut bool_with_confidence = value.into().into();
unsafe { BNTypeBuilderSetConst(self.handle, &mut bool_with_confidence) };
self
}
pub fn set_volatile<T: Into<Conf<bool>>>(&self, value: T) -> &Self {
let mut bool_with_confidence = value.into().into();
unsafe { BNTypeBuilderSetVolatile(self.handle, &mut bool_with_confidence) };
self
}
pub fn type_class(&self) -> TypeClass {
unsafe { BNGetTypeBuilderClass(self.handle) }
}
pub fn width(&self) -> u64 {
unsafe { BNGetTypeBuilderWidth(self.handle) }
}
pub fn alignment(&self) -> usize {
unsafe { BNGetTypeBuilderAlignment(self.handle) }
}
pub fn is_signed(&self) -> Conf<bool> {
unsafe { BNIsTypeBuilderSigned(self.handle).into() }
}
pub fn is_const(&self) -> Conf<bool> {
unsafe { BNIsTypeBuilderConst(self.handle).into() }
}
pub fn is_volatile(&self) -> Conf<bool> {
unsafe { BNIsTypeBuilderVolatile(self.handle).into() }
}
pub fn is_floating_point(&self) -> bool {
unsafe { BNIsTypeBuilderFloatingPoint(self.handle) }
}
pub fn target(&self) -> Result<Conf<Ref<Type>>> {
let raw_target = unsafe { BNGetTypeBuilderChildType(self.handle) };
if raw_target.type_.is_null() {
Err(())
} else {
Ok(raw_target.into())
}
}
pub fn element_type(&self) -> Result<Conf<Ref<Type>>> {
let raw_target = unsafe { BNGetTypeBuilderChildType(self.handle) };
if raw_target.type_.is_null() {
Err(())
} else {
Ok(raw_target.into())
}
}
pub fn return_value(&self) -> Result<Conf<Ref<Type>>> {
let raw_target = unsafe { BNGetTypeBuilderChildType(self.handle) };
if raw_target.type_.is_null() {
Err(())
} else {
Ok(raw_target.into())
}
}
pub fn calling_convention(&self) -> Result<Conf<Ref<CallingConvention<CoreArchitecture>>>> {
let convention_confidence = unsafe { BNGetTypeBuilderCallingConvention(self.handle) };
if convention_confidence.convention.is_null() {
Err(())
} else {
Ok(convention_confidence.into())
}
}
pub fn parameters(&self) -> Result<Vec<FunctionParameter>> {
unsafe {
let mut count = 0;
let parameters_raw = BNGetTypeBuilderParameters(self.handle, &mut count);
if parameters_raw.is_null() {
Err(())
} else {
let parameters: &[BNFunctionParameter] =
slice::from_raw_parts(parameters_raw, count);
let result = (0..count)
.map(|i| FunctionParameter::from_raw(parameters[i]))
.collect();
BNFreeTypeParameterList(parameters_raw, count);
Ok(result)
}
}
}
pub fn has_variable_arguments(&self) -> Conf<bool> {
unsafe { BNTypeBuilderHasVariableArguments(self.handle).into() }
}
pub fn can_return(&self) -> Conf<bool> {
unsafe { BNFunctionTypeBuilderCanReturn(self.handle).into() }
}
pub fn pure(&self) -> Conf<bool> {
unsafe { BNIsTypeBuilderPure(self.handle).into() }
}
pub fn get_structure(&self) -> Result<Ref<Structure>> {
let result = unsafe { BNGetTypeBuilderStructure(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { Structure::ref_from_raw(result) })
}
}
pub fn get_enumeration(&self) -> Result<Ref<Enumeration>> {
let result = unsafe { BNGetTypeBuilderEnumeration(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { Enumeration::ref_from_raw(result) })
}
}
pub fn get_named_type_reference(&self) -> Result<Ref<NamedTypeReference>> {
let result = unsafe { BNGetTypeBuilderNamedTypeReference(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { NamedTypeReference::ref_from_raw(result) })
}
}
pub fn count(&self) -> u64 {
unsafe { BNGetTypeBuilderElementCount(self.handle) }
}
pub fn offset(&self) -> u64 {
unsafe { BNGetTypeBuilderOffset(self.handle) }
}
pub fn stack_adjustment(&self) -> Conf<i64> {
unsafe { BNGetTypeBuilderStackAdjustment(self.handle).into() }
}
pub fn void() -> Self {
unsafe { Self::from_raw(BNCreateVoidTypeBuilder()) }
}
pub fn bool() -> Self {
unsafe { Self::from_raw(BNCreateBoolTypeBuilder()) }
}
pub fn char() -> Self {
Self::int(1, true)
}
pub fn int(width: usize, is_signed: bool) -> Self {
let mut is_signed = Conf::new(is_signed, max_confidence()).into();
unsafe {
Self::from_raw(BNCreateIntegerTypeBuilder(
width,
&mut is_signed,
BnString::new("").as_ptr() as *mut _,
))
}
}
pub fn named_int<S: BnStrCompatible>(width: usize, is_signed: bool, alt_name: S) -> Self {
let mut is_signed = Conf::new(is_signed, max_confidence()).into();
let alt_name = alt_name.into_bytes_with_nul(); unsafe {
Self::from_raw(BNCreateIntegerTypeBuilder(
width,
&mut is_signed,
alt_name.as_ref().as_ptr() as _,
))
}
}
pub fn float(width: usize) -> Self {
unsafe {
Self::from_raw(BNCreateFloatTypeBuilder(
width,
BnString::new("").as_ptr() as *mut _,
))
}
}
pub fn named_float<S: BnStrCompatible>(width: usize, alt_name: S) -> Self {
let alt_name = alt_name.into_bytes_with_nul(); unsafe {
Self::from_raw(BNCreateFloatTypeBuilder(
width,
alt_name.as_ref().as_ptr() as _,
))
}
}
pub fn array<'a, T: Into<Conf<&'a Type>>>(t: T, count: u64) -> Self {
unsafe { Self::from_raw(BNCreateArrayTypeBuilder(&t.into().into(), count)) }
}
pub fn enumeration<T: Into<Conf<bool>>>(
enumeration: &Enumeration,
width: usize,
is_signed: T,
) -> Self {
unsafe {
let mut fake_arch: BNArchitecture = mem::zeroed();
Self::from_raw(BNCreateEnumerationTypeBuilder(
&mut fake_arch,
enumeration.handle,
width,
&mut is_signed.into().into(),
))
}
}
pub fn structure(structure_type: &Structure) -> Self {
unsafe { Self::from_raw(BNCreateStructureTypeBuilder(structure_type.handle)) }
}
pub fn named_type(type_reference: NamedTypeReference) -> Self {
let mut is_const = Conf::new(false, min_confidence()).into();
let mut is_volatile = Conf::new(false, min_confidence()).into();
unsafe {
Self::from_raw(BNCreateNamedTypeReferenceBuilder(
type_reference.handle,
0,
1,
&mut is_const,
&mut is_volatile,
))
}
}
pub fn named_type_from_type<S: BnStrCompatible>(name: S, t: &Type) -> Self {
let mut name = QualifiedName::from(name);
unsafe {
Self::from_raw(BNCreateNamedTypeReferenceBuilderFromTypeAndId(
BnString::new("").as_ptr() as *mut _,
&mut name.0,
t.handle,
))
}
}
pub fn pointer<'a, A: Architecture, T: Into<Conf<&'a Type>>>(arch: &A, t: T) -> Self {
let mut is_const = Conf::new(false, min_confidence()).into();
let mut is_volatile = Conf::new(false, min_confidence()).into();
unsafe {
Self::from_raw(BNCreatePointerTypeBuilder(
arch.as_ref().0,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ReferenceType::PointerReferenceType,
))
}
}
pub fn const_pointer<'a, A: Architecture, T: Into<Conf<&'a Type>>>(arch: &A, t: T) -> Self {
let mut is_const = Conf::new(true, max_confidence()).into();
let mut is_volatile = Conf::new(false, min_confidence()).into();
unsafe {
Self::from_raw(BNCreatePointerTypeBuilder(
arch.as_ref().0,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ReferenceType::PointerReferenceType,
))
}
}
pub fn pointer_of_width<'a, T: Into<Conf<&'a Type>>>(
t: T,
size: usize,
is_const: bool,
is_volatile: bool,
ref_type: Option<ReferenceType>,
) -> Self {
let mut is_const = Conf::new(is_const, max_confidence()).into();
let mut is_volatile = Conf::new(is_volatile, max_confidence()).into();
unsafe {
Self::from_raw(BNCreatePointerTypeBuilderOfWidth(
size,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ref_type.unwrap_or(ReferenceType::PointerReferenceType),
))
}
}
pub fn pointer_with_options<'a, A: Architecture, T: Into<Conf<&'a Type>>>(
arch: &A,
t: T,
is_const: bool,
is_volatile: bool,
ref_type: Option<ReferenceType>,
) -> Self {
let mut is_const = Conf::new(is_const, max_confidence()).into();
let mut is_volatile = Conf::new(is_volatile, max_confidence()).into();
unsafe {
Self::from_raw(BNCreatePointerTypeBuilder(
arch.as_ref().0,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ref_type.unwrap_or(ReferenceType::PointerReferenceType),
))
}
}
}
impl fmt::Display for TypeBuilder {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", unsafe {
BnString::from_raw(BNGetTypeBuilderString(self.handle, ptr::null_mut()))
})
}
}
impl Drop for TypeBuilder {
fn drop(&mut self) {
unsafe { BNFreeTypeBuilder(self.handle) };
}
}
#[repr(transparent)]
pub struct Type {
pub(crate) handle: *mut BNType,
}
impl Type {
unsafe fn from_raw(handle: *mut BNType) -> Self {
debug_assert!(!handle.is_null());
Self { handle }
}
pub(crate) unsafe fn ref_from_raw(handle: *mut BNType) -> Ref<Self> {
debug_assert!(!handle.is_null());
Ref::new(Self { handle })
}
pub fn to_builder(&self) -> TypeBuilder {
TypeBuilder::new(self)
}
pub fn type_class(&self) -> TypeClass {
unsafe { BNGetTypeClass(self.handle) }
}
pub fn width(&self) -> u64 {
unsafe { BNGetTypeWidth(self.handle) }
}
pub fn alignment(&self) -> usize {
unsafe { BNGetTypeAlignment(self.handle) }
}
pub fn is_signed(&self) -> Conf<bool> {
unsafe { BNIsTypeSigned(self.handle).into() }
}
pub fn is_const(&self) -> Conf<bool> {
unsafe { BNIsTypeConst(self.handle).into() }
}
pub fn is_volatile(&self) -> Conf<bool> {
unsafe { BNIsTypeVolatile(self.handle).into() }
}
pub fn is_floating_point(&self) -> bool {
unsafe { BNIsTypeFloatingPoint(self.handle) }
}
pub fn target(&self) -> Result<Conf<Ref<Type>>> {
let raw_target = unsafe { BNGetChildType(self.handle) };
if raw_target.type_.is_null() {
Err(())
} else {
Ok(raw_target.into())
}
}
pub fn element_type(&self) -> Result<Conf<Ref<Type>>> {
let raw_target = unsafe { BNGetChildType(self.handle) };
if raw_target.type_.is_null() {
Err(())
} else {
Ok(raw_target.into())
}
}
pub fn return_value(&self) -> Result<Conf<Ref<Type>>> {
let raw_target = unsafe { BNGetChildType(self.handle) };
if raw_target.type_.is_null() {
Err(())
} else {
Ok(raw_target.into())
}
}
pub fn calling_convention(&self) -> Result<Conf<Ref<CallingConvention<CoreArchitecture>>>> {
let convention_confidence = unsafe { BNGetTypeCallingConvention(self.handle) };
if convention_confidence.convention.is_null() {
Err(())
} else {
Ok(convention_confidence.into())
}
}
pub fn parameters(&self) -> Result<Vec<FunctionParameter>> {
unsafe {
let mut count = 0;
let parameters_raw: *mut BNFunctionParameter =
BNGetTypeParameters(self.handle, &mut count);
if parameters_raw.is_null() {
Err(())
} else {
let parameters: &[BNFunctionParameter] =
slice::from_raw_parts(parameters_raw, count);
let result = (0..count)
.map(|i| FunctionParameter::from_raw(parameters[i]))
.collect();
BNFreeTypeParameterList(parameters_raw, count);
Ok(result)
}
}
}
pub fn has_variable_arguments(&self) -> Conf<bool> {
unsafe { BNTypeHasVariableArguments(self.handle).into() }
}
pub fn can_return(&self) -> Conf<bool> {
unsafe { BNFunctionTypeCanReturn(self.handle).into() }
}
pub fn pure(&self) -> Conf<bool> {
unsafe { BNIsTypePure(self.handle).into() }
}
pub fn get_structure(&self) -> Result<Ref<Structure>> {
let result = unsafe { BNGetTypeStructure(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { Structure::ref_from_raw(result) })
}
}
pub fn get_enumeration(&self) -> Result<Ref<Enumeration>> {
let result = unsafe { BNGetTypeEnumeration(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { Enumeration::ref_from_raw(result) })
}
}
pub fn get_named_type_reference(&self) -> Result<Ref<NamedTypeReference>> {
let result = unsafe { BNGetTypeNamedTypeReference(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { NamedTypeReference::ref_from_raw(result) })
}
}
pub fn count(&self) -> u64 {
unsafe { BNGetTypeElementCount(self.handle) }
}
pub fn offset(&self) -> u64 {
unsafe { BNGetTypeOffset(self.handle) }
}
pub fn stack_adjustment(&self) -> Conf<i64> {
unsafe { BNGetTypeStackAdjustment(self.handle).into() }
}
pub fn registered_name(&self) -> Result<Ref<NamedTypeReference>> {
let result = unsafe { BNGetRegisteredTypeName(self.handle) };
if result.is_null() {
Err(())
} else {
Ok(unsafe { NamedTypeReference::ref_from_raw(result) })
}
}
pub fn void() -> Ref<Self> {
unsafe { Self::ref_from_raw(BNCreateVoidType()) }
}
pub fn bool() -> Ref<Self> {
unsafe { Self::ref_from_raw(BNCreateBoolType()) }
}
pub fn char() -> Ref<Self> {
Self::int(1, true)
}
pub fn wide_char(width: usize) -> Ref<Self> {
unsafe {
Self::ref_from_raw(BNCreateWideCharType(
width,
BnString::new("").as_ptr() as *mut _,
))
}
}
pub fn int(width: usize, is_signed: bool) -> Ref<Self> {
let mut is_signed = Conf::new(is_signed, max_confidence()).into();
unsafe {
Self::ref_from_raw(BNCreateIntegerType(
width,
&mut is_signed,
BnString::new("").as_ptr() as *mut _,
))
}
}
pub fn named_int<S: BnStrCompatible>(width: usize, is_signed: bool, alt_name: S) -> Ref<Self> {
let mut is_signed = Conf::new(is_signed, max_confidence()).into();
let alt_name = alt_name.into_bytes_with_nul(); unsafe {
Self::ref_from_raw(BNCreateIntegerType(
width,
&mut is_signed,
alt_name.as_ref().as_ptr() as _,
))
}
}
pub fn float(width: usize) -> Ref<Self> {
unsafe {
Self::ref_from_raw(BNCreateFloatType(
width,
BnString::new("").as_ptr() as *mut _,
))
}
}
pub fn named_float<S: BnStrCompatible>(width: usize, alt_name: S) -> Ref<Self> {
let alt_name = alt_name.into_bytes_with_nul(); unsafe { Self::ref_from_raw(BNCreateFloatType(width, alt_name.as_ref().as_ptr() as _)) }
}
pub fn array<'a, T: Into<Conf<&'a Type>>>(t: T, count: u64) -> Ref<Self> {
unsafe { Self::ref_from_raw(BNCreateArrayType(&t.into().into(), count)) }
}
pub fn enumeration<T: Into<Conf<bool>>>(
enumeration: &Enumeration,
width: usize,
is_signed: T,
) -> Ref<Self> {
unsafe {
let mut fake_arch: BNArchitecture = mem::zeroed();
Self::ref_from_raw(BNCreateEnumerationType(
&mut fake_arch,
enumeration.handle,
width,
&mut is_signed.into().into(),
))
}
}
pub fn structure(structure: &Structure) -> Ref<Self> {
unsafe { Self::ref_from_raw(BNCreateStructureType(structure.handle)) }
}
pub fn named_type(type_reference: &NamedTypeReference) -> Ref<Self> {
let mut is_const = Conf::new(false, min_confidence()).into();
let mut is_volatile = Conf::new(false, min_confidence()).into();
unsafe {
Self::ref_from_raw(BNCreateNamedTypeReference(
type_reference.handle,
0,
1,
&mut is_const,
&mut is_volatile,
))
}
}
pub fn named_type_from_type<S: BnStrCompatible>(name: S, t: &Type) -> Ref<Self> {
let mut name = QualifiedName::from(name);
unsafe {
Self::ref_from_raw(BNCreateNamedTypeReferenceFromTypeAndId(
BnString::new("").as_ptr() as *mut _,
&mut name.0,
t.handle,
))
}
}
pub fn function<'a, T: Into<Conf<&'a Type>>>(
return_type: T,
parameters: &[FunctionParameter],
variable_arguments: bool,
) -> Ref<Self> {
let mut return_type = return_type.into().into();
let mut variable_arguments = Conf::new(variable_arguments, max_confidence()).into();
let mut can_return = Conf::new(true, min_confidence()).into();
let mut pure = Conf::new(false, min_confidence()).into();
let mut raw_calling_convention: BNCallingConventionWithConfidence =
BNCallingConventionWithConfidence {
convention: ptr::null_mut(),
confidence: min_confidence(),
};
let mut stack_adjust = Conf::<i64>::new(0, min_confidence()).into();
let mut raw_parameters = Vec::<BNFunctionParameter>::with_capacity(parameters.len());
let mut parameter_name_references = Vec::with_capacity(parameters.len());
for parameter in parameters {
let raw_name = parameter.name.as_str().into_bytes_with_nul();
let location = match ¶meter.location {
Some(location) => location.raw(),
None => unsafe { mem::zeroed() },
};
raw_parameters.push(BNFunctionParameter {
name: raw_name.as_slice().as_ptr() as *mut _,
type_: parameter.t.contents.handle,
typeConfidence: parameter.t.confidence,
defaultLocation: parameter.location.is_none(),
location,
});
parameter_name_references.push(raw_name);
}
let reg_stack_adjust_regs = ptr::null_mut();
let reg_stack_adjust_values = ptr::null_mut();
let mut return_regs: BNRegisterSetWithConfidence = BNRegisterSetWithConfidence {
regs: ptr::null_mut(),
count: 0,
confidence: 0,
};
unsafe {
Self::ref_from_raw(BNNewTypeReference(BNCreateFunctionType(
&mut return_type,
&mut raw_calling_convention,
raw_parameters.as_mut_ptr(),
raw_parameters.len(),
&mut variable_arguments,
&mut can_return,
&mut stack_adjust,
reg_stack_adjust_regs,
reg_stack_adjust_values,
0,
&mut return_regs,
BNNameType::NoNameType,
&mut pure,
)))
}
}
pub fn function_with_options<
'a,
A: Architecture,
T: Into<Conf<&'a Type>>,
C: Into<Conf<&'a CallingConvention<A>>>,
>(
return_type: T,
parameters: &[FunctionParameter],
variable_arguments: bool,
calling_convention: C,
stack_adjust: Conf<i64>,
) -> Ref<Self> {
let mut return_type = return_type.into().into();
let mut variable_arguments = Conf::new(variable_arguments, max_confidence()).into();
let mut can_return = Conf::new(true, min_confidence()).into();
let mut pure = Conf::new(false, min_confidence()).into();
let mut raw_calling_convention: BNCallingConventionWithConfidence =
calling_convention.into().into();
let mut stack_adjust = stack_adjust.into();
let mut raw_parameters = Vec::<BNFunctionParameter>::with_capacity(parameters.len());
let mut parameter_name_references = Vec::with_capacity(parameters.len());
let mut name_ptrs = vec![];
for parameter in parameters {
name_ptrs.push(parameter.name.clone());
}
for (name, parameter) in zip(name_ptrs, parameters) {
let raw_name = name.as_str().into_bytes_with_nul();
let location = match ¶meter.location {
Some(location) => location.raw(),
None => unsafe { mem::zeroed() },
};
raw_parameters.push(BNFunctionParameter {
name: raw_name.as_slice().as_ptr() as *mut _,
type_: parameter.t.contents.handle,
typeConfidence: parameter.t.confidence,
defaultLocation: parameter.location.is_none(),
location,
});
parameter_name_references.push(raw_name);
}
let reg_stack_adjust_regs = ptr::null_mut();
let reg_stack_adjust_values = ptr::null_mut();
let mut return_regs: BNRegisterSetWithConfidence = BNRegisterSetWithConfidence {
regs: ptr::null_mut(),
count: 0,
confidence: 0,
};
unsafe {
Self::ref_from_raw(BNCreateFunctionType(
&mut return_type,
&mut raw_calling_convention,
raw_parameters.as_mut_ptr(),
raw_parameters.len(),
&mut variable_arguments,
&mut can_return,
&mut stack_adjust,
reg_stack_adjust_regs,
reg_stack_adjust_values,
0,
&mut return_regs,
BNNameType::NoNameType,
&mut pure,
))
}
}
pub fn pointer<'a, A: Architecture, T: Into<Conf<&'a Type>>>(arch: &A, t: T) -> Ref<Self> {
let mut is_const = Conf::new(false, min_confidence()).into();
let mut is_volatile = Conf::new(false, min_confidence()).into();
unsafe {
Self::ref_from_raw(BNCreatePointerType(
arch.as_ref().0,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ReferenceType::PointerReferenceType,
))
}
}
pub fn const_pointer<'a, A: Architecture, T: Into<Conf<&'a Type>>>(
arch: &A,
t: T,
) -> Ref<Self> {
let mut is_const = Conf::new(true, max_confidence()).into();
let mut is_volatile = Conf::new(false, min_confidence()).into();
unsafe {
Self::ref_from_raw(BNCreatePointerType(
arch.as_ref().0,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ReferenceType::PointerReferenceType,
))
}
}
pub fn pointer_of_width<'a, T: Into<Conf<&'a Type>>>(
t: T,
size: usize,
is_const: bool,
is_volatile: bool,
ref_type: Option<ReferenceType>,
) -> Ref<Self> {
let mut is_const = Conf::new(is_const, max_confidence()).into();
let mut is_volatile = Conf::new(is_volatile, max_confidence()).into();
unsafe {
Self::ref_from_raw(BNCreatePointerTypeOfWidth(
size,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ref_type.unwrap_or(ReferenceType::PointerReferenceType),
))
}
}
pub fn pointer_with_options<'a, A: Architecture, T: Into<Conf<&'a Type>>>(
arch: &A,
t: T,
is_const: bool,
is_volatile: bool,
ref_type: Option<ReferenceType>,
) -> Ref<Self> {
let mut is_const = Conf::new(is_const, max_confidence()).into();
let mut is_volatile = Conf::new(is_volatile, max_confidence()).into();
unsafe {
Self::ref_from_raw(BNCreatePointerType(
arch.as_ref().0,
&t.into().into(),
&mut is_const,
&mut is_volatile,
ref_type.unwrap_or(ReferenceType::PointerReferenceType),
))
}
}
pub fn generate_auto_demangled_type_id<S: BnStrCompatible>(name: S) -> BnString {
let mut name = QualifiedName::from(name);
unsafe { BnString::from_raw(BNGenerateAutoDemangledTypeId(&mut name.0)) }
}
}
impl fmt::Display for Type {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", unsafe {
BnString::from_raw(BNGetTypeString(
self.handle,
ptr::null_mut(),
BNTokenEscapingType::NoTokenEscapingType,
))
})
}
}
lazy_static! {
static ref TYPE_DEBUG_BV: Mutex<Option<Ref<BinaryView>>> =
Mutex::new(BinaryView::from_data(&FileMetadata::new(), &[]).ok());
}
impl fmt::Debug for Type {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if let Ok(lock) = TYPE_DEBUG_BV.lock() {
if let Some(bv) = &*lock {
let container = unsafe { BNGetAnalysisTypeContainer(bv.handle) };
let printer = if f.alternate() {
unsafe { BNGetTypePrinterByName(c"_DebugTypePrinter".as_ptr()) }
} else {
unsafe { BNGetTypePrinterByName(c"CoreTypePrinter".as_ptr()) }
};
if printer.is_null() {
return Err(fmt::Error);
}
let mut name = QualifiedName::from("");
let mut lines: *mut BNTypeDefinitionLine = null_mut();
let mut count: usize = 0;
unsafe {
BNGetTypePrinterTypeLines(
printer,
self.handle,
container,
&mut name.0,
64,
false,
BNTokenEscapingType::NoTokenEscapingType,
&mut lines,
&mut count,
)
};
unsafe {
BNFreeTypeContainer(container);
}
if lines.is_null() {
return Err(fmt::Error);
}
let line_slice: &[BNTypeDefinitionLine] =
unsafe { slice::from_raw_parts(lines, count) };
for (i, line) in line_slice.iter().enumerate() {
if i > 0 {
writeln!(f)?;
}
let tokens: &[BNInstructionTextToken] =
unsafe { slice::from_raw_parts(line.tokens, line.count) };
for token in tokens {
let text: *const c_char = token.text;
let str = unsafe { CStr::from_ptr(text) };
write!(f, "{}", str.to_string_lossy())?;
}
}
unsafe {
BNFreeTypeDefinitionLineList(lines, count);
}
return Ok(());
}
}
Err(fmt::Error)
}
}
impl PartialEq for Type {
fn eq(&self, other: &Self) -> bool {
unsafe { BNTypesEqual(self.handle, other.handle) }
}
}
impl Eq for Type {}
impl Hash for Type {
fn hash<H: Hasher>(&self, state: &mut H) {
self.handle.hash(state);
}
}
unsafe impl Send for Type {}
unsafe impl Sync for Type {}
unsafe impl RefCountable for Type {
unsafe fn inc_ref(handle: &Self) -> Ref<Self> {
Self::ref_from_raw(BNNewTypeReference(handle.handle))
}
unsafe fn dec_ref(handle: &Self) {
BNFreeType(handle.handle);
}
}
impl ToOwned for Type {
type Owned = Ref<Self>;
fn to_owned(&self) -> Self::Owned {
unsafe { RefCountable::inc_ref(self) }
}
}
pub struct ComponentReferencedTypes;
impl CoreArrayProvider for ComponentReferencedTypes {
type Raw = *mut BNType;
type Context = ();
type Wrapped<'a> = &'a Type;
}
unsafe impl CoreArrayProviderInner for ComponentReferencedTypes {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNComponentFreeReferencedTypes(raw, count)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
core::mem::transmute(raw)
}
}
#[derive(Clone, Debug)]
pub struct FunctionParameter {
pub t: Conf<Ref<Type>>,
pub name: String,
pub location: Option<Variable>,
}
impl FunctionParameter {
pub fn new<T: Into<Conf<Ref<Type>>>>(t: T, name: String, location: Option<Variable>) -> Self {
Self {
t: t.into(),
name,
location,
}
}
pub(crate) fn from_raw(member: BNFunctionParameter) -> Self {
let name = if member.name.is_null() {
if member.location.type_ == VariableSourceType::RegisterVariableSourceType {
format!("reg_{}", member.location.storage)
} else if member.location.type_ == VariableSourceType::StackVariableSourceType {
format!("arg_{}", member.location.storage)
} else {
String::new()
}
} else {
unsafe { CStr::from_ptr(member.name) }
.to_str()
.unwrap()
.to_owned()
};
Self {
t: Conf::new(
unsafe { Type::ref_from_raw(BNNewTypeReference(member.type_)) },
member.typeConfidence,
),
name,
location: if member.defaultLocation {
None
} else {
Some(unsafe { Variable::from_raw(member.location) })
},
}
}
}
#[derive(Clone, Copy, Debug, Hash, Eq, PartialEq)]
pub struct Variable {
pub t: VariableSourceType,
pub index: u32,
pub storage: i64,
}
impl Variable {
pub fn new(t: VariableSourceType, index: u32, storage: i64) -> Self {
Self { t, index, storage }
}
pub(crate) unsafe fn from_raw(var: BNVariable) -> Self {
Self {
t: var.type_,
index: var.index,
storage: var.storage,
}
}
pub(crate) unsafe fn from_identifier(var: u64) -> Self {
Self::from_raw(unsafe { BNFromVariableIdentifier(var) })
}
pub(crate) fn raw(&self) -> BNVariable {
BNVariable {
type_: self.t,
index: self.index,
storage: self.storage,
}
}
}
impl CoreArrayProvider for Variable {
type Raw = BNVariable;
type Context = ();
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for Variable {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeVariableList(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
Variable::from_raw(*raw)
}
}
impl CoreArrayProvider for (&str, Variable, &Type) {
type Raw = BNVariableNameAndType;
type Context = ();
type Wrapped<'a> = (&'a str, Variable, &'a Type) where Self: 'a;
}
unsafe impl CoreArrayProviderInner for (&str, Variable, &Type) {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeVariableNameAndTypeList(raw, count)
}
unsafe fn wrap_raw<'a>(
raw: &'a Self::Raw,
_context: &'a Self::Context,
) -> (&'a str, Variable, &'a Type) {
let name = CStr::from_ptr(raw.name).to_str().unwrap();
let var = Variable::from_raw(raw.var);
let var_type = core::mem::transmute(&raw.type_);
(name, var, var_type)
}
}
#[derive(Clone, Copy, Debug, Hash, Eq, PartialEq)]
pub struct SSAVariable {
pub variable: Variable,
pub version: usize,
}
impl SSAVariable {
pub fn new(variable: Variable, version: usize) -> Self {
Self { variable, version }
}
}
impl CoreArrayProvider for SSAVariable {
type Raw = usize;
type Context = Variable;
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for SSAVariable {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeILInstructionList(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, context: &'a Self::Context) -> Self::Wrapped<'a> {
SSAVariable::new(*context, *raw)
}
}
impl CoreArrayProvider for Array<SSAVariable> {
type Raw = BNVariable;
type Context = Ref<MediumLevelILFunction>;
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for Array<SSAVariable> {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeVariableList(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, context: &'a Self::Context) -> Self::Wrapped<'a> {
let mut count = 0;
let versions =
unsafe { BNGetMediumLevelILVariableSSAVersions(context.handle, raw, &mut count) };
Array::new(versions, count, Variable::from_raw(*raw))
}
}
#[derive(Clone, Debug, Hash, Eq, PartialEq)]
pub struct NamedTypedVariable {
pub name: String,
pub ty: Conf<Ref<Type>>,
pub var: Variable,
pub auto_defined: bool,
}
impl NamedTypedVariable {
pub fn new(var: Variable, name: String, ty: Conf<Ref<Type>>, auto_defined: bool) -> Self {
Self {
name,
ty,
var,
auto_defined,
}
}
pub(crate) unsafe fn from_raw(var: &BNVariableNameAndType) -> Self {
Self {
var: Variable::from_raw(var.var),
auto_defined: var.autoDefined,
name: CStr::from_ptr(var.name).to_str().unwrap().to_string(),
ty: Conf::new(Type::ref_from_raw(var.type_), var.typeConfidence),
}
}
pub fn name(&self) -> &str {
&self.name
}
pub fn var(&self) -> Variable {
self.var
}
pub fn auto_defined(&self) -> bool {
self.auto_defined
}
pub fn type_confidence(&self) -> u8 {
self.ty.confidence
}
pub fn var_type(&self) -> Ref<Type> {
self.ty.contents.clone()
}
}
impl CoreArrayProvider for NamedTypedVariable {
type Raw = BNVariableNameAndType;
type Context = ();
type Wrapped<'a> = Guard<'a, NamedTypedVariable>;
}
unsafe impl CoreArrayProviderInner for NamedTypedVariable {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeVariableNameAndTypeList(raw, count)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
unsafe { Guard::new(NamedTypedVariable::from_raw(raw), raw) }
}
}
#[derive(Debug, Clone)]
pub struct EnumerationMember {
pub name: String,
pub value: u64,
pub is_default: bool,
}
impl EnumerationMember {
pub fn new(name: String, value: u64, is_default: bool) -> Self {
Self {
name,
value,
is_default,
}
}
pub(crate) unsafe fn from_raw(member: BNEnumerationMember) -> Self {
Self {
name: raw_to_string(member.name).unwrap(),
value: member.value,
is_default: member.isDefault,
}
}
}
#[derive(PartialEq, Eq, Hash)]
pub struct EnumerationBuilder {
pub(crate) handle: *mut BNEnumerationBuilder,
}
impl EnumerationBuilder {
pub fn new() -> Self {
Self {
handle: unsafe { BNCreateEnumerationBuilder() },
}
}
pub(crate) unsafe fn from_raw(handle: *mut BNEnumerationBuilder) -> Self {
Self { handle }
}
pub fn finalize(&self) -> Ref<Enumeration> {
unsafe { Enumeration::ref_from_raw(BNFinalizeEnumerationBuilder(self.handle)) }
}
pub fn append<S: BnStrCompatible>(&self, name: S) -> &Self {
let name = name.into_bytes_with_nul();
unsafe {
BNAddEnumerationBuilderMember(self.handle, name.as_ref().as_ptr() as _);
}
self
}
pub fn insert<S: BnStrCompatible>(&self, name: S, value: u64) -> &Self {
let name = name.into_bytes_with_nul();
unsafe {
BNAddEnumerationBuilderMemberWithValue(self.handle, name.as_ref().as_ptr() as _, value);
}
self
}
pub fn replace<S: BnStrCompatible>(&self, id: usize, name: S, value: u64) -> &Self {
let name = name.into_bytes_with_nul();
unsafe {
BNReplaceEnumerationBuilderMember(self.handle, id, name.as_ref().as_ptr() as _, value);
}
self
}
pub fn remove(&self, id: usize) -> &Self {
unsafe {
BNRemoveEnumerationBuilderMember(self.handle, id);
}
self
}
pub fn members(&self) -> Vec<EnumerationMember> {
unsafe {
let mut count = 0;
let members_raw = BNGetEnumerationBuilderMembers(self.handle, &mut count);
let members: &[BNEnumerationMember] = slice::from_raw_parts(members_raw, count);
let result = (0..count)
.map(|i| EnumerationMember::from_raw(members[i]))
.collect();
BNFreeEnumerationMemberList(members_raw, count);
result
}
}
}
impl Default for EnumerationBuilder {
fn default() -> Self {
Self::new()
}
}
impl From<&Enumeration> for EnumerationBuilder {
fn from(enumeration: &Enumeration) -> Self {
unsafe {
Self::from_raw(BNCreateEnumerationBuilderFromEnumeration(
enumeration.handle,
))
}
}
}
impl Drop for EnumerationBuilder {
fn drop(&mut self) {
unsafe { BNFreeEnumerationBuilder(self.handle) };
}
}
#[derive(PartialEq, Eq, Hash)]
pub struct Enumeration {
pub(crate) handle: *mut BNEnumeration,
}
impl Enumeration {
pub(crate) unsafe fn ref_from_raw(handle: *mut BNEnumeration) -> Ref<Self> {
debug_assert!(!handle.is_null());
Ref::new(Self { handle })
}
pub fn builder() -> EnumerationBuilder {
EnumerationBuilder::new()
}
pub fn members(&self) -> Vec<EnumerationMember> {
unsafe {
let mut count = 0;
let members_raw = BNGetEnumerationMembers(self.handle, &mut count);
let members: &[BNEnumerationMember] = slice::from_raw_parts(members_raw, count);
let result = (0..count)
.map(|i| EnumerationMember::from_raw(members[i]))
.collect();
BNFreeEnumerationMemberList(members_raw, count);
result
}
}
}
unsafe impl RefCountable for Enumeration {
unsafe fn inc_ref(handle: &Self) -> Ref<Self> {
Self::ref_from_raw(BNNewEnumerationReference(handle.handle))
}
unsafe fn dec_ref(handle: &Self) {
BNFreeEnumeration(handle.handle);
}
}
impl ToOwned for Enumeration {
type Owned = Ref<Self>;
fn to_owned(&self) -> Self::Owned {
unsafe { RefCountable::inc_ref(self) }
}
}
pub type StructureType = BNStructureVariant;
#[derive(PartialEq, Eq, Hash)]
pub struct StructureBuilder {
pub(crate) handle: *mut BNStructureBuilder,
}
impl StructureBuilder {
pub fn new() -> Self {
Self {
handle: unsafe { BNCreateStructureBuilder() },
}
}
pub(crate) unsafe fn from_raw(handle: *mut BNStructureBuilder) -> Self {
debug_assert!(!handle.is_null());
Self { handle }
}
pub fn finalize(&self) -> Ref<Structure> {
unsafe { Structure::ref_from_raw(BNFinalizeStructureBuilder(self.handle)) }
}
pub fn set_width(&self, width: u64) -> &Self {
unsafe {
BNSetStructureBuilderWidth(self.handle, width);
}
self
}
pub fn set_alignment(&self, alignment: usize) -> &Self {
unsafe {
BNSetStructureBuilderAlignment(self.handle, alignment);
}
self
}
pub fn set_packed(&self, packed: bool) -> &Self {
unsafe {
BNSetStructureBuilderPacked(self.handle, packed);
}
self
}
pub fn set_structure_type(&self, t: StructureType) -> &Self {
unsafe { BNSetStructureBuilderType(self.handle, t) };
self
}
pub fn set_pointer_offset(&self, offset: i64) -> &Self {
unsafe { BNSetStructureBuilderPointerOffset(self.handle, offset) };
self
}
pub fn set_propagates_data_var_refs(&self, does: bool) -> &Self {
unsafe { BNSetStructureBuilderPropagatesDataVariableReferences(self.handle, does) };
self
}
pub fn set_base_structures(&self, bases: Vec<BaseStructure>) -> &Self {
let mut bases_api = vec![];
for base in &bases {
bases_api.push(BNBaseStructure {
type_: base.ty.handle,
offset: base.offset,
width: base.width,
});
}
unsafe {
BNSetBaseStructuresForStructureBuilder(
self.handle,
bases_api.as_mut_ptr(),
bases_api.len(),
)
};
self
}
pub fn append<'a, S: BnStrCompatible, T: Into<Conf<&'a Type>>>(
&self,
t: T,
name: S,
access: MemberAccess,
scope: MemberScope,
) -> &Self {
let name = name.into_bytes_with_nul();
unsafe {
BNAddStructureBuilderMember(
self.handle,
&t.into().into(),
name.as_ref().as_ptr() as _,
access,
scope,
);
}
self
}
pub fn insert_member(&self, member: &StructureMember, overwrite_existing: bool) -> &Self {
let ty = member.ty.clone();
self.insert(
ty.as_ref(),
member.name.clone(),
member.offset,
overwrite_existing,
member.access,
member.scope,
);
self
}
pub fn insert<'a, S: BnStrCompatible, T: Into<Conf<&'a Type>>>(
&self,
t: T,
name: S,
offset: u64,
overwrite_existing: bool,
access: MemberAccess,
scope: MemberScope,
) -> &Self {
let name = name.into_bytes_with_nul();
unsafe {
BNAddStructureBuilderMemberAtOffset(
self.handle,
&t.into().into(),
name.as_ref().as_ptr() as _,
offset,
overwrite_existing,
access,
scope,
);
}
self
}
pub fn with_members<'a, S: BnStrCompatible, T: Into<Conf<&'a Type>>>(
&self,
members: impl IntoIterator<Item = (T, S)>,
) -> &Self {
for (t, name) in members {
self.append(t, name, MemberAccess::NoAccess, MemberScope::NoScope);
}
self
}
pub fn width(&self) -> u64 {
unsafe { BNGetStructureBuilderWidth(self.handle) }
}
pub fn alignment(&self) -> usize {
unsafe { BNGetStructureBuilderAlignment(self.handle) }
}
pub fn packed(&self) -> bool {
unsafe { BNIsStructureBuilderPacked(self.handle) }
}
pub fn structure_type(&self) -> StructureType {
unsafe { BNGetStructureBuilderType(self.handle) }
}
pub fn pointer_offset(&self) -> i64 {
unsafe { BNGetStructureBuilderPointerOffset(self.handle) }
}
pub fn propagates_data_var_refs(&self) -> bool {
unsafe { BNStructureBuilderPropagatesDataVariableReferences(self.handle) }
}
pub fn base_structures(&self) -> Result<Vec<BaseStructure>> {
let mut count = 0usize;
let bases = unsafe { BNGetBaseStructuresForStructureBuilder(self.handle, &mut count) };
if bases.is_null() {
Err(())
} else {
let bases_slice = unsafe { slice::from_raw_parts_mut(bases, count) };
let result = bases_slice
.iter()
.map(|base| unsafe { BaseStructure::from_raw(*base) })
.collect::<Vec<_>>();
unsafe {
BNFreeBaseStructureList(bases, count);
}
Ok(result)
}
}
pub fn members(&self) -> Array<StructureMember> {
let mut count = 0;
let members_raw = unsafe { BNGetStructureBuilderMembers(self.handle, &mut count) };
unsafe { Array::new(members_raw, count, ()) }
}
pub fn index_by_name(&self, name: &str) -> Option<usize> {
self.members().iter().position(|member| member.name == name)
}
pub fn index_by_offset(&self, offset: u64) -> Option<usize> {
self.members()
.iter()
.position(|member| member.offset == offset)
}
pub fn clear_members(&self) {
let len = self.members().len();
for idx in (0..len).rev() {
self.remove(idx)
}
}
pub fn add_members<'a>(&self, members: impl IntoIterator<Item = &'a StructureMember>) {
for member in members {
self.append(&member.ty, &member.name, member.access, member.scope);
}
}
pub fn set_members<'a>(&self, members: impl IntoIterator<Item = &'a StructureMember>) {
self.clear_members();
self.add_members(members);
}
pub fn remove(&self, index: usize) {
unsafe { BNRemoveStructureBuilderMember(self.handle, index) }
}
pub fn replace(&self, index: usize, type_: Conf<&Type>, name: &str, overwrite: bool) {
let name = name.into_bytes_with_nul();
let name_ptr = name.as_ptr() as *const _;
let raw_type_ = BNTypeWithConfidence {
type_: type_.contents as *const Type as *mut _,
confidence: type_.confidence,
};
unsafe {
BNReplaceStructureBuilderMember(self.handle, index, &raw_type_, name_ptr, overwrite)
}
}
}
impl From<&Structure> for StructureBuilder {
fn from(structure: &Structure) -> StructureBuilder {
unsafe { Self::from_raw(BNCreateStructureBuilderFromStructure(structure.handle)) }
}
}
impl From<Vec<StructureMember>> for StructureBuilder {
fn from(members: Vec<StructureMember>) -> StructureBuilder {
let builder = StructureBuilder::new();
for m in members {
builder.insert_member(&m, false);
}
builder
}
}
impl Debug for StructureBuilder {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "StructureBuilder {{ ... }}")
}
}
impl Drop for StructureBuilder {
fn drop(&mut self) {
unsafe { BNFreeStructureBuilder(self.handle) };
}
}
impl Default for StructureBuilder {
fn default() -> Self {
Self::new()
}
}
#[derive(PartialEq, Eq, Hash)]
pub struct Structure {
pub(crate) handle: *mut BNStructure,
}
impl Structure {
unsafe fn from_raw(handle: *mut BNStructure) -> Self {
debug_assert!(!handle.is_null());
Self { handle }
}
pub(crate) unsafe fn ref_from_raw(handle: *mut BNStructure) -> Ref<Self> {
debug_assert!(!handle.is_null());
Ref::new(Self { handle })
}
pub fn builder() -> StructureBuilder {
StructureBuilder::new()
}
pub fn width(&self) -> u64 {
unsafe { BNGetStructureWidth(self.handle) }
}
pub fn structure_type(&self) -> StructureType {
unsafe { BNGetStructureType(self.handle) }
}
pub fn members(&self) -> Result<Vec<StructureMember>> {
unsafe {
let mut count = 0;
let members_raw: *mut BNStructureMember =
BNGetStructureMembers(self.handle, &mut count);
if members_raw.is_null() {
return Err(());
}
let members = slice::from_raw_parts(members_raw, count);
let result = (0..count)
.map(|i| StructureMember::from_raw(members[i]))
.collect();
BNFreeStructureMemberList(members_raw, count);
Ok(result)
}
}
pub fn base_structures(&self) -> Result<Vec<BaseStructure>> {
let mut count = 0usize;
let bases = unsafe { BNGetBaseStructuresForStructure(self.handle, &mut count) };
if bases.is_null() {
Err(())
} else {
let bases_slice = unsafe { slice::from_raw_parts_mut(bases, count) };
let result = bases_slice
.iter()
.map(|base| unsafe { BaseStructure::from_raw(*base) })
.collect::<Vec<_>>();
unsafe {
BNFreeBaseStructureList(bases, count);
}
Ok(result)
}
}
}
impl Debug for Structure {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "Structure {{")?;
if let Ok(members) = self.members() {
for member in members {
write!(f, " {:?}", member)?;
}
}
write!(f, "}}")
}
}
unsafe impl RefCountable for Structure {
unsafe fn inc_ref(handle: &Self) -> Ref<Self> {
Ref::new(Self::from_raw(BNNewStructureReference(handle.handle)))
}
unsafe fn dec_ref(handle: &Self) {
BNFreeStructure(handle.handle);
}
}
impl ToOwned for Structure {
type Owned = Ref<Self>;
fn to_owned(&self) -> Self::Owned {
unsafe { RefCountable::inc_ref(self) }
}
}
#[derive(Debug, Clone)]
pub struct StructureMember {
pub ty: Conf<Ref<Type>>,
pub name: String,
pub offset: u64,
pub access: MemberAccess,
pub scope: MemberScope,
}
impl StructureMember {
pub fn new(
ty: Conf<Ref<Type>>,
name: String,
offset: u64,
access: MemberAccess,
scope: MemberScope,
) -> Self {
Self {
ty,
name,
offset,
access,
scope,
}
}
pub(crate) unsafe fn from_raw(handle: BNStructureMember) -> Self {
Self {
ty: Conf::new(
RefCountable::inc_ref(&Type::from_raw(handle.type_)),
handle.typeConfidence,
),
name: CStr::from_ptr(handle.name).to_string_lossy().to_string(),
offset: handle.offset,
access: handle.access,
scope: handle.scope,
}
}
}
impl CoreArrayProvider for StructureMember {
type Raw = BNStructureMember;
type Context = ();
type Wrapped<'a> = Guard<'a, StructureMember>;
}
unsafe impl CoreArrayProviderInner for StructureMember {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeStructureMemberList(raw, count)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
Guard::new(StructureMember::from_raw(*raw), &())
}
}
#[derive(Debug, Clone)]
pub struct InheritedStructureMember {
pub base: Ref<NamedTypeReference>,
pub base_offset: u64,
pub member: StructureMember,
pub member_index: usize,
}
impl InheritedStructureMember {
pub fn new(
base: Ref<NamedTypeReference>,
base_offset: u64,
member: StructureMember,
member_index: usize,
) -> Self {
Self {
base,
base_offset,
member,
member_index,
}
}
}
#[derive(Debug, Clone)]
pub struct BaseStructure {
pub ty: Ref<NamedTypeReference>,
pub offset: u64,
pub width: u64,
}
impl BaseStructure {
pub fn new(ty: Ref<NamedTypeReference>, offset: u64, width: u64) -> Self {
Self { ty, offset, width }
}
pub(crate) unsafe fn from_raw(handle: BNBaseStructure) -> Self {
Self {
ty: RefCountable::inc_ref(&NamedTypeReference::from_raw(handle.type_)),
offset: handle.offset,
width: handle.width,
}
}
}
#[derive(PartialEq, Eq, Hash)]
pub struct NamedTypeReference {
pub(crate) handle: *mut BNNamedTypeReference,
}
impl NamedTypeReference {
pub(crate) unsafe fn from_raw(handle: *mut BNNamedTypeReference) -> Self {
debug_assert!(!handle.is_null());
Self { handle }
}
pub(crate) unsafe fn ref_from_raw(handle: *mut BNNamedTypeReference) -> Ref<Self> {
debug_assert!(!handle.is_null());
Ref::new(Self { handle })
}
pub fn new(type_class: NamedTypeReferenceClass, mut name: QualifiedName) -> Ref<Self> {
unsafe {
Self::ref_from_raw(BNCreateNamedType(type_class, ptr::null() as *const _, &mut name.0))
}
}
pub fn new_with_id<S: BnStrCompatible>(
type_class: NamedTypeReferenceClass,
type_id: S,
mut name: QualifiedName,
) -> Ref<Self> {
let type_id = type_id.into_bytes_with_nul();
unsafe {
Self::ref_from_raw(BNCreateNamedType(type_class, type_id.as_ref().as_ptr() as _, &mut name.0))
}
}
pub fn name(&self) -> QualifiedName {
let named_ref: BNQualifiedName = unsafe { BNGetTypeReferenceName(self.handle) };
QualifiedName(named_ref)
}
pub fn id(&self) -> BnString {
unsafe { BnString::from_raw(BNGetTypeReferenceId(self.handle)) }
}
pub fn class(&self) -> NamedTypeReferenceClass {
unsafe { BNGetTypeReferenceClass(self.handle) }
}
fn target_helper(&self, bv: &BinaryView, visited: &mut HashSet<BnString>) -> Option<Ref<Type>> {
#[allow(clippy::manual_filter)]
if let Some(t) = bv.get_type_by_id(self.id()) {
if t.type_class() != TypeClass::NamedTypeReferenceClass {
Some(t)
} else {
let t = t.get_named_type_reference().unwrap();
if visited.contains(&t.id()) {
error!("Can't get target for recursively defined type!");
None
} else {
visited.insert(t.id());
t.target_helper(bv, visited)
}
}
} else {
None
}
}
pub fn target(&self, bv: &BinaryView) -> Option<Ref<Type>> {
self.target_helper(bv, &mut HashSet::new())
}
}
impl ToOwned for NamedTypeReference {
type Owned = Ref<Self>;
fn to_owned(&self) -> Self::Owned {
unsafe { RefCountable::inc_ref(self) }
}
}
unsafe impl RefCountable for NamedTypeReference {
unsafe fn inc_ref(handle: &Self) -> Ref<Self> {
Self::ref_from_raw(BNNewNamedTypeReference(handle.handle))
}
unsafe fn dec_ref(handle: &Self) {
BNFreeNamedTypeReference(handle.handle)
}
}
impl Debug for NamedTypeReference {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{} (id: {})", self.name(), self.id())
}
}
#[repr(transparent)]
pub struct QualifiedName(pub(crate) BNQualifiedName);
impl QualifiedName {
pub fn string(&self) -> String {
unsafe {
slice::from_raw_parts(self.0.name, self.0.nameCount)
.iter()
.map(|c| CStr::from_ptr(*c).to_string_lossy())
.collect::<Vec<_>>()
.join("::")
}
}
pub fn join(&self) -> Cow<str> {
let join: *mut c_char = self.0.join;
unsafe { CStr::from_ptr(join) }.to_string_lossy()
}
pub fn strings(&self) -> Vec<Cow<str>> {
let names: *mut *mut c_char = self.0.name;
unsafe {
slice::from_raw_parts(names, self.0.nameCount)
.iter()
.map(|name| CStr::from_ptr(*name).to_string_lossy())
.collect::<Vec<_>>()
}
}
pub fn len(&self) -> usize {
self.0.nameCount
}
pub fn is_empty(&self) -> bool {
self.0.nameCount == 0
}
}
impl<S: BnStrCompatible> From<S> for QualifiedName {
fn from(name: S) -> Self {
let join = BnString::new("::");
let name = name.into_bytes_with_nul();
let mut list = vec![name.as_ref().as_ptr() as *const _];
QualifiedName(BNQualifiedName {
name: unsafe { BNAllocStringList(list.as_mut_ptr(), 1) },
join: join.into_raw(),
nameCount: 1,
})
}
}
impl<S: BnStrCompatible> From<Vec<S>> for QualifiedName {
fn from(names: Vec<S>) -> Self {
let join = BnString::new("::");
let names = names
.into_iter()
.map(|n| n.into_bytes_with_nul())
.collect::<Vec<_>>();
let mut list = names
.iter()
.map(|n| n.as_ref().as_ptr() as *const _)
.collect::<Vec<_>>();
QualifiedName(BNQualifiedName {
name: unsafe { BNAllocStringList(list.as_mut_ptr(), list.len()) },
join: join.into_raw(),
nameCount: list.len(),
})
}
}
impl Clone for QualifiedName {
fn clone(&self) -> Self {
let strings = self.strings();
let name = Self::from(strings.iter().collect::<Vec<&Cow<str>>>());
name
}
}
impl Hash for QualifiedName {
fn hash<H: Hasher>(&self, state: &mut H) {
self.join().hash(state);
self.strings().hash(state);
}
}
impl Debug for QualifiedName {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.string())
}
}
impl Display for QualifiedName {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.string())
}
}
impl PartialEq for QualifiedName {
fn eq(&self, other: &Self) -> bool {
self.strings() == other.strings()
}
}
impl Eq for QualifiedName {}
impl Drop for QualifiedName {
fn drop(&mut self) {
unsafe {
BNFreeQualifiedName(&mut self.0);
}
}
}
impl CoreArrayProvider for QualifiedName {
type Raw = BNQualifiedName;
type Context = ();
type Wrapped<'a> = &'a QualifiedName;
}
unsafe impl CoreArrayProviderInner for QualifiedName {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeTypeNameList(raw, count);
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
mem::transmute(raw)
}
}
#[repr(transparent)]
pub struct QualifiedNameAndType(pub(crate) BNQualifiedNameAndType);
impl QualifiedNameAndType {
pub fn name(&self) -> &QualifiedName {
unsafe { mem::transmute(&self.0.name) }
}
pub fn type_object(&self) -> Guard<Type> {
unsafe { Guard::new(Type::from_raw(self.0.type_), self) }
}
}
impl Drop for QualifiedNameAndType {
fn drop(&mut self) {
unsafe {
BNFreeQualifiedNameAndType(&mut self.0);
}
}
}
impl CoreArrayProvider for QualifiedNameAndType {
type Raw = BNQualifiedNameAndType;
type Context = ();
type Wrapped<'a> = &'a QualifiedNameAndType;
}
unsafe impl CoreArrayProviderInner for QualifiedNameAndType {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeTypeAndNameList(raw, count);
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
mem::transmute(raw)
}
}
#[repr(transparent)]
pub struct QualifiedNameTypeAndId(pub(crate) BNQualifiedNameTypeAndId);
impl QualifiedNameTypeAndId {
pub fn name(&self) -> &QualifiedName {
unsafe { mem::transmute(&self.0.name) }
}
pub fn id(&self) -> &str {
unsafe { CStr::from_ptr(self.0.id).to_str().unwrap() }
}
pub fn type_object(&self) -> Guard<Type> {
unsafe { Guard::new(Type::from_raw(self.0.type_), self) }
}
}
impl Drop for QualifiedNameTypeAndId {
fn drop(&mut self) {
unsafe {
BNFreeQualifiedNameTypeAndId(&mut self.0);
}
}
}
impl CoreArrayProvider for QualifiedNameTypeAndId {
type Raw = BNQualifiedNameTypeAndId;
type Context = ();
type Wrapped<'a> = &'a QualifiedNameTypeAndId;
}
unsafe impl CoreArrayProviderInner for QualifiedNameTypeAndId {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeTypeIdList(raw, count);
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
mem::transmute(raw)
}
}
pub struct NameAndType(pub(crate) BNNameAndType);
impl NameAndType {
pub(crate) unsafe fn from_raw(raw: &BNNameAndType) -> Self {
Self(*raw)
}
}
impl NameAndType {
pub fn new<S: BnStrCompatible>(name: S, t: &Type, confidence: u8) -> Ref<Self> {
unsafe {
Ref::new(Self(BNNameAndType {
name: BNAllocString(name.into_bytes_with_nul().as_ref().as_ptr() as *mut _),
type_: Ref::into_raw(t.to_owned()).handle,
typeConfidence: confidence,
}))
}
}
pub fn name(&self) -> &str {
let c_str = unsafe { CStr::from_ptr(self.0.name) };
c_str.to_str().unwrap()
}
pub fn t(&self) -> &Type {
unsafe { mem::transmute::<_, &Type>(&self.0.type_) }
}
pub fn type_with_confidence(&self) -> Conf<&Type> {
Conf::new(self.t(), self.0.typeConfidence)
}
}
impl ToOwned for NameAndType {
type Owned = Ref<Self>;
fn to_owned(&self) -> Self::Owned {
unsafe { RefCountable::inc_ref(self) }
}
}
unsafe impl RefCountable for NameAndType {
unsafe fn inc_ref(handle: &Self) -> Ref<Self> {
Self::new(
CStr::from_ptr(handle.0.name),
handle.t(),
handle.0.typeConfidence,
)
}
unsafe fn dec_ref(handle: &Self) {
unsafe {
BNFreeString(handle.0.name);
RefCountable::dec_ref(handle.t());
}
}
}
impl CoreArrayProvider for NameAndType {
type Raw = BNNameAndType;
type Context = ();
type Wrapped<'a> = Guard<'a, NameAndType>;
}
unsafe impl CoreArrayProviderInner for NameAndType {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeNameAndTypeList(raw, count);
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
unsafe { Guard::new(NameAndType::from_raw(raw), raw) }
}
}
#[repr(transparent)]
pub struct DataVariable(pub(crate) BNDataVariable);
impl DataVariable {
pub fn address(&self) -> u64 {
self.0.address
}
pub fn auto_discovered(&self) -> bool {
self.0.autoDiscovered
}
pub fn t(&self) -> &Type {
unsafe { mem::transmute(&self.0.type_) }
}
pub fn type_with_confidence(&self) -> Conf<&Type> {
Conf::new(self.t(), self.0.typeConfidence)
}
pub fn symbol(&self, bv: &BinaryView) -> Option<Ref<Symbol>> {
bv.symbol_by_address(self.0.address).ok()
}
}
impl ToOwned for DataVariable {
type Owned = Ref<Self>;
fn to_owned(&self) -> Self::Owned {
unsafe { RefCountable::inc_ref(self) }
}
}
unsafe impl RefCountable for DataVariable {
unsafe fn inc_ref(handle: &Self) -> Ref<Self> {
unsafe {
Ref::new(Self(BNDataVariable {
type_: Ref::into_raw(handle.t().to_owned()).handle,
..handle.0
}))
}
}
unsafe fn dec_ref(handle: &Self) {
unsafe { BNFreeType(handle.0.type_) }
}
}
impl CoreArrayProvider for DataVariable {
type Raw = BNDataVariable;
type Context = ();
type Wrapped<'a> = &'a DataVariable;
}
unsafe impl CoreArrayProviderInner for DataVariable {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeDataVariables(raw, count);
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
mem::transmute(raw)
}
}
pub struct DataVariableAndName<S: BnStrCompatible> {
pub address: u64,
pub t: Conf<Ref<Type>>,
pub auto_discovered: bool,
pub name: S,
}
impl DataVariableAndName<String> {
pub(crate) fn from_raw(var: &BNDataVariableAndName) -> Self {
Self {
address: var.address,
t: Conf::new(unsafe { Type::ref_from_raw(var.type_) }, var.typeConfidence),
auto_discovered: var.autoDiscovered,
name: raw_to_string(var.name).unwrap(),
}
}
}
impl<S: BnStrCompatible> DataVariableAndName<S> {
pub fn new(address: u64, t: Conf<Ref<Type>>, auto_discovered: bool, name: S) -> Self {
Self {
address,
t,
auto_discovered,
name,
}
}
pub fn type_with_confidence(&self) -> Conf<Ref<Type>> {
Conf::new(self.t.contents.clone(), self.t.confidence)
}
}
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)]
pub enum RegisterValueType {
UndeterminedValue,
EntryValue,
ConstantValue,
ConstantPointerValue,
ExternalPointerValue,
StackFrameOffset,
ReturnAddressValue,
ImportedAddressValue,
SignedRangeValue,
UnsignedRangeValue,
LookupTableValue,
InSetOfValues,
NotInSetOfValues,
ConstantDataValue,
ConstantDataZeroExtendValue,
ConstantDataSignExtendValue,
ConstantDataAggregateValue,
}
impl RegisterValueType {
pub(crate) fn from_raw_value(value: u32) -> Option<Self> {
use BNRegisterValueType::*;
Some(match value {
x if x == UndeterminedValue as u32 => Self::UndeterminedValue,
x if x == EntryValue as u32 => Self::EntryValue,
x if x == ConstantValue as u32 => Self::ConstantValue,
x if x == ConstantPointerValue as u32 => Self::ConstantPointerValue,
x if x == ExternalPointerValue as u32 => Self::ExternalPointerValue,
x if x == StackFrameOffset as u32 => Self::StackFrameOffset,
x if x == ReturnAddressValue as u32 => Self::ReturnAddressValue,
x if x == ImportedAddressValue as u32 => Self::ImportedAddressValue,
x if x == SignedRangeValue as u32 => Self::SignedRangeValue,
x if x == UnsignedRangeValue as u32 => Self::UnsignedRangeValue,
x if x == LookupTableValue as u32 => Self::LookupTableValue,
x if x == InSetOfValues as u32 => Self::InSetOfValues,
x if x == NotInSetOfValues as u32 => Self::NotInSetOfValues,
x if x == ConstantDataValue as u32 => Self::ConstantDataValue,
x if x == ConstantDataZeroExtendValue as u32 => Self::ConstantDataZeroExtendValue,
x if x == ConstantDataSignExtendValue as u32 => Self::ConstantDataSignExtendValue,
x if x == ConstantDataAggregateValue as u32 => Self::ConstantDataAggregateValue,
_ => return None,
})
}
pub(crate) fn into_raw_value(self) -> BNRegisterValueType {
use BNRegisterValueType::*;
match self {
Self::UndeterminedValue => UndeterminedValue,
Self::EntryValue => EntryValue,
Self::ConstantValue => ConstantValue,
Self::ConstantPointerValue => ConstantPointerValue,
Self::ExternalPointerValue => ExternalPointerValue,
Self::StackFrameOffset => StackFrameOffset,
Self::ReturnAddressValue => ReturnAddressValue,
Self::ImportedAddressValue => ImportedAddressValue,
Self::SignedRangeValue => SignedRangeValue,
Self::UnsignedRangeValue => UnsignedRangeValue,
Self::LookupTableValue => LookupTableValue,
Self::InSetOfValues => InSetOfValues,
Self::NotInSetOfValues => NotInSetOfValues,
Self::ConstantDataValue => ConstantDataValue,
Self::ConstantDataZeroExtendValue => ConstantDataZeroExtendValue,
Self::ConstantDataSignExtendValue => ConstantDataSignExtendValue,
Self::ConstantDataAggregateValue => ConstantDataAggregateValue,
}
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub struct RegisterValue {
pub(crate) state: RegisterValueType,
pub(crate) value: i64,
pub(crate) offset: i64,
pub(crate) size: usize,
}
impl RegisterValue {
pub fn new(state: RegisterValueType, value: i64, offset: i64, size: usize) -> Self {
Self {
state,
value,
offset,
size,
}
}
}
impl From<BNRegisterValue> for RegisterValue {
fn from(value: BNRegisterValue) -> Self {
Self {
state: RegisterValueType::from_raw_value(value.state as u32).unwrap(),
value: value.value,
offset: value.offset,
size: value.size,
}
}
}
impl From<RegisterValue> for BNRegisterValue {
fn from(value: RegisterValue) -> Self {
Self {
state: value.state.into_raw_value(),
value: value.value,
offset: value.offset,
size: value.size,
}
}
}
#[derive(Clone, Debug, PartialEq, Hash)]
pub struct ConstantData {
function: Ref<Function>,
value: RegisterValue,
}
impl ConstantData {
pub(crate) fn new(function: Ref<Function>, value: RegisterValue) -> Self {
Self { function, value }
}
}
#[repr(transparent)]
#[derive(Copy, Clone, Debug)]
pub struct ValueRange<T> {
raw: BNValueRange,
_t: core::marker::PhantomData<T>,
}
impl<T> ValueRange<T> {
fn from_raw(value: BNValueRange) -> Self {
Self {
raw: value,
_t: core::marker::PhantomData,
}
}
fn into_raw(self) -> BNValueRange {
self.raw
}
}
impl IntoIterator for ValueRange<u64> {
type Item = u64;
type IntoIter = core::iter::StepBy<Range<u64>>;
fn into_iter(self) -> Self::IntoIter {
(self.raw.start..self.raw.end).step_by(self.raw.step.try_into().unwrap())
}
}
impl IntoIterator for ValueRange<i64> {
type Item = i64;
type IntoIter = core::iter::StepBy<Range<i64>>;
fn into_iter(self) -> Self::IntoIter {
(self.raw.start as i64..self.raw.end as i64).step_by(self.raw.step.try_into().unwrap())
}
}
#[derive(Clone, Debug)]
pub enum PossibleValueSet {
UndeterminedValue,
EntryValue {
reg: i64,
},
ConstantValue {
value: i64,
},
ConstantPointerValue {
value: i64,
},
ExternalPointerValue,
StackFrameOffset {
offset: i64,
},
ReturnAddressValue,
ImportedAddressValue,
SignedRangeValue {
offset: i64,
ranges: Vec<ValueRange<i64>>,
},
UnsignedRangeValue {
offset: i64,
ranges: Vec<ValueRange<u64>>,
},
LookupTableValue {
tables: Vec<LookupTableEntry>,
},
InSetOfValues {
values: HashSet<i64>,
},
NotInSetOfValues {
values: HashSet<i64>,
},
ConstantDataValue {
value_type: ConstantDataType,
value: i64,
},
}
#[derive(Copy, Clone, Debug)]
pub enum ConstantDataType {
Value,
ZeroExtend,
SignExtend,
Aggregate,
}
impl PossibleValueSet {
pub(crate) unsafe fn from_raw(value: BNPossibleValueSet) -> Self {
unsafe fn from_range<T>(value: BNPossibleValueSet) -> Vec<ValueRange<T>> {
core::slice::from_raw_parts(value.ranges, value.count)
.iter()
.copied()
.map(|range| ValueRange::from_raw(range))
.collect()
}
let from_sets = |value: BNPossibleValueSet| {
unsafe { core::slice::from_raw_parts(value.valueSet, value.count) }
.iter()
.copied()
.collect()
};
use BNRegisterValueType::*;
match value.state {
UndeterminedValue => Self::UndeterminedValue,
EntryValue => Self::EntryValue { reg: value.value },
ConstantValue => Self::ConstantValue { value: value.value },
ConstantPointerValue => Self::ConstantPointerValue { value: value.value },
StackFrameOffset => Self::StackFrameOffset {
offset: value.value,
},
ConstantDataValue => Self::ConstantDataValue {
value_type: ConstantDataType::Value,
value: value.value,
},
ConstantDataZeroExtendValue => Self::ConstantDataValue {
value_type: ConstantDataType::ZeroExtend,
value: value.value,
},
ConstantDataSignExtendValue => Self::ConstantDataValue {
value_type: ConstantDataType::SignExtend,
value: value.value,
},
ConstantDataAggregateValue => Self::ConstantDataValue {
value_type: ConstantDataType::Aggregate,
value: value.value,
},
SignedRangeValue => Self::SignedRangeValue {
offset: value.value,
ranges: from_range(value),
},
UnsignedRangeValue => Self::UnsignedRangeValue {
offset: value.value,
ranges: from_range(value),
},
LookupTableValue => {
let raw_tables = unsafe { core::slice::from_raw_parts(value.table, value.count) };
let raw_from_tables = |i: &BNLookupTableEntry| unsafe {
core::slice::from_raw_parts(i.fromValues, i.fromCount)
};
let tables = raw_tables
.iter()
.map(|table| LookupTableEntry {
from_values: raw_from_tables(table).to_vec(),
to_value: table.toValue,
})
.collect();
Self::LookupTableValue { tables }
}
NotInSetOfValues => Self::NotInSetOfValues {
values: from_sets(value),
},
InSetOfValues => Self::InSetOfValues {
values: from_sets(value),
},
ImportedAddressValue => Self::ImportedAddressValue,
ReturnAddressValue => Self::ReturnAddressValue,
ExternalPointerValue => Self::ExternalPointerValue,
}
}
pub(crate) fn into_raw(self) -> PossibleValueSetRaw {
let mut raw: BNPossibleValueSet = unsafe { core::mem::zeroed() };
raw.state = self.value_type().into_raw_value();
match self {
PossibleValueSet::UndeterminedValue
| PossibleValueSet::ExternalPointerValue
| PossibleValueSet::ReturnAddressValue
| PossibleValueSet::ImportedAddressValue => {}
PossibleValueSet::EntryValue { reg: value }
| PossibleValueSet::ConstantValue { value }
| PossibleValueSet::ConstantPointerValue { value }
| PossibleValueSet::ConstantDataValue { value, .. }
| PossibleValueSet::StackFrameOffset { offset: value } => raw.value = value,
PossibleValueSet::NotInSetOfValues { values }
| PossibleValueSet::InSetOfValues { values } => {
let values = Box::leak(values.into_iter().collect());
raw.valueSet = values.as_mut_ptr();
raw.count = values.len();
}
PossibleValueSet::SignedRangeValue { offset, ranges } => {
let ranges = Box::leak(ranges.into_iter().map(|x| x.into_raw()).collect());
raw.value = offset;
raw.ranges = ranges.as_mut_ptr();
raw.count = ranges.len();
}
PossibleValueSet::UnsignedRangeValue { offset, ranges } => {
let ranges = Box::leak(ranges.into_iter().map(|x| x.into_raw()).collect());
raw.value = offset;
raw.ranges = ranges.as_mut_ptr();
raw.count = ranges.len();
}
PossibleValueSet::LookupTableValue { tables } => {
let tables = Box::leak(tables.into_iter().map(|table| table.into_raw()).collect());
raw.table = tables.as_mut_ptr() as *mut BNLookupTableEntry;
raw.count = tables.len();
}
}
PossibleValueSetRaw(raw)
}
pub fn value_type(&self) -> RegisterValueType {
use RegisterValueType::*;
match self {
PossibleValueSet::UndeterminedValue => UndeterminedValue,
PossibleValueSet::EntryValue { .. } => EntryValue,
PossibleValueSet::ConstantValue { .. } => ConstantValue,
PossibleValueSet::ConstantPointerValue { .. } => ConstantPointerValue,
PossibleValueSet::ExternalPointerValue => ExternalPointerValue,
PossibleValueSet::StackFrameOffset { .. } => StackFrameOffset,
PossibleValueSet::ReturnAddressValue => ReturnAddressValue,
PossibleValueSet::ImportedAddressValue => ImportedAddressValue,
PossibleValueSet::SignedRangeValue { .. } => SignedRangeValue,
PossibleValueSet::UnsignedRangeValue { .. } => UnsignedRangeValue,
PossibleValueSet::LookupTableValue { .. } => LookupTableValue,
PossibleValueSet::InSetOfValues { .. } => InSetOfValues,
PossibleValueSet::NotInSetOfValues { .. } => NotInSetOfValues,
PossibleValueSet::ConstantDataValue {
value_type: ConstantDataType::Value,
..
} => ConstantDataValue,
PossibleValueSet::ConstantDataValue {
value_type: ConstantDataType::ZeroExtend,
..
} => ConstantDataZeroExtendValue,
PossibleValueSet::ConstantDataValue {
value_type: ConstantDataType::SignExtend,
..
} => ConstantDataSignExtendValue,
PossibleValueSet::ConstantDataValue {
value_type: ConstantDataType::Aggregate,
..
} => ConstantDataAggregateValue,
}
}
}
#[repr(transparent)]
pub(crate) struct PossibleValueSetRaw(BNPossibleValueSet);
impl PossibleValueSetRaw {
pub fn as_ffi(&self) -> &BNPossibleValueSet {
&self.0
}
}
impl Drop for PossibleValueSetRaw {
fn drop(&mut self) {
use BNRegisterValueType::*;
match self.0.state {
UndeterminedValue
| ExternalPointerValue
| ReturnAddressValue
| ImportedAddressValue
| EntryValue
| ConstantValue
| ConstantPointerValue
| StackFrameOffset
| ConstantDataValue
| ConstantDataZeroExtendValue
| ConstantDataSignExtendValue
| ConstantDataAggregateValue => {}
InSetOfValues | NotInSetOfValues => {
let _values: Box<[i64]> = unsafe {
Box::from_raw(ptr::slice_from_raw_parts_mut(self.0.valueSet, self.0.count))
};
}
SignedRangeValue | UnsignedRangeValue => {
let _ranges: Box<[BNValueRange]> = unsafe {
Box::from_raw(ptr::slice_from_raw_parts_mut(self.0.ranges, self.0.count))
};
}
LookupTableValue => {
let table_ptr = self.0.table as *mut LookupTableEntryRaw;
let _table: Box<[LookupTableEntryRaw]> = unsafe {
Box::from_raw(ptr::slice_from_raw_parts_mut(table_ptr, self.0.count))
};
}
}
}
}
#[derive(Clone, Debug)]
pub struct LookupTableEntry {
pub from_values: Vec<i64>,
pub to_value: i64,
}
impl LookupTableEntry {
fn into_raw(self) -> LookupTableEntryRaw {
let from_value = Box::leak(self.from_values.into_boxed_slice());
LookupTableEntryRaw(BNLookupTableEntry {
toValue: self.to_value,
fromValues: from_value.as_mut_ptr(),
fromCount: from_value.len(),
})
}
}
#[repr(transparent)]
struct LookupTableEntryRaw(BNLookupTableEntry);
impl Drop for LookupTableEntryRaw {
fn drop(&mut self) {
let _from_value: Box<[i64]> = unsafe {
Box::from_raw(ptr::slice_from_raw_parts_mut(
self.0.fromValues,
self.0.fromCount,
))
};
}
}
#[derive(Copy, Clone, Eq, Hash, PartialEq)]
pub struct ArchAndAddr {
pub arch: CoreArchitecture,
pub address: u64,
}
pub struct UserVariableValues {
pub(crate) vars: *const [BNUserVariableValue],
}
impl UserVariableValues {
pub fn into_hashmap(self) -> HashMap<Variable, HashMap<ArchAndAddr, PossibleValueSet>> {
let mut result: HashMap<Variable, HashMap<ArchAndAddr, PossibleValueSet>> = HashMap::new();
for (var, def_site, possible_val) in self.all() {
result
.entry(var)
.or_default()
.entry(def_site)
.or_insert(possible_val);
}
result
}
pub fn all(&self) -> impl Iterator<Item = (Variable, ArchAndAddr, PossibleValueSet)> {
unsafe { &*self.vars }.iter().map(|var_val| {
let var = unsafe { Variable::from_raw(var_val.var) };
let def_site = ArchAndAddr {
arch: unsafe { CoreArchitecture::from_raw(var_val.defSite.arch) },
address: var_val.defSite.address,
};
let possible_val = unsafe { PossibleValueSet::from_raw(var_val.value) };
(var, def_site, possible_val)
})
}
pub fn values_from_variable(
&self,
var: Variable,
) -> impl Iterator<Item = (ArchAndAddr, PossibleValueSet)> {
self.all()
.filter(move |(t_var, _, _)| t_var == &var)
.map(|(_var, def_site, possible_val)| (def_site, possible_val))
}
}
impl Drop for UserVariableValues {
fn drop(&mut self) {
unsafe { BNFreeUserVariableValues(self.vars as *mut BNUserVariableValue) };
}
}
#[derive(Copy, Clone, Eq, Hash, PartialEq)]
pub struct ConstantReference {
pub value: i64,
pub size: usize,
pub pointer: bool,
pub intermediate: bool,
}
impl ConstantReference {
pub fn from_raw(value: BNConstantReference) -> Self {
Self {
value: value.value,
size: value.size,
pointer: value.pointer,
intermediate: value.intermediate,
}
}
pub fn into_raw(self) -> BNConstantReference {
BNConstantReference {
value: self.value,
size: self.size,
pointer: self.pointer,
intermediate: self.intermediate,
}
}
}
impl CoreArrayProvider for ConstantReference {
type Raw = BNConstantReference;
type Context = ();
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for ConstantReference {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeConstantReferenceList(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
Self::from_raw(*raw)
}
}
pub struct IndirectBranchInfo {
pub source_arch: CoreArchitecture,
pub source_addr: u64,
pub dest_arch: CoreArchitecture,
pub dest_addr: u64,
pub auto_defined: bool,
}
impl IndirectBranchInfo {
pub fn from_raw(value: BNIndirectBranchInfo) -> Self {
Self {
source_arch: unsafe { CoreArchitecture::from_raw(value.sourceArch) },
source_addr: value.sourceAddr,
dest_arch: unsafe { CoreArchitecture::from_raw(value.destArch) },
dest_addr: value.destAddr,
auto_defined: value.autoDefined,
}
}
pub fn into_raw(self) -> BNIndirectBranchInfo {
BNIndirectBranchInfo {
sourceArch: self.source_arch.0,
sourceAddr: self.source_addr,
destArch: self.dest_arch.0,
destAddr: self.dest_addr,
autoDefined: self.auto_defined,
}
}
}
impl CoreArrayProvider for IndirectBranchInfo {
type Raw = BNIndirectBranchInfo;
type Context = ();
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for IndirectBranchInfo {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeIndirectBranchList(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
Self::from_raw(*raw)
}
}
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)]
pub enum HighlightStandardColor {
BlueHighlightColor,
GreenHighlightColor,
CyanHighlightColor,
RedHighlightColor,
MagentaHighlightColor,
YellowHighlightColor,
OrangeHighlightColor,
WhiteHighlightColor,
BlackHighlightColor,
}
impl HighlightStandardColor {
pub fn from_raw(value: BNHighlightStandardColor) -> Option<Self> {
Some(match value {
BNHighlightStandardColor::NoHighlightColor => return None,
BNHighlightStandardColor::BlueHighlightColor => Self::BlueHighlightColor,
BNHighlightStandardColor::GreenHighlightColor => Self::GreenHighlightColor,
BNHighlightStandardColor::CyanHighlightColor => Self::CyanHighlightColor,
BNHighlightStandardColor::RedHighlightColor => Self::RedHighlightColor,
BNHighlightStandardColor::MagentaHighlightColor => Self::MagentaHighlightColor,
BNHighlightStandardColor::YellowHighlightColor => Self::YellowHighlightColor,
BNHighlightStandardColor::OrangeHighlightColor => Self::OrangeHighlightColor,
BNHighlightStandardColor::WhiteHighlightColor => Self::WhiteHighlightColor,
BNHighlightStandardColor::BlackHighlightColor => Self::BlackHighlightColor,
})
}
pub fn into_raw(self) -> BNHighlightStandardColor {
match self {
Self::BlueHighlightColor => BNHighlightStandardColor::BlueHighlightColor,
Self::GreenHighlightColor => BNHighlightStandardColor::GreenHighlightColor,
Self::CyanHighlightColor => BNHighlightStandardColor::CyanHighlightColor,
Self::RedHighlightColor => BNHighlightStandardColor::RedHighlightColor,
Self::MagentaHighlightColor => BNHighlightStandardColor::MagentaHighlightColor,
Self::YellowHighlightColor => BNHighlightStandardColor::YellowHighlightColor,
Self::OrangeHighlightColor => BNHighlightStandardColor::OrangeHighlightColor,
Self::WhiteHighlightColor => BNHighlightStandardColor::WhiteHighlightColor,
Self::BlackHighlightColor => BNHighlightStandardColor::BlackHighlightColor,
}
}
}
#[derive(Debug, Copy, Clone)]
pub enum HighlightColor {
NoHighlightColor {
alpha: u8,
},
StandardHighlightColor {
color: HighlightStandardColor,
alpha: u8,
},
MixedHighlightColor {
color: HighlightStandardColor,
mix_color: HighlightStandardColor,
mix: u8,
alpha: u8,
},
CustomHighlightColor {
r: u8,
g: u8,
b: u8,
alpha: u8,
},
}
impl HighlightColor {
pub fn from_raw(raw: BNHighlightColor) -> Self {
const HIGHLIGHT_COLOR: u32 = BNHighlightColorStyle::StandardHighlightColor as u32;
const MIXED_HIGHLIGHT_COLOR: u32 = BNHighlightColorStyle::MixedHighlightColor as u32;
const CUSTOM_HIGHLIHGT_COLOR: u32 = BNHighlightColorStyle::CustomHighlightColor as u32;
match raw.style as u32 {
HIGHLIGHT_COLOR => {
let Some(color) = HighlightStandardColor::from_raw(raw.color) else {
return Self::NoHighlightColor { alpha: raw.alpha };
};
Self::StandardHighlightColor {
color,
alpha: raw.alpha,
}
}
MIXED_HIGHLIGHT_COLOR => {
let Some(color) = HighlightStandardColor::from_raw(raw.color) else {
panic!("Highlight mixed color with no color");
};
let Some(mix_color) = HighlightStandardColor::from_raw(raw.mixColor) else {
panic!("Highlight mixed color with no mix_color");
};
Self::MixedHighlightColor {
color,
mix_color,
mix: raw.mix,
alpha: raw.alpha,
}
}
CUSTOM_HIGHLIHGT_COLOR => Self::CustomHighlightColor {
r: raw.r,
g: raw.g,
b: raw.b,
alpha: raw.alpha,
},
_ => Self::NoHighlightColor { alpha: u8::MAX },
}
}
pub fn into_raw(self) -> BNHighlightColor {
let zeroed: BNHighlightColor = unsafe { core::mem::zeroed() };
match self {
Self::NoHighlightColor { alpha } => BNHighlightColor {
style: BNHighlightColorStyle::StandardHighlightColor,
color: BNHighlightStandardColor::NoHighlightColor,
alpha,
..zeroed
},
Self::StandardHighlightColor { color, alpha } => BNHighlightColor {
style: BNHighlightColorStyle::StandardHighlightColor,
color: color.into_raw(),
alpha,
..zeroed
},
Self::MixedHighlightColor {
color,
mix_color,
mix,
alpha,
} => BNHighlightColor {
color: color.into_raw(),
mixColor: mix_color.into_raw(),
mix,
alpha,
..zeroed
},
Self::CustomHighlightColor { r, g, b, alpha } => BNHighlightColor {
r,
g,
b,
alpha,
..zeroed
},
}
}
}
pub type IntegerDisplayType = binaryninjacore_sys::BNIntegerDisplayType;
#[derive(Debug, Clone)]
pub struct StackVariableReference {
_source_operand: u32,
var_type: Conf<Ref<Type>>,
name: BnString,
var: Variable,
offset: i64,
size: usize,
}
impl StackVariableReference {
pub fn from_raw(value: BNStackVariableReference) -> Self {
let var_type = Conf::new(
unsafe { Type::ref_from_raw(value.type_) },
value.typeConfidence,
);
let name = unsafe { BnString::from_raw(value.name) };
let var = unsafe { Variable::from_identifier(value.varIdentifier) };
let offset = value.referencedOffset;
let size = value.size;
Self {
_source_operand: value.sourceOperand,
var_type,
name,
var,
offset,
size,
}
}
pub fn variable(&self) -> &Variable {
&self.var
}
pub fn variable_type(&self) -> Conf<&Type> {
self.var_type.as_ref()
}
pub fn name(&self) -> &str {
self.name.as_str()
}
pub fn offset(&self) -> i64 {
self.offset
}
pub fn size(&self) -> usize {
self.size
}
}
impl CoreArrayProvider for StackVariableReference {
type Raw = BNStackVariableReference;
type Context = ();
type Wrapped<'a> = Guard<'a, Self>;
}
unsafe impl CoreArrayProviderInner for StackVariableReference {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeStackVariableReferenceList(raw, count)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, context: &'a Self::Context) -> Self::Wrapped<'a> {
Guard::new(Self::from_raw(*raw), context)
}
}
#[derive(Debug, Copy, Clone)]
pub struct RegisterStackAdjustment<A: Architecture> {
reg_id: u32,
adjustment: Conf<i32>,
arch: A::Handle,
}
impl<A: Architecture> RegisterStackAdjustment<A> {
pub(crate) unsafe fn from_raw(value: BNRegisterStackAdjustment, arch: A::Handle) -> Self {
RegisterStackAdjustment {
reg_id: value.regStack,
adjustment: Conf::new(value.adjustment, value.confidence),
arch,
}
}
pub(crate) fn into_raw(self) -> BNRegisterStackAdjustment {
BNRegisterStackAdjustment {
regStack: self.reg_id,
adjustment: self.adjustment.contents,
confidence: self.adjustment.confidence,
}
}
pub fn new<I>(reg_id: u32, adjustment: I, arch_handle: A::Handle) -> Self
where
I: Into<Conf<i32>>,
{
Self {
reg_id,
adjustment: adjustment.into(),
arch: arch_handle,
}
}
pub const fn register_id(&self) -> u32 {
self.reg_id
}
pub fn register(&self) -> A::Register {
self.arch.borrow().register_from_id(self.reg_id).unwrap()
}
}
impl<A: Architecture> CoreArrayProvider for RegisterStackAdjustment<A> {
type Raw = BNRegisterStackAdjustment;
type Context = A::Handle;
type Wrapped<'a> = Self;
}
unsafe impl<A: Architecture> CoreArrayProviderInner for RegisterStackAdjustment<A> {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeRegisterStackAdjustments(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, context: &'a Self::Context) -> Self::Wrapped<'a> {
Self::from_raw(*raw, context.clone())
}
}
pub struct MergedVariable {
target: Variable,
sources: ManuallyDrop<Array<Variable>>,
}
impl MergedVariable {
pub fn target(&self) -> Variable {
self.target
}
pub fn sources(&self) -> &Array<Variable> {
&self.sources
}
}
impl CoreArrayProvider for MergedVariable {
type Raw = BNMergedVariable;
type Context = ();
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for MergedVariable {
unsafe fn free(raw: *mut Self::Raw, count: usize, _context: &Self::Context) {
BNFreeMergedVariableList(raw, count)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
Self {
target: Variable::from_raw(raw.target),
sources: ManuallyDrop::new(Array::new(raw.sources, raw.sourceCount, ())),
}
}
}
pub struct UnresolvedIndirectBranches(u64);
impl UnresolvedIndirectBranches {
pub fn address(&self) -> u64 {
self.0
}
}
impl CoreArrayProvider for UnresolvedIndirectBranches {
type Raw = u64;
type Context = ();
type Wrapped<'a> = Self;
}
unsafe impl CoreArrayProviderInner for UnresolvedIndirectBranches {
unsafe fn free(raw: *mut Self::Raw, _count: usize, _context: &Self::Context) {
BNFreeAddressList(raw)
}
unsafe fn wrap_raw<'a>(raw: &'a Self::Raw, _context: &'a Self::Context) -> Self::Wrapped<'a> {
Self(*raw)
}
}