Frida 17 JavaScript API compatibility checker and fixer. Use when writing, reviewing, or fixing Frida scripts, especially when migrating from older Frida versions. Detects deprecated APIs removed in Frida 17 (May 2025) and provides correct replacements. Covers Module, Memory, Process APIs and common naming conflicts.
Install
npx skillscat add yfe404/frida-17-skill Install via the SkillsCat registry.
Frida 17 Scripting Guide
This skill helps write and fix Frida scripts compatible with Frida 17.0.0 (released May 2025).
Breaking Changes in Frida 17
1. Static Module Methods - REMOVED
// OLD - No longer works in Frida 17
Module.findBaseAddress('libriver.so')
Module.getBaseAddress('libriver.so')
Module.findExportByName(null, 'open')
Module.findExportByName('libc.so', 'open')
Module.getExportByName(null, 'open')
Module.ensureInitialized('libc.so')
Module.enumerateExports('libc.so')
Module.enumerateSymbols('libc.so')
// NEW - Use Process and instance methods instead
var lib = Process.findModuleByName('libriver.so'); // returns Module or null
var lib = Process.getModuleByName('libriver.so'); // throws if not found
lib.base // module base address
lib.findExportByName('open') // returns address or null
lib.getExportByName('open') // throws if not found
lib.enumerateExports() // returns array
lib.enumerateSymbols() // returns array2. Static Memory Methods - REMOVED
// OLD - No longer works
Memory.readU32(ptr)
Memory.writeU32(ptr, value)
// NEW - Use NativePointer instance methods
ptr.readU32()
ptr.writeU32(value)3. Legacy Enumeration APIs - REMOVED
// OLD - Callback style removed
Process.enumerateModules({ onMatch: fn, onComplete: fn })
Process.enumerateModulesSync()
// NEW - Returns array directly
Process.enumerateModules()4. Reserved Function Names - DO NOT OVERRIDE
The following are built-in Frida functions. Defining custom functions with these names causes:TypeError: cannot define variable 'hexdump'
Reserved names:
hexdump- UsedumpHexinstead for custom hex dump functionsptr- pointer constructor shorthandNULL- null pointer constant
// BAD - conflicts with built-in
function hexdump(ptr, len) { ... }
// GOOD - use different name
function dumpHex(ptr, len) { ... }NativePointer Methods (Valid in Frida 17)
Conversion:
toInt32()- cast to signed 32-bit integertoNumber()- convert to JavaScript numbertoString([radix])- convert to string
NOT available:
toUInt32()- DOES NOT EXIST, usetoInt32()for sizes < 2^31
Memory reading:
readU8(),readS8(),readU16(),readS16()readU32(),readS32(),readU64(),readS64()readByteArray(length)- returns ArrayBufferreadPointer(),readCString(),readUtf8String()
Memory writing:
writeU8(value),writeS8(value), etc.writeByteArray(bytes)- bytes must be ArrayBuffer or JS arraywritePointer(ptr),writeUtf8String(str)
Pointer arithmetic:
add(rhs),sub(rhs),and(rhs),or(rhs),xor(rhs)shr(n),shl(n),not()isNull(),equals(rhs),compare(rhs)
Java Bridge API (Unchanged in Frida 17)
Java.perform(function() {
var MyClass = Java.use('com.example.MyClass');
// Hook with overload
MyClass.myMethod.overload('int', 'java.lang.String').implementation = function(a, b) {
console.log('Called with: ' + a + ', ' + b);
// Call original
return this.myMethod.overload('int', 'java.lang.String').call(this, a, b);
};
// Hook all overloads
MyClass.myMethod.overloads.forEach(function(overload) {
overload.implementation = function() {
return overload.apply(this, arguments);
};
});
});Java byte[] handling:
Java byte arrays cannot be passed directly to Memory.alloc().writeByteArray().
Convert manually:
// BAD - throws "expected a buffer-like object"
var hex = dumpHex(Memory.alloc(javaByteArray.length).writeByteArray(javaByteArray), len);
// GOOD - iterate and convert
var hex = "";
for (var i = 0; i < javaByteArray.length; i++) {
hex += ("0" + (javaByteArray[i] & 0xff).toString(16)).slice(-2);
}Common Patterns for Frida 17
Waiting for a library to load
function waitForLibrary(libName, callback) {
var lib = Process.findModuleByName(libName);
if (lib) {
callback(lib.base);
return;
}
var pollInterval = setInterval(function() {
var lib = Process.findModuleByName(libName);
if (lib) {
clearInterval(pollInterval);
callback(lib.base);
}
}, 500);
}Hooking libc functions
var libc = Process.findModuleByName('libc.so');
var open = libc ? libc.findExportByName('open') : null;
if (open) {
Interceptor.attach(open, {
onEnter: function(args) {
console.log('open(' + args[0].readCString() + ')');
}
});
}Custom hex dump function
function dumpHex(ptr, len) {
if (!ptr || ptr.isNull()) return 'null';
try {
var bytes = ptr.readByteArray(len);
if (!bytes) return 'null';
var arr = new Uint8Array(bytes);
var hex = '';
for (var i = 0; i < arr.length; i++) {
hex += ('0' + arr[i].toString(16)).slice(-2);
}
return hex;
} catch (e) {
return 'error: ' + e;
}
}Checklist for Frida 17 Compatibility
When reviewing a Frida script, check for:
-
Module.findBaseAddress()->Process.findModuleByName().base -
Module.getBaseAddress()->Process.getModuleByName().base -
Module.findExportByName(null, name)->Process.findModuleByName('libc.so').findExportByName(name) -
Module.findExportByName(lib, name)->Process.findModuleByName(lib).findExportByName(name) -
Module.enumerateExports(lib)->Process.getModuleByName(lib).enumerateExports() -
Module.enumerateSymbols(lib)->Process.getModuleByName(lib).enumerateSymbols() -
Memory.readU32(ptr)->ptr.readU32() -
toUInt32()->toInt32()(toUInt32 never existed) -
function hexdump()->function dumpHex()(name conflict) - Java byte[] with
writeByteArray()-> manual hex conversion