-
Notifications
You must be signed in to change notification settings - Fork 254
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for Rust v0 symbol mangling scheme #491
Changes from 5 commits
9560a1b
5f7f56e
1a015f0
82d7cd1
ff9c63f
5f70af5
4e75a98
a3a836c
09f41d1
a517ff5
b7a2788
6ffa4d3
5bd2d01
85c51f9
6d2eacd
62eafd9
5b58f86
8b31d4e
c317bda
faadf24
eb6926d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,5 @@ | ||
node_modules | ||
.cache | ||
.parcel-cache | ||
dist | ||
.idea | ||
coverage | ||
|
Large diffs are not rendered by default.
This file was deleted.
This file was deleted.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
gcc |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
EMCC = emcc | ||
CFLAGS = -Os -Igcc/include -DHAVE_STDLIB_H -DHAVE_STRING_H | ||
LDFLAGS_COMMON = \ | ||
-s EXPORTED_RUNTIME_METHODS=stringToUTF8OnStack,UTF8ToString \ | ||
-s EXPORTED_FUNCTIONS=_demangle,_free \ | ||
-s MODULARIZE=1 \ | ||
-s WASM=1 \ | ||
-s FILESYSTEM=0 | ||
|
||
# To use inside speedscope, we have to disable EXPORT_ES6 as otherwise | ||
# jest complain because it doesn't support es6 module export. It doesn't happen | ||
# with .ts because they are compiled to compatible js. | ||
ifeq ($(TEST),1) | ||
LDFLAGS = $(LDFLAGS_COMMON) -s EXPORT_ES6=1 | ||
else | ||
LDFLAGS = $(LDFLAGS_COMMON) -s SINGLE_FILE=1 -s ENVIRONMENT=web | ||
endif | ||
|
||
SRC_FILES = \ | ||
gcc/libiberty/safe-ctype.c \ | ||
gcc/libiberty/rust-demangle.c \ | ||
gcc/libiberty/cp-demangle.c \ | ||
demangle.c | ||
POST_JS = demangle.post.js | ||
OUTPUT = demangle.wasm.js | ||
|
||
all: $(OUTPUT) | ||
|
||
$(OUTPUT): $(SRC_FILES) $(POST_JS) | ||
$(EMCC) $(CFLAGS) $(SRC_FILES) $(LDFLAGS) --post-js $(POST_JS) --no-entry -o $(OUTPUT) | ||
|
||
clean: | ||
rm -f $(OUTPUT) | ||
|
||
.PHONY: all clean |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
# demangle | ||
|
||
A wrapper function on top of demangling functions from the `GNU libiberty`, | ||
using emscripten. | ||
|
||
# Build dependencies | ||
|
||
Follow the official `emsdk` installation instructions: | ||
|
||
https://emscripten.org/docs/getting_started/downloads.html#installation-instructions-using-the-emsdk-recommended | ||
|
||
And make sure you have `emcc` in your PATH. | ||
|
||
# Source dependencies | ||
|
||
## GCC | ||
|
||
Make sure to fetch `gcc` sources. | ||
|
||
* `git clone https://github.com/gcc-mirror/gcc` | ||
* `git reset --hard 40754a3b9bef83bf4da0675fcb378e8cd1675602` | ||
|
||
# Build instructions | ||
|
||
`make` to produce a single CommonJS module that contains also contain the base64 encoded wasm file. | ||
`make TEST=1` to produce both a ES6 module AND the wasm file. | ||
|
||
Using `make TEST=1` produce a file that can be used by `node` for testing purposes. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
#include "gcc/include/demangle.h" | ||
|
||
#include <string.h> | ||
|
||
static char *non_microsoft_demangle(const char *mangled) { | ||
int is_itanium_symbol = strncmp(mangled, "_Z", 2) == 0; | ||
if (is_itanium_symbol) { | ||
// Note: __cxa_demangle default is DMGL_PARAMS | DMGL_TYPES | ||
return cplus_demangle_v3(mangled, DMGL_PARAMS | DMGL_TYPES); | ||
} | ||
|
||
int is_rust_symbol = strncmp(mangled, "_R", 2) == 0; | ||
if (is_rust_symbol) { | ||
// Note: rust_demangle uses only DMGL_VERBOSE and DMGL_NO_RECURSE_LIMIT, | ||
// so no need to pass any options in our case. | ||
return rust_demangle(mangled, DMGL_NO_OPTS); | ||
} | ||
|
||
return NULL; | ||
} | ||
|
||
// Logic is inspired by llvm::demangle. | ||
// It is the caller's responsibility to free the string which is returned. | ||
char *demangle(const char *mangled) { | ||
char *demangled = non_microsoft_demangle(mangled); | ||
if (demangled) { | ||
return demangled; | ||
} | ||
|
||
if (mangled[0] == '_') { | ||
demangled = non_microsoft_demangle(&mangled[1]); | ||
if (demangled) { | ||
return demangled; | ||
} | ||
} | ||
|
||
return NULL; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
/** | ||
* DO NOT USE THIS FILE DIRECTLY. | ||
* | ||
* This file is only used as --post-js of emcc. | ||
* | ||
* This file provides a higher level demangle function ready to use | ||
* in JavaScript. | ||
*/ | ||
Module['wasm_demangle'] = function(mangled) { | ||
/* | ||
* We are manually calling the lower-level generated functions | ||
* instead of using `cwrap` because we need to `free` the pointer | ||
* returned by `_demangle`. | ||
*/ | ||
const param_ptr = stringToUTF8OnStack(mangled); | ||
const result_ptr = _demangle(param_ptr); | ||
const result = UTF8ToString(result_ptr); | ||
if (result_ptr !== null && result_ptr !== undefined) { | ||
_free(result_ptr); | ||
} | ||
return result; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
import {loadDemangling} from './demangle' | ||
|
||
test('demangle', async () => { | ||
const demangle = await loadDemangling() | ||
|
||
expect(demangle('a')).toBe('a') | ||
expect(demangle('someUnobfuscatedFunction')).toBe('someUnobfuscatedFunction') | ||
|
||
// C++ mangling | ||
expect(demangle('__ZNK7Support6ColorFeqERKS0_')).toBe( | ||
'Support::ColorF::operator==(Support::ColorF const&) const', | ||
) | ||
// Running a second time to test the cache | ||
expect(demangle('__ZNK7Support6ColorFeqERKS0_')).toBe( | ||
'Support::ColorF::operator==(Support::ColorF const&) const', | ||
) | ||
|
||
// Rust v0 mangling | ||
expect(demangle('_RNvCskwGfYPst2Cb_3foo16example_function')).toBe('foo::example_function') | ||
|
||
// Rust legacy mangling | ||
expect(demangle('_ZN3std2fs8Metadata7created17h8df207f105c5d474E')).toBe( | ||
'std::fs::Metadata::created::h8df207f105c5d474', | ||
) | ||
}) | ||
cerisier marked this conversation as resolved.
Show resolved
Hide resolved
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
import createWasmDemangleModule from './demangle.wasm' | ||
|
||
const cache = new Map<string, string>() | ||
|
||
export async function loadDemangling(): Promise<(name: string) => string> { | ||
// This function converts a mangled C++ name such as "__ZNK7Support6ColorFeqERKS0_" | ||
// into a human-readable symbol (in this case "Support::ColorF::==(Support::ColorF&)") | ||
const wasmDemangleModule = await createWasmDemangleModule() | ||
return cached(wasmDemangleModule.wasm_demangle) | ||
} | ||
|
||
function cached(demangle: (name: string) => string): (name: string) => string { | ||
return (name: string): string => { | ||
let result = cache.get(name) | ||
if (result !== undefined) { | ||
name = result | ||
} else { | ||
result = demangle(name) | ||
result = result === '' ? name : result | ||
cache.set(name, result) | ||
name = result | ||
} | ||
return name | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
interface WasmDemangleModule { | ||
wasm_demangle(mangled: string): string | ||
} | ||
|
||
export default function ModuleFactory(options?: unknown): Promise<WasmDemangleModule> |
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export {loadDemangling} from './demangle' |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,6 @@ | ||
import {lastOf, KeyedSet} from './utils' | ||
import {ValueFormatter, RawValueFormatter} from './value-formatters' | ||
import {FileFormat} from './file-format-spec' | ||
const demangleCppModule = import('./demangle-cpp') | ||
|
||
export interface FrameInfo { | ||
key: string | number | ||
|
@@ -404,16 +403,20 @@ export class Profile { | |
|
||
// Demangle symbols for readability | ||
async demangle() { | ||
let demangleCpp: ((name: string) => string) | null = null | ||
let demangle: ((name: string) => string) | null = null | ||
|
||
for (let frame of this.frames) { | ||
// This function converts a mangled C++ name such as "__ZNK7Support6ColorFeqERKS0_" | ||
// into a human-readable symbol (in this case "Support::ColorF::==(Support::ColorF&)") | ||
if (frame.name.startsWith('__Z')) { | ||
if (!demangleCpp) { | ||
demangleCpp = (await demangleCppModule).demangleCpp | ||
// This function converts a mangled C++ and Rust name into a human-readable symbol. | ||
if ( | ||
frame.name.startsWith('__Z') || | ||
frame.name.startsWith('_R') || | ||
frame.name.startsWith('_Z') | ||
) { | ||
if (!demangle) { | ||
const demangleModule = await import('./demangle') | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The In this case, I think we want both the I think the right way of doing that is to eagerly do the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. From what I read, demangle is already imported eagerly by As for diff --git a/src/lib/demangle/demangle.ts b/src/lib/demangle/demangle.ts
index b69adeb..70f17a0 100644
--- a/src/lib/demangle/demangle.ts
+++ b/src/lib/demangle/demangle.ts
@@ -1,11 +1,13 @@
import createWasmDemangleModule from './demangle.wasm'
+const wasmDemangleModulePromise = createWasmDemangleModule().then((module) => module)
+
const cache = new Map<string, string>()
export async function loadDemangling(): Promise<(name: string) => string> {
// This function converts a mangled C++ name such as "__ZNK7Support6ColorFeqERKS0_"
// into a human-readable symbol (in this case "Support::ColorF::==(Support::ColorF&)")
- const wasmDemangleModule = await createWasmDemangleModule()
+ const wasmDemangleModule = await wasmDemangleModulePromise
return cached(wasmDemangleModule.wasm_demangle)
} That way, everything is eager from |
||
demangle = await demangleModule.loadDemangling() | ||
} | ||
frame.name = demangleCpp(frame.name) | ||
frame.name = demangle(frame.name) | ||
} | ||
} | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Was this necessary as part of this change? I'm not opposed per-se, but I don't like upgrading dependencies as part of PRs that introduce other changes in case they need reverting
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fully understand. I added the explanation in the commit. Sadly the emscripten generated code is a raw JavaScript file that contains features that are not supported by That version of parcel-bundled hence why I had to upgrade it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.