Error occurred while disassembling class Landroid.support Invalid method index: 35
Closed this issue · 21 comments
What seems to be the problem?
I'am trying to deodex Mms.apk from a Chinese rom for Jiayu G4s phone, because
it has a bug related to incorrect Caller ID
matching. When trying to deodex de apk everything is fine, but when trying to
deodex de odex file, I receive an error.
What is the exact smali/baksmali command that you ran?
java -jar baksmali-2.0.3.jar -a 19 -d \framework -c
\framework\core.jar:\framework\conscrypt.jar:\framework\okhttp.jar:\framework\co
re-junit.jar:\framework\bouncycastle.jar:\framework\ext.jar:\framework\framework
.jar:\framework\framework2.jar:\framework\telephony-common.jar:\framework\voip-c
ommon.jar:\framework\mms-common.jar:\framework\android.policy.jar:\framework\ser
vices.jar:\framework\apache-xml.jar:\framework\webviewchromium.jar:\framework\me
diatek-common.jar:\framework\mediatek-framework.jar:\framework\CustomProperties.
jar:\framework\mediatek-telephony-common.jar:\framework\mediatek-tablet.jar -x
nou\Mms.apk -o nou\MMS
What version of smali/baksmali are you using? What rom are you working
from?
smali-2.0.3.jar and baksmali-2.0.3.jar
G4S-20140927-1826499 (Android 4.4.2)
The rom can be found here:
http://pan.baidu.com/share/link?shareid=187936476&uk=1680007479
What is the airspeed velocity of an unladen swallow?
Please provide any additional information below: error messages, symptoms,
etc.
The error message is:
Error occurred while disassembling class
Landroid.support.v4.accessibilityservice.AccessibilityServiceInfoCompat; -
skipping class
java.lang.RuntimeException: Invalid method index: 35
at
org.jf.dexlib2.analysis.InlineMethodResolver$InlineMethodResolver_version36.reso
lveExecuteInline(InlineMethodResolver.java:175)
at
org.jf.dexlib2.analysis.MethodAnalyzer.analyzeExecuteInline(MethodAnalyzer.java:
1422)
at
org.jf.dexlib2.analysis.MethodAnalyzer.analyzeInstruction(MethodAnalyzer.java:94
1)
at org.jf.dexlib2.analysis.MethodAnalyzer.analyze(MethodAnalyzer.java:202)
at org.jf.dexlib2.analysis.MethodAnalyzer.<init>(MethodAnalyzer.java:137)
at
org.jf.baksmali.Adaptors.MethodDefinition.addAnalyzedInstructionMethodItems(Meth
odDefinition.java:387)
at
org.jf.baksmali.Adaptors.MethodDefinition.getMethodItems(MethodDefinition.java:2
96)
at org.jf.baksmali.Adaptors.MethodDefinition.writeTo(MethodDefinition.java:198)
at
org.jf.baksmali.Adaptors.ClassDefinition.writeDirectMethods(ClassDefinition.java
:277)
at org.jf.baksmali.Adaptors.ClassDefinition.writeTo(ClassDefinition.java:112)
at org.jf.baksmali.baksmali.disassembleClass(baksmali.java:220)
at org.jf.baksmali.baksmali.access$000(baksmali.java:55)
at org.jf.baksmali.baksmali$1.call(baksmali.java:144)
at org.jf.baksmali.baksmali$1.call(baksmali.java:142)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:334)
at java.util.concurrent.FutureTask.run(FutureTask.java:166)
at
java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
at
java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
at java.lang.Thread.run(Thread.java:722)
Error occurred while disassembling class
Landroid.support.v4.view.accessibility.AccessibilityNodeInfoCompat; - skipping
class
java.lang.RuntimeException: Invalid method index: 35
at
org.jf.dexlib2.analysis.InlineMethodResolver$InlineMethodResolver_version36.reso
lveExecuteInline(InlineMethodResolver.java:175)
at
org.jf.dexlib2.analysis.MethodAnalyzer.analyzeExecuteInline(MethodAnalyzer.java:
1422)
at
org.jf.dexlib2.analysis.MethodAnalyzer.analyzeInstruction(MethodAnalyzer.java:94
1)
at org.jf.dexlib2.analysis.MethodAnalyzer.analyze(MethodAnalyzer.java:202)
at org.jf.dexlib2.analysis.MethodAnalyzer.<init>(MethodAnalyzer.java:137)
at
org.jf.baksmali.Adaptors.MethodDefinition.addAnalyzedInstructionMethodItems(Meth
odDefinition.java:387)
at
org.jf.baksmali.Adaptors.MethodDefinition.getMethodItems(MethodDefinition.java:2
96)
at org.jf.baksmali.Adaptors.MethodDefinition.writeTo(MethodDefinition.java:198)
at
org.jf.baksmali.Adaptors.ClassDefinition.writeVirtualMethods(ClassDefinition.jav
a:322)
at org.jf.baksmali.Adaptors.ClassDefinition.writeTo(ClassDefinition.java:113)
at org.jf.baksmali.baksmali.disassembleClass(baksmali.java:220)
at org.jf.baksmali.baksmali.access$000(baksmali.java:55)
at org.jf.baksmali.baksmali$1.call(baksmali.java:144)
at org.jf.baksmali.baksmali$1.call(baksmali.java:142)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:334)
at java.util.concurrent.FutureTask.run(FutureTask.java:166)
at
java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
at
java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
at java.lang.Thread.run(Thread.java:722)
Error occurred while disassembling class
Lcom.android.mms.util.ThumbnailManager$ThumbnailTask; - skipping class
java.lang.RuntimeException: Invalid method index: 32
at
org.jf.dexlib2.analysis.InlineMethodResolver$InlineMethodResolver_version36.reso
lveExecuteInline(InlineMethodResolver.java:175)
at
org.jf.dexlib2.analysis.MethodAnalyzer.analyzeExecuteInline(MethodAnalyzer.java:
1422)
at
org.jf.dexlib2.analysis.MethodAnalyzer.analyzeInstruction(MethodAnalyzer.java:94
1)
at org.jf.dexlib2.analysis.MethodAnalyzer.analyze(MethodAnalyzer.java:202)
at org.jf.dexlib2.analysis.MethodAnalyzer.<init>(MethodAnalyzer.java:137)
at
org.jf.baksmali.Adaptors.MethodDefinition.addAnalyzedInstructionMethodItems(Meth
odDefinition.java:387)
at
org.jf.baksmali.Adaptors.MethodDefinition.getMethodItems(MethodDefinition.java:2
96)
at org.jf.baksmali.Adaptors.MethodDefinition.writeTo(MethodDefinition.java:198)
at
org.jf.baksmali.Adaptors.ClassDefinition.writeDirectMethods(ClassDefinition.java
:277)
at org.jf.baksmali.Adaptors.ClassDefinition.writeTo(ClassDefinition.java:112)
at org.jf.baksmali.baksmali.disassembleClass(baksmali.java:220)
at org.jf.baksmali.baksmali.access$000(baksmali.java:55)
at org.jf.baksmali.baksmali$1.call(baksmali.java:144)
at org.jf.baksmali.baksmali$1.call(baksmali.java:142)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:334)
at java.util.concurrent.FutureTask.run(FutureTask.java:166)
at
java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
at
java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
at java.lang.Thread.run(Thread.java:722)
I've tested also with simplfied command line:
java -jar baksmali-2.0.3.jar -d \framework -x nou\Mms.apk -o nou\MMS
But I still receive the same error.
I've also tried to deodex framework.odex and the errors are similar.
Original issue reported on code.google.com by catalin....@gmail.com
on 6 Oct 2014 at 6:30
You'll need to dump the inline method table for your device, and then provide
that file via the -T option to baksmali. see baksmali -?? (yes, -?? not -?)
for more info on that option.
Original comment by jesusfreke@jesusfreke.com
on 6 Oct 2014 at 6:34
- Changed state: Done
I forgot to mention: see
https://github.com/JesusFreke/smali/tree/master/deodexerant for dumping the
inline method table.
Original comment by jesusfreke@jesusfreke.com
on 6 Oct 2014 at 6:35
Hi jesusfreke,
I've had a small adventure with deodexerant yestarday and today. I made a
virtual pc with ubuntu and prepared the AOSP android 4.4.2 and compiled
deodexerant (is the first time I do this). I put it in my device.. I run it
just like in the tutorial but into the inline.txt I receive only Failed to load
dvmGetInlineOpsTable :(
Is there some solution to this?
Original comment by catalin....@gmail.com
on 8 Oct 2014 at 7:52
You might try something like
adb pull /system/lib/libdvm.so /tmp/libdvm.so
strings /tmp/libdvm.so | grep dvmGetInlineOpsTable
If that finds something, but it's different than the symbol name that
deodexerant looks for (lines 55 and 58 in deodexerant.cc), you can try
replacing the symbol name there.
Original comment by jesusfreke@jesusfreke.com
on 8 Oct 2014 at 8:11
The result of grep is:
gDvmInlineOpsTable
_Z20dvmGetInlineOpsTablev
_Z26dvmGetInlineOpsTableLengthv
I guess gDvmInlineOpsTable won't work..
Is possible to exist a incompatiblity between the AOSP Android 4.4.2 that I
build deodexerant with.. and the chinese made one? Or perhaps some core app is
not running..
I don't know if is relevant but before compiling deodexerant I did played a
little with Titanium backup pro and tried Integrate sys Dalvik into ROM and
Undo sys Dalvik integration.
Many thanks for your help! Even if is not working I've learned some new stuff
these days.
Original comment by catalin....@gmail.com
on 8 Oct 2014 at 8:35
Based on that, it looks like it should be working. I'm not sure why it wouldn't
be.
I don't understand what you said about integrating/undo dalvik integration.
Original comment by jesusfreke@jesusfreke.com
on 8 Oct 2014 at 8:39
Regarding integrating and undo dalvik integration, I've tried these functions
of Titanium Backup Pro on my device because I've read on some various blogs
that is automatically deodexing the apps on the device.
Actually in my system/apps , there is no more odex files. But I don't think it
did actually deodexed the apps or it just moved them somewhere else. I will try
to do a factory reset and see if it helps to make the inline.txt file.
Original comment by catalin....@gmail.com
on 8 Oct 2014 at 9:15
You can also load up libdvm.so in a hex editor, and try to find the inline
method table manually. You'll want to look for the methods mentioned in
https://github.com/JesusFreke/smali/blob/master/dexlib2/src/main/java/org/jf/dex
lib2/analysis/InlineMethodResolver.java. There will likely be a handful of new
methods at the end of the table, or possibly even in the middle of the table.
Once you find the table, you could manually create an inline.txt to give to
baksmali.
Some background: one of the optimizations in an odex file is to replace some
invoke instructions with an invoke-inline instruction, with an index into the
inline table. So baksmali has to know the contents of the inline table in order
to be able to "reverse" this, and replace the invoke-inline instruction with
the original invoke instruction.
By default, baksmali knows about the inline table from "stock" dalvik, but some
OEMs choose to add or modify the inline table, in which case you have to give
the new inline table to baksmali via the inline.txt file.
What you'll be looking for in libdvm.so is the compiled gDvmInlineOpsTable
struct, which is defined in dalvik/vm/InlineNative.cpp.
Original comment by jesusfreke@jesusfreke.com
on 8 Oct 2014 at 9:40
Hi,
I think I finally extrated the table from libdvm.so.
I've used IDA Pro, it makes everything much clearer :)
But unfortunately for me, when I try do use baksmali, I receive an error:
Exception in thread "main" java.lang.NullPointerException
ineMethod<CustomInLineMethodResolver.java:120> .. screenshot attached.
Also I noticed that if I change de encoding of the file from UTF8-without BOM
to UTF8, the error is located in CustomInLineMethodResolver.java:107. It is
possible that the file could contain invalid characters?
The command line:
java -jar baksmali-2.0.3.jar -a 19 -d framework -T inline.txt -c
\framework\core.jar:\framework\conscrypt.jar:\framework\okhttp.jar:\framework\co
re-junit.jar:\framework\bouncycastle.jar:\framework\ext.jar:\framework\framework
.jar:\framework\framework2.jar:\framework\telephony-common.jar:\framework\voip-c
ommon.jar:\framework\mms-common.jar:\framework\android.policy.jar:\framework\ser
vices.jar:\framework\apache-xml.jar:\framework\webviewchromium.jar:\framework\me
diatek-common.jar:\framework\mediatek-framework.jar:\framework\CustomProperties.
jar:\framework\mediatek-telephony-common.jar:\framework\mediatek-tablet.jar -x
lucru\Mms.odex -o lucru\MMS
pause
Inline.txt also attached.
tableIDA.jpg is a screenshot of IDA at the location of the table :)
Original comment by catalin....@gmail.com
on 9 Oct 2014 at 7:05
Attachments:
Actually now I see if I save it to UTF8 the error is:
Invalid method descriptor:
?Lorg/apache/harmony/dalvik/NativeTestTarget;->emptyInlineMethod()V line
107 in CustomInLineMethodResolver.
Original comment by catalin....@gmail.com
on 9 Oct 2014 at 7:11
It looks like it might be a newline problem.. does it work if you use
unix-style newlines?
Original comment by jesusfreke@jesusfreke.com
on 9 Oct 2014 at 7:14
Oh, nevermind. The initial '?' is likely just the BOM. (don't save with BOM :)).
Let me look at the NPE, and see if I can figure out what's going on there
Original comment by jesusfreke@jesusfreke.com
on 9 Oct 2014 at 7:16
I did that now and the error is back to the line 120 which is my first error
received.
Original comment by catalin....@gmail.com
on 9 Oct 2014 at 7:19
Yeah, I still think the NPE might be related to the newlines anyway. Can you
try with unix-style newlines?
The inline.txt is typically generated by deodexerant on the device, and so it
normally has unix-style newlines.
Original comment by jesusfreke@jesusfreke.com
on 9 Oct 2014 at 7:27
Oh! Nevermind. This is a known (and fixed) issue.
https://github.com/JesusFreke/smali/commit/dd2079cd53c94056436d8c7c26df801fb210d
f4d
The fix just isn't in 2.0.3, so you'll need to build a fresh baksmali from the
current source. Sorry for the trouble! I should probably do a new release soon
:)
Original comment by jesusfreke@jesusfreke.com
on 9 Oct 2014 at 7:37
Wooooow it worked!!! :) OMG :)
Many thanks for your help! I felt like all the technology is against me,
nothing worked as expected but finally made it!
I think I should make a tutorial of how to extract the table from the
libdvm.so, but without IDA I don't think there is a chance to spot all of the
entries. I tried first without IDA, using a hexeditor, but after the first
lines in the table it started to don't make any sense.
Again! Many thanks and you did a perfect job creating these tools! Congrats!
Original comment by catalin....@gmail.com
on 9 Oct 2014 at 7:53
Great! I'm still curious why it wasn't able to load the dvmGetInlineOpsTable
method from the .so, but oh well.
Original comment by jesusfreke@jesusfreke.com
on 9 Oct 2014 at 8:00
Hi!
I was also curious about why deodexerant wasn't giving the tables and I started
to do some tests.
I've searched on google for dlsym() and I found out this:
http://pubs.opengroup.org/onlinepubs/009695399/functions/dlsym.html
And from here I come with the ideea to include some dlerror() to see what kind
of error I receive to see if I cand find something relevant.
http://pubs.opengroup.org/onlinepubs/009695399/functions/dlerror.html
After I included two error checks, suprisingly it started working!
You can see the changes in the attached deodexerant.c
So when I'm running deodexerant on the device I receive the attached inline.txt
So the first line is the first error occoured: A dynamic linking error
occurred: (undefined symbol: dvmGetInlineOpsTable)
After the error I have the table.
I can't explain why is doing this and why is not executing the second dlsym..
but I see that checking for errors with dlerror(); helped!
Original comment by catalin....@gmail.com
on 10 Oct 2014 at 6:55
Attachments:
Huh, weird. I have no idea why adding the error checking would help. It's
probably a good idea to add the error check anyway, so that deodexerant can
print a better error message when it doesn't work.
Anyway, thanks for looking into this, I'm glad you were able to get it working
:)
Original comment by jesusfreke@jesusfreke.com
on 10 Oct 2014 at 7:03
Now I read in more detail and they say:
The function dlsym() takes a "handle" of a dynamic library returned by dlopen()
and the null-terminated symbol name, returning the address where that symbol is
loaded into memory. If the symbol is not found, in the specified library or any
of the libraries that were automatically loaded by dlopen() when that library
was loaded, dlsym() returns NULL. (The search performed by dlsym() is breadth
first through the dependency tree of these libraries.) Since the value of the
symbol could actually be NULL (so that a NULL return from dlsym() need not
indicate an error), the correct way to test for an error is to call dlerror()
to clear any old error conditions, then call dlsym(), and then call dlerror()
again, saving its return value into a variable, and check whether this saved
value is not NULL.
Original comment by catalin....@gmail.com
on 10 Oct 2014 at 7:07
I found what the problem is.
I modified the code and included some printf to see if is executing the second
dlsym() call.
And I found out that the second dlsym() is not executing if the error is not
cleared before.
I guess this is how dlsym is working. I also found in the official examples
this line
dlerror(); /* Clear any existing error */
So I guess is not mandatory to check for errors but is mandatory to clear the
previous errors before executing another dlsym() call.
I hope this help :)
Original comment by catalin....@gmail.com
on 10 Oct 2014 at 7:29
Attachments: