Skip to content

Commit 62017d8

Browse files
Merge pull request #30 from ShaderFrog/rename-beta
5.0.0: Rename utility functions breaking API change
2 parents 2397a8d + e61bd62 commit 62017d8

File tree

7 files changed

+250
-137
lines changed

7 files changed

+250
-137
lines changed

README.md

+76-6
Original file line numberDiff line numberDiff line change
@@ -61,15 +61,18 @@ type ParserOptions = {
6161
// like undefined functions and variables. If `failOnWarn` is set to true,
6262
// warnings will still cause the parser to raise an error. Defaults to false.
6363
quiet: boolean;
64+
6465
// An optional string representing the origin of the GLSL, for debugging and
6566
// error messages. For example, "main.js". If the parser raises an error, the
6667
// grammarSource shows up in the error.source field. If you format the error
6768
// (see the errors section), the grammarSource shows up in the formatted error
6869
// string. Defaults to undefined.
6970
grammarSource: string;
71+
7072
// If true, sets location information on each AST node, in the form of
7173
// { column: number, line: number, offset: number }. Defaults to false.
7274
includeLocation: boolean;
75+
7376
// If true, causes the parser to raise an error instead of log a warning.
7477
// The parser does limited type checking, and things like undeclared variables
7578
// are treated as warnings. Defaults to false.
@@ -427,19 +430,86 @@ visitors.
427430
428431
### Utility Functions
429432
430-
Rename all the variables in a program:
433+
#### Rename variables / identifiers in a program
434+
435+
You can rename bindings (aka variables), functions, and types (aka structs) with `renameBindings`, `renameFunctions`, and `renameTypes` respectively.
436+
437+
The signature for these methods:
438+
439+
```ts
440+
const renameBindings = (
441+
// The scope to rename the bindings in. ast.scopes[0] is the global scope.
442+
// Passing this ast.scopes[0] renames all global variables
443+
bindings: ScopeIndex,
444+
445+
// The rename function. This is called once per scope entry with the original
446+
// name in the scope, to generate the renamed variable.
447+
mangle: (name: string) => string
448+
): ScopeIndex
449+
```
450+
451+
These scope renaming functions, `renameBindings`, `renameFunctions`, and `renameTypes`, do two things:
452+
1. Each function *mutates* the AST to rename identifiers in place.
453+
2. They *return* an *immutable* new ScopeIndex where the scope references
454+
themselves are renamed.
455+
456+
If you want your ast.scopes array to stay in sync with your AST, you need to
457+
re-assign it to the output of the functions! Examples:
431458
432459
```typescript
433460
import { renameBindings, renameFunctions, renameTypes } from '@shaderfrog/glsl-parser/utils';
434461

435-
// ... parse an ast...
462+
// Suffix top level variables with _x, and update the scope
463+
ast.scopes[0].bindings = renameBindings(ast.scopes[0].bindings, (name) => `${name}_x`);
436464

437-
// Suffix top level variables with _x
438-
renameBindings(ast.scopes[0], (name, node) => `${name}_x`);
439465
// Suffix function names with _x
440-
renameFunctions(ast.scopes[0], (name, node) => `${name}_x`);
466+
ast.scopes[0].functions = renameFunctions(ast.scopes[0].functions, (name) => `${name}_x`);
467+
441468
// Suffix struct names and usages (including constructors) with _x
442-
renameTypes(ast.scopes[0], (name, node) => `${name}_x`);
469+
ast.scopes[0].types = renameTypes(ast.scopes[0].types, (name) => `${name}_x`);
470+
```
471+
472+
There are also functions to rename only one variable/identifier in a given
473+
scope. Use these if you know specifically which variable you want to rename.
474+
475+
```typescript
476+
import { renameBinding, renameFunction, renameType } from '@shaderfrog/glsl-parser/utils';
477+
478+
// Replace all instances of "oldVar" with "newVar" in the global scope, and
479+
// creates a new global scope entry named "newVar"
480+
ast.scopes[0].bindings.newVar = renameBinding(
481+
ast.scopes[0].bindings.oldVar,
482+
'newVar',
483+
);
484+
// You need to manually delete the old scope entry if you want the scope to stay
485+
// in sync with your program AST
486+
delete ast.scopes[0].bindings.oldVar;
487+
488+
// Rename a specific function
489+
ast.scopes[0].functions.newFn = renameFunction(
490+
ast.scopes[0].functions.oldFn,
491+
'newFn',
492+
);
493+
delete ast.scopes[0].functions.oldFn;
494+
495+
// Rename a specific type/struct
496+
ast.scopes[0].functions.newType = renametype(
497+
ast.scopes[0].functions.oldType,
498+
'newType',
499+
);
500+
delete ast.scopes[0].functions.oldType;
501+
```
502+
503+
#### Debugging utility functions
504+
505+
The parser also exports a debugging function, useful for logging information
506+
about the AST.
507+
508+
```ts
509+
import { debugScopes } from '@shaderfrog/glsl-parser/parser/utils';
510+
511+
// Print a condensed representation of the AST scopes to the console
512+
debugScopes(ast);
443513
```
444514
445515
## What are "parsing" and "preprocessing"?

package-lock.json

+9-9
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
"engines": {
44
"node": ">=16"
55
},
6-
"version": "4.1.1",
6+
"version": "5.0.0",
77
"type": "module",
88
"description": "A GLSL ES 1.0 and 3.0 parser and preprocessor that can preserve whitespace and comments",
99
"scripts": {
@@ -49,6 +49,6 @@
4949
"prettier": "^2.1.2",
5050
"ts-jest": "^29.1.2",
5151
"ts-jest-resolver": "^2.0.1",
52-
"typescript": "^5.3.3"
52+
"typescript": "^5.5.3"
5353
}
5454
}

src/parser/scope.test.ts

+22-9
Original file line numberDiff line numberDiff line change
@@ -176,7 +176,10 @@ float fn() {
176176
`);
177177

178178
expect(ast.scopes[0].functions.noise);
179-
renameFunctions(ast.scopes[0], (name) => `${name}_FUNCTION`);
179+
ast.scopes[0].functions = renameFunctions(
180+
ast.scopes[0].functions,
181+
(name) => `${name}_FUNCTION`
182+
);
180183
expect(generate(ast)).toBe(`
181184
float noise_FUNCTION() {}
182185
float fn_FUNCTION() {
@@ -233,8 +236,14 @@ vec4 linearToOutputTexel( vec4 value ) { return LinearToLinear( value ); }
233236
{ quiet: true }
234237
);
235238

236-
renameBindings(ast.scopes[0], (name) => `${name}_VARIABLE`);
237-
renameFunctions(ast.scopes[0], (name) => `${name}_FUNCTION`);
239+
ast.scopes[0].bindings = renameBindings(
240+
ast.scopes[0].bindings,
241+
(name) => `${name}_VARIABLE`
242+
);
243+
ast.scopes[0].functions = renameFunctions(
244+
ast.scopes[0].functions,
245+
(name) => `${name}_FUNCTION`
246+
);
238247

239248
expect(generate(ast)).toBe(`
240249
float selfref_VARIABLE, b_VARIABLE = 1.0, c_VARIABLE = selfref_VARIABLE;
@@ -306,7 +315,8 @@ StructName main(in StructName x, StructName[3] y) {
306315
float a2 = 1.0 + StructName(1.0).color.x;
307316
}
308317
`);
309-
renameTypes(ast.scopes[0], (name) => `${name}_x`);
318+
ast.scopes[0].types = renameTypes(ast.scopes[0].types, (name) => `${name}_x`);
319+
310320
expect(generate(ast)).toBe(`
311321
struct StructName_x {
312322
vec3 color;
@@ -342,10 +352,10 @@ StructName_x main(in StructName_x x, StructName_x[3] y) {
342352
]);
343353
expect(Object.keys(ast.scopes[0].bindings)).toEqual(['reflectedLight']);
344354
expect(Object.keys(ast.scopes[0].types)).toEqual([
345-
'StructName',
346-
'OtherStruct',
355+
'StructName_x',
356+
'OtherStruct_x',
347357
]);
348-
expect(ast.scopes[0].types.StructName.references).toHaveLength(16);
358+
expect(ast.scopes[0].types.StructName_x.references).toHaveLength(16);
349359

350360
// Inner struct definition should be found in inner fn scope
351361
expect(Object.keys(ast.scopes[2].types)).toEqual(['StructName']);
@@ -357,7 +367,7 @@ attribute vec3 position;
357367
vec3 func(vec3 position) {
358368
return position;
359369
}`);
360-
renameBindings(ast.scopes[0], (name) =>
370+
ast.scopes[0].bindings = renameBindings(ast.scopes[0].bindings, (name) =>
361371
name === 'position' ? 'renamed' : name
362372
);
363373
// The func arg "position" shadows the global binding, it should be untouched
@@ -378,7 +388,10 @@ uniform vec2 vProp;
378388
};`);
379389

380390
// This shouldn't crash - see the comment block in renameBindings()
381-
renameBindings(ast.scopes[0], (name) => `${name}_x`);
391+
ast.scopes[0].bindings = renameBindings(
392+
ast.scopes[0].bindings,
393+
(name) => `${name}_x`
394+
);
382395
expect(generate(ast)).toBe(`
383396
layout(std140,column_major) uniform;
384397
float a_x;

src/parser/scope.ts

-1
Original file line numberDiff line numberDiff line change
@@ -118,7 +118,6 @@ export const functionDeclarationSignature = (
118118
const quantifiers = specifier.quantifier || [];
119119

120120
const parameterTypes = proto?.parameters?.map(({ specifier, quantifier }) => {
121-
// todo: saving place on putting quantifiers here
122121
const quantifiers =
123122
// vec4[1][2] param
124123
specifier.quantifier ||

src/parser/test-helpers.ts

-31
Original file line numberDiff line numberDiff line change
@@ -67,37 +67,6 @@ export const buildParser = () => {
6767
// }
6868
// };
6969

70-
export const debugEntry = (bindings: ScopeIndex) => {
71-
return Object.entries(bindings).map(
72-
([k, v]) =>
73-
`${k}: (${v.references.length} references, ${
74-
v.declaration ? '' : 'un'
75-
}declared): ${v.references.map((r) => r.type).join(', ')}`
76-
);
77-
};
78-
export const debugFunctionEntry = (bindings: FunctionScopeIndex) =>
79-
Object.entries(bindings).flatMap(([name, overloads]) =>
80-
Object.entries(overloads).map(
81-
([signature, overload]) =>
82-
`${name} (${signature}): (${overload.references.length} references, ${
83-
overload.declaration ? '' : 'un'
84-
}declared): ${overload.references.map((r) => r.type).join(', ')}`
85-
)
86-
);
87-
88-
export const debugScopes = (astOrScopes: Program | Scope[]) =>
89-
console.log(
90-
'Scopes:',
91-
'scopes' in astOrScopes
92-
? astOrScopes.scopes
93-
: astOrScopes.map((s) => ({
94-
name: s.name,
95-
types: debugEntry(s.types),
96-
bindings: debugEntry(s.bindings),
97-
functions: debugFunctionEntry(s.functions),
98-
}))
99-
);
100-
10170
const middle = /\/\* start \*\/((.|[\r\n])+)(\/\* end \*\/)?/m;
10271

10372
type ParseSrc = (src: string, options?: ParserOptions) => Program;

0 commit comments

Comments
 (0)