7 raco decompile: Decompiling Bytecode (original) (raw)

7 raco decompile: Decompiling Bytecode🔗

The raco decompile command takes the path of a bytecode file (which usually has the file extension ".zo") or a source file with an associated bytecode file (usually created with raco make) and converts the bytecode file’s content back to an approximation of Racket code. When the “bytecode” file contains machine code, as for the CS variant of Racket, then it cannot be converted back to an approximation of Racket, but installing the "disassemble" package may enable disassembly of the machine code. Decompilation is mostly useful for checking the compiler’s transformation and optimization of the source program.

The raco decompile command accepts the following command-line flags:

Changed in version 1.8: Added --no-disassemble.
Changed in version 1.9: Added --partial-fasl.

7.1 Racket CS Decompilation🔗

Decompilation of Racket CS bytecode mostly shows the structure of a module around machine-code implementations of procedures.

7.2 Racket BC Decompilation🔗

Racket BC bytecode has a structure that is close enough to Racket’s core language that it can more often be converted to an approximation of Racket code. To the degree that it can be converted back, many forms in the decompiled code have the same meanings as always, such as module, define, and lambda. Other forms and transformations are specific to the rendering of bytecode, and they reflect a specific execution model:

7.3 API for Decompiling🔗

Consumes the result of parsing bytecode and returns an S-expression (as described above) that represents the compiled code.

7.4 API for Parsing Bytecode🔗

The compiler/zo-parse module re-exportscompiler/zo-structs in addition tozo-parse.

Parses a port (typically the result of opening a ".zo" file) containing bytecode. Beware that the structure types used to represent the bytecode are subject to frequent changes across Racket versions.

The parsed bytecode is returned in a linkl-directory orlinkl-bundle structure—the latter only for the compilation of a module that contains no submodules.

Beyond the linklet bundle or directory structure, the result ofzo-parse contains linklets that depend on the machine for which the bytecode is compiled:

The rest of this section is specific to BC bytecode.

Within a linklet, the bytecode representation of an expression is closer to an S-expression than a traditional, flat control string. For example, anif form is represented by a branch structure that has three fields: a test expression, a “then” expression, and an “else” expression. Similarly, a function call is represented by anapplication structure that has a list of argument expressions.

Storage for local variables or intermediate values (such as the arguments for a function call) is explicitly specified in terms of a stack. For example, execution of an application structure reserves space on the stack for each argument result. Similarly, when a let-one structure (for a simple let) is executed, the value obtained by evaluating the right-hand side expression is pushed onto the stack, and then the body is evaluated. Local variables are always accessed as offsets from the current stack position. When a function is called, its arguments are passed on the stack. A closure is created by transferring values from the stack to a flat closure record, and when a closure is applied, the saved values are restored on the stack (though possibly in a different order and likely in a more compact layout than when they were captured).

When a sub-expression produces a value, then the stack pointer is restored to its location from before evaluating the sub-expression. For example, evaluating the right-hand size for a let-onestructure may temporarily push values onto the stack, but the stack is restored to its pre-let-one position before pushing the resulting value and continuing with the body. In addition, a tail call resets the stack pointer to the position that follows the enclosing function’s arguments, and then the tail call continues by pushing onto the stack the arguments for the tail-called function.

Values for global and module-level variables are not put directly on the stack, but instead stored in “buckets,” and an array of accessible buckets is kept on the stack. When a closure body needs to access a global variable, the closure captures and later restores the bucket array in the same way that it captured and restores a local variable. Mutable local variables are boxed similarly to global variables, but individual boxes are referenced from the stack and closures.

7.5 API for Marshaling Bytecode🔗

Consumes a representation of bytecode and writes it to out.

Consumes a representation of bytecode and generates a byte string for the marshaled bytecode.

7.6 Bytecode Representation🔗

The compiler/zo-structs library defines the bytecode structures that are produced by zo-parse and consumed bydecompile and zo-marshal.

Warning: The compiler/zo-structs library exposes internals of the Racket bytecode abstraction. Unlike other Racket libraries, compiler/zo-structs is subject to incompatible changes across Racket versions.

#:extra-constructor-name make-zo
#:prefab)

A supertype for all forms that can appear in compiled code.

7.6.1 Prefix🔗

Wraps compiled code.

Module and top-level compilation produce one or more linklets that represent independent evaluation in a specific phase. Even a single top-level expression or a module with only run-time code will generate multiple linklets to implement metadata and syntax data. A module with no submodules is represented directly by alinkl-bundle, while any other compiled form is represented by a linkl-directory.

A linklet bundle maps an integer to a linklet representing forms to evaluate at the integer-indicated phase. Symbols are mapped to metadata, such as a module’s name as compiled or a linklet implementing literal syntax objects. A linklet directory normally maps '() to the main linklet bundle for a module or a single top-level form; for a linklet directory that corresponds to a sequence of top-level forms, however, there is no “main” linklet bundle, and symbol forms of integers are used to order the linkets.

For a module with submodules, the linklet directory maps submodule paths (as lists of symbols) to linklet bundles for the corresponding submodules.

An individual linklet is represented as a linkl only if the source bytecode file was for Racket BC. A CS bytecode linklet will be represented by an opaque linklet (in the sense of linklet?from racket/linklet). A machine-independent linklet is represented as a faslable-correlated-linklet structure.

Represents a linklet, which corresponds to a module body or a top-level sequence at a single phase.

The name of a linklet is for debugging purposes, similar to the inferred name of a lambda form.

The importss list of lists describes the linklet’s imports. Each of the elements of the out list corresponds to an import source, and each element of an inner list is the symbolic name of an export from that source. The import-shapess list is in parallel to imports; it reflects optimization assumptions by the compiler that are used by the bytecode validator and checked when the linklet is instantiated.

The exports list describes the linklet’s defined names that are exported. The internals list describes additional definitions within the linket, but they are not accessible from the outside of a linklet or one of its instances; a #f can appear in place of an unreferenced internal definition that has been removed. The lifts list is an extension of internals for procedures that are lifted by the compiler; these procedures have certain properties that can be checked by the bytecode validator.

Each symbol in exports,internals, and lifts must be distinct from any other symbol in those lists. The source-names table maps symbols in exports, internals, and liftsto other symbols, potentially not distinct, that correspond to original source names for the definition. The source-namestable is used only for debugging.

When a linklet is instantiated, variables corresponding to the flattening of the lists importss, exports,internals, and lifts are placed in an array (in that order) for access via toplevel references. The initial slot is reserved for a variable-like reference that strongly retains a connection to an instance of its enclosing linklet.

The bodys list is the executable content of the linklet. The value of the last element in bodys may be returned when the linklet is instantiated, depending on the way that it’s instantiated.

The max-let-depth field indicates the maximum size of the stack that will be created by any body.

The need-instance-access? boolean indicates whether the linklet contains a toplevel for position 0. A #t is allowed (but suboptimal) if not such reference is present in the linklet body.

Represents the shape of an expected import, which should be a function having the arity specified by arity. Thepreserves-marks? field is true if calling the function is expected to leave continuation marks unchanged by the time it returns.

#:extra-constructor-name make-struct-shape #:prefab)
#:extra-constructor-name make-predicate-shape #:prefab) authentic? : boolean?
#:extra-constructor-name make-struct-type-property-shape #:prefab) has-guard? : boolean?
#:extra-constructor-name make-property-predicate-shape #:prefab)
#:extra-constructor-name make-property-accessor-shape #:prefab)
#:extra-constructor-name make-struct-other-shape #:prefab)

Represents the shape of an expected import as a structure-type binding, constructor, etc.

7.6.2 Forms and Inline Variants🔗
#:extra-constructor-name make-form
#:prefab)

A supertype for all forms that can appear in a linklet body (includingexprs), except for literals that are represented as themselves.

Represents a define-values form. Each element ofids references a defined variable in the enclosing linklet.

After rhs is evaluated, the stack is restored to its depth from before evaluating rhs.

#:extra-constructor-name make-inline-variant
#:prefab)
direct : expr?
inline : expr?

Represents a function that is bound by define-values, where the function has two variants. The first variant is used for normal calls to the function. The second may be used for cross-module inlining of the function.

7.6.3 Expressions🔗
#:extra-constructor-name make-expr
#:prefab)

A supertype for all expression forms that can appear in compiled code, except for literals that are represented as themselves.

Represents a lambda form. The name field is a name for debugging purposes. The num-params field indicates the number of arguments accepted by the procedure, not counting a rest argument; the rest? field indicates whether extra arguments are accepted and collected into a “rest” variable. Theparam-types list contains num-params symbols indicating the type of each argumet, either 'val for a normal argument, 'ref for a boxed argument (representing a mutable local variable), 'flonum for a flonum argument, or 'extflonum for an extflonum argument.

Theclosure-map field is a vector of stack positions that are captured when evaluating the lambda form to create a closure. The closure-types field provides a corresponding list of types, but no distinction is made between normal values and boxed values; also, this information is redundant, since it can be inferred by the bindings referenced though closure-map.

When a closure captures top-level or module-level variables or refers to a syntax-object constant, the variables and constants are represented in the closure by capturing a prefix (in the sense of prefix). The toplevel-map field indicates which top-level variables (i.e., linklet imports and definitions) are actually used by the closure (so that variables in a prefix can be pruned by the run-time system if they become unused) and whether any syntax objects are used (so that the syntax objects as a group can be similarly pruned). A #f value indicates either that no prefix is captured or all variables and syntax objects in the prefix should be considered used. Otherwise, numbers in the set indicate which variables and lifted variables are used. Variables are numbered consecutively by position in the prefix starting from0, but the number equal to the number of non-lifted variables corresponds to syntax objects (i.e., the number is include if any syntax-object constant is used). Lifted variables are numbered immediately afterward—which means that, if the prefix contains any syntax objects, lifted-variable numbers are shifted down relative to atoplevel by the number of syntax object in the prefix (which makes the toplevel-map set more compact).

When the function is called, the rest-argument list (if any) is pushed onto the stack, then the normal arguments in reverse order, then the closure-captured values in reverse order. Thus, when body is run, the first value on the stack is the first value captured by theclosure-map array, and so on.

The max-let-depth field indicates the maximum stack depth created by body plus the arguments and closure-captured values pushed onto the stack. The body field is the expression for the closure’s body.

Changed in version 6.1.1.8 of package zo-lib: Added a number totoplevel-map to indicate whether any syntax object is used, shifting numbers for lifted variables up by one if any syntax object is in the prefix.

#:extra-constructor-name make-closure
#:prefab)
code : lam?
gen-id : symbol?

A lambda form with an empty closure, which is a procedure constant. The procedure constant can appear multiple times in the graph of expressions for bytecode, and the code field can be a cycle for a recursive constant procedure; the gen-id is different for each such constant.

Represents a case-lambda form as a combination oflambda forms that are tried (in order) based on the number of arguments given.

Pushes an uninitialized slot onto the stack, evaluates rhsand puts its value into the slot, and then runs body. Iftype is not #f, then rhs must produce a value of the corresponding type, and the slot must be accessed by localrefs that expect the type. If unused? is #t, then the slot must not be used, and the value of rhs is not actually pushed onto the stack (but rhs is constrained to produce a single value).

After rhs is evaluated, the stack is restored to its depth from before evaluating rhs. Note that the new slot is created before evaluating rhs.

Pushes count uninitialized slots onto the stack and then runsbody. If boxes? is #t, then the slots are filled with boxes that contain #.

Runs rhs to obtain count results, and installs them into existing slots on the stack in order, skipping the firstpos stack positions. If boxes? is #t, then the values are put into existing boxes in the stack slots.

After rhs is evaluated, the stack is restored to its depth from before evaluating rhs.

Represents a letrec form with lambda bindings. It allocates a closure shell for each lambda form inprocs, installs each onto the stack in previously allocated slots in reverse order (so that the closure shell for the last element of procs is installed at stack position 0), fills out each shell’s closure (where each closure normally references some other just-created closures, which is possible because the shells have been installed on the stack), and then evaluates body.

Skips pos elements of the stack, setting the slot afterward to a new box containing the slot’s old value, and then runsbody. This form appears when a lambda argument is mutated using set! within its body; calling the function initially pushes the value directly on the stack, and this form boxes the value so that it can be mutated later.

Represents a local-variable reference; it accesses the value in the stack slot after the first pos slots. If unbox? is#t, the stack slot contains a box, and a value is extracted from the box. If clear? is #t, then after the value is obtained, the stack slot is cleared (to avoid retaining a reference that can prevent reclamation of the value as garbage). Ifother-clears? is #t, then some later reference to the same stack slot may clear after reading. If type is not #f, the slot is known to hold a specific type of value.

Represents a reference to an imported or defined variable within a linklet. The depth field indicates the number of stack slots to skip to reach the prefix array, and pos is the offset into the array.

When the toplevel is an expression, if both const?and ready? are #t, then the variable definitely will be defined, its value stays constant, and the constant is effectively the same for every module instantiation. If onlyconst? is #t, then the value is constant, but it may vary across instantiations. If only ready? is#t, then the variable definitely will be defined, but its value may change. If const? and ready? are both#f, then a check is needed to determine whether the variable is defined.

When the toplevel is the left-hand side fordef-values, then const? is #f. Ifready? is #t, the variable is marked as immutable after it is defined.

Represents a function call. The rator field is the expression for the function, and rands are the argument expressions. Before any of the expressions are evaluated,(length rands) uninitialized stack slots are created (to be used as temporary space).

Represents an if form.

After test is evaluated, the stack is restored to its depth from before evaluating test.

After each of key and val is evaluated, the stack is restored to its depth from before evaluating key orval.

Represents a begin form.

After each form in forms is evaluated, the stack is restored to its depth from before evaluating the form.

Represents a begin0 expression.

After each expression in seq is evaluated, the stack is restored to its depth from before evaluating the expression.

Unlike the begin0 source form, the first expression inseq is never in tail position, even if it is the only expression in the list.

Represents a #%variable-reference form. Thetoplevel field is #t if the original reference was to a constant local binding, #f if the variable reference is for (#%variable-reference) and does not refer to a specific variable, or a symbol if the variable reference refers to a primitive variable. The dummy field accesses a variable bucket that strongly references its namespace (as opposed to a normal variable bucket, which only weakly references its namespace); it can be #f.

The value of constant? is true when the toplevelfield is not #t but the referenced variable is known to be constant. The value of from-unsafe? is true when the module that created the reference was compiled in unsafe mode.

Represents a set! expression that assigns to a top-level or module-level variable. (Assignments to local variables are represented by install-value expressions.) If undef-ok? is true, the assignment to id succeeds even if id was not previously defined (see also compile-allow-set!-undefined).

After rhs is evaluated, the stack is restored to its depth from before evaluating rhs.

Represents (call-with-values (lambda () args-expr) proc), which is handled specially by the run-time system.

Represents a (call-with-immediate-continuation-mark key (lambda (arg) body) val) expression that is handled specially by the run-time system to avoid a closure allocation. One initialized slot is pushed onto the stack after expr and valare evaluated and before body is evaluated.

After each of key and val is evaluated, the stack is restored to its depth from before evaluating key orval.

Represents a direct reference to a variable imported from the run-time kernel.

7.7 Machine-Independent Linklets🔗

Warning: The compiler/faslable-correlated library exposes internals of the Racket bytecode abstraction. Unlike other Racket libraries, compiler/faslable-correlated is subject to incompatible changes across Racket versions.

Added in version 1.3 of package zo-lib.

#:extra-constructor-name make-faslable-correlated-linklet
#:prefab)
expr : any/c
name : symbol?

A faslable-correlated-linklet structure represents alinklet that has been “compiled” to machine-independent form, which just contains an S-expression representing the linklet form. The S-expression is enriched with source-location information by wrapping some nested S-expressions with faslable-correlated structures.

Since faslable-correlated-linklet is a prefab structure type, the contracts documented above for its fields are not enforced.

Wraps an S-expression e to give it a source location. The S-expression e may contain nested faslable-correlated structures, but nesting is expected only within pairs.

Since faslable-correlated is a prefab structure type, the contracts documented above for its fields are not enforced.

Recurs through e to strip away anyfaslable-correlated structures that are reachable through pairs. The given e must not contain any cycles that are reachable through pairs.