| /* |
| * Copyright 2016, Google Inc. |
| * All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are |
| * met: |
| * |
| * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following disclaimer |
| * in the documentation and/or other materials provided with the |
| * distribution. |
| * Neither the name of Google Inc. nor the names of its |
| * contributors may be used to endorse or promote products derived from |
| * this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| package org.jf.baksmali; |
| |
| import com.beust.jcommander.JCommander; |
| import com.beust.jcommander.Parameter; |
| import com.beust.jcommander.Parameters; |
| import com.google.common.collect.Lists; |
| import org.jf.util.ConsoleUtil; |
| import org.jf.util.StringWrapper; |
| import org.jf.util.jcommander.*; |
| |
| import javax.annotation.Nonnull; |
| import java.util.List; |
| |
| @Parameters(commandDescription = "Shows usage information") |
| @ExtendedParameters( |
| commandName = "help", |
| commandAliases = "h") |
| public class HelpCommand extends Command { |
| |
| public HelpCommand(@Nonnull List<JCommander> commandAncestors) { |
| super(commandAncestors); |
| } |
| |
| @Parameter(description = "If specified, show the detailed usage information for the given commands") |
| @ExtendedParameter(argumentNames = "commands") |
| private List<String> commands = Lists.newArrayList(); |
| |
| public void run() { |
| JCommander parentJc = commandAncestors.get(commandAncestors.size() - 1); |
| |
| if (commands == null || commands.isEmpty()) { |
| System.out.println(new HelpFormatter() |
| .width(ConsoleUtil.getConsoleWidth()) |
| .format(commandAncestors)); |
| } else { |
| boolean printedHelp = false; |
| for (String cmd : commands) { |
| if (cmd.equals("register-info")) { |
| printedHelp = true; |
| String registerInfoHelp = "The --register-info parameter will cause baksmali to generate " + |
| "comments before and after every instruction containing register type " + |
| "information about some subset of registers. This parameter accepts a comma-separated list" + |
| "of values specifying which registers and how much information to include.\n" + |
| " ALL: all pre- and post-instruction registers\n" + |
| " ALLPRE: all pre-instruction registers\n" + |
| " ALLPOST: all post-instruction registers\n" + |
| " ARGS: any pre-instruction registers used as arguments to the instruction\n" + |
| " DEST: the post-instruction register used as the output of the instruction\n" + |
| " MERGE: any pre-instruction register that has been merged from multiple " + |
| "incoming code paths\n" + |
| " FULLMERGE: an extended version of MERGE that also includes a list of all " + |
| "the register types from incoming code paths that were merged"; |
| |
| Iterable<String> lines = StringWrapper.wrapStringOnBreaks(registerInfoHelp, |
| ConsoleUtil.getConsoleWidth()); |
| for (String line : lines) { |
| System.out.println(line); |
| } |
| } else if (cmd.equals("input")) { |
| printedHelp = true; |
| String registerInfoHelp = "Apks and oat files can contain multiple dex files. In order to " + |
| "specify a particular dex file, the basic syntax is to treat the apk/oat file as a " + |
| "directory. For example, to load the \"classes2.dex\" entry from \"app.apk\", you can " + |
| "use \"app.apk/classes2.dex\".\n" + |
| "\n" + |
| "For ease of use, you can also specify a partial path to the dex file to load. For " + |
| "example, to load a entry named \"/system/framework/framework.jar:classes2.dex\" from " + |
| "\"framework.oat\", you can use any of the following:\n" + |
| "\"framework.oat/classes2.dex\"\n" + |
| "\"framework.oat/framework.jar:classes2.dex\"\n" + |
| "\"framework.oat/framework/framework.jar:classes2.dex\"\n" + |
| "\"framework.oat/system/framework/framework.jar:classes2.dex\"\n" + |
| "\n" + |
| "In some rare cases, an oat file could have entries that can't be differentiated with " + |
| "the above syntax. For example \"/blah/blah.dex\" and \"blah/blah.dex\". In this case, " + |
| "the \"blah.oat/blah/blah.dex\" would match both entries and generate an error. To get " + |
| "around this, you can add double quotes around the entry name to specify an exact entry " + |
| "name. E.g. blah.oat/\"/blah/blah.dex\" or blah.oat/\"blah/blah.dex\" respectively."; |
| |
| Iterable<String> lines = StringWrapper.wrapStringOnBreaks(registerInfoHelp, |
| ConsoleUtil.getConsoleWidth()); |
| for (String line : lines) { |
| System.out.println(line); |
| } |
| } else if (cmd.equals("classpath")) { |
| printedHelp = true; |
| String registerInfoHelp = "When deodexing odex/oat files or when using the --register-info " + |
| "option, baksmali needs to load all classes from the framework files on the device " + |
| "in order to fully understand the class hierarchy. There are several options that " + |
| "control how baksmali finds and loads the classpath entries.\n" + |
| "\n"+ |
| "L+ devices (ART):\n" + |
| "When deodexing or disassembling a file from an L+ device using ART, you generally " + |
| "just need to specify the path to the boot.oat file via the --bootclasspath/-b " + |
| "parameter. On pre-N devices, the boot.oat file is self-contained and no other files are " + |
| "needed. In N, boot.oat was split into multiple files. In this case, the other " + |
| "files should be in the same directory as the boot.oat file, but you still only need to " + |
| "specify the boot.oat file in the --bootclasspath/-b option. The other files will be " + |
| "automatically loaded from the same directory.\n" + |
| "\n" + |
| "Pre-L devices (dalvik):\n" + |
| "When deodexing odex files from a pre-L device using dalvik, you " + |
| "generally just need to specify the path to a directory containing the framework files " + |
| "from the device via the --classpath-dir/-d option. odex files contain a list of " + |
| "framework files they depend on and baksmali will search for these dependencies in the " + |
| "directory that you specify.\n" + |
| "\n" + |
| "Dex files don't contain a list of dependencies like odex files, so when disassembling a " + |
| "dex file using the --register-info option, and using the framework files from a " + |
| "pre-L device, baksmali will attempt to use a reasonable default list of classpath files " + |
| "based on the api level set via the -a option. If this default list is incorrect, you " + |
| "can override the classpath using the --bootclasspath/-b option. This option accepts a " + |
| "colon separated list of classpath entries. Each entry can be specified in a few " + |
| "different ways.\n" + |
| " - A simple filename like \"framework.jar\"\n" + |
| " - A device path like \"/system/framework/framework.jar\"\n" + |
| " - A local relative or absolute path like \"/tmp/framework/framework.jar\"\n" + |
| "When using the first or second formats, you should also specify the directory " + |
| "containing the framework files via the --classpath-dir/-d option. When using the third " + |
| "format, this option is not needed.\n" + |
| "It's worth noting that the second format matches the format used by Android for the " + |
| "BOOTCLASSPATH environment variable, so you can simply grab the value of that variable " + |
| "from the device and use it as-is.\n" + |
| "\n" + |
| "Examples:\n" + |
| " For an M device:\n" + |
| " adb pull /system/framework/arm/boot.oat /tmp/boot.oat\n" + |
| " baksmali deodex blah.oat -b /tmp/boot.oat\n" + |
| " For an N+ device:\n" + |
| " adb pull /system/framework/arm /tmp/framework\n" + |
| " baksmali deodex blah.oat -b /tmp/framework/boot.oat\n" + |
| " For a pre-L device:\n" + |
| " adb pull /system/framework /tmp/framework\n" + |
| " baksmali deodex blah.odex -d /tmp/framework\n" + |
| " Using the BOOTCLASSPATH on a pre-L device:\n" + |
| " adb pull /system/framework /tmp/framework\n" + |
| " export BOOTCLASSPATH=`adb shell \"echo \\\\$BOOTCLASPATH\"`\n" + |
| " baksmali disassemble --register-info ARGS,DEST blah.apk -b $BOOTCLASSPATH -d " + |
| "/tmp/framework"; |
| |
| Iterable<String> lines = StringWrapper.wrapStringOnBreaks(registerInfoHelp, |
| ConsoleUtil.getConsoleWidth()); |
| for (String line : lines) { |
| System.out.println(line); |
| } |
| } else { |
| JCommander command = ExtendedCommands.getSubcommand(parentJc, cmd); |
| if (command == null) { |
| System.err.println("No such command: " + cmd); |
| } else { |
| printedHelp = true; |
| System.out.println(new HelpFormatter() |
| .width(ConsoleUtil.getConsoleWidth()) |
| .format(((Command)command.getObjects().get(0)).getCommandHierarchy())); |
| } |
| } |
| } |
| if (!printedHelp) { |
| System.out.println(new HelpFormatter() |
| .width(ConsoleUtil.getConsoleWidth()) |
| .format(commandAncestors)); |
| } |
| } |
| } |
| |
| @Parameters(hidden = true) |
| @ExtendedParameters(commandName = "hlep") |
| public static class HlepCommand extends HelpCommand { |
| public HlepCommand(@Nonnull List<JCommander> commandAncestors) { |
| super(commandAncestors); |
| } |
| } |
| } |