Skip to content

Call register_assignment_scope() in visit_export_default_expr() and visit_export_default_decl()#94581

Open
sampoder wants to merge 1 commit into
canaryfrom
sp/turbopack/register_assignment_scope
Open

Call register_assignment_scope() in visit_export_default_expr() and visit_export_default_decl()#94581
sampoder wants to merge 1 commit into
canaryfrom
sp/turbopack/register_assignment_scope

Conversation

@sampoder

@sampoder sampoder commented Jun 9, 2026

Copy link
Copy Markdown
Member

Talked IRL, we found that calling this in visit_export_default_expr() cut down the module-cost benchmark on ESM execution by about 10ms (approx. 35%).

Doing this unlocks a codegen optimization to avoid arrow functions, this special case was needed to support export default ... which is a kind of declaration all on its own.

@sampoder sampoder requested a review from lukesandberg June 9, 2026 00:28
@github-actions

github-actions Bot commented Jun 9, 2026

Copy link
Copy Markdown
Contributor

Failing test suites

Commit: ab16b36 | About building and testing Next.js

pnpm test-start test/production/app-dir/metadata-static-route-cache/metadata-static-route-cache.test.ts (job)

  • app dir - metadata static routes cache > should generate different content after replace the static metadata file (DD)
Expand output

● app dir - metadata static routes cache › should generate different content after replace the static metadata file

next already started

  65 |   ) {
  66 |     if (this.childProcess) {
> 67 |       throw new Error('next already started')
     |             ^
  68 |     }
  69 |
  70 |     this._cliOutput = ''

  at NextStartInstance.start (lib/next-modes/next-start.ts:67:13)
  at Object.start (production/app-dir/metadata-static-route-cache/metadata-static-route-cache.test.ts:17:16)

Other failing CI jobs

@github-actions

github-actions Bot commented Jun 9, 2026

Copy link
Copy Markdown
Contributor

Stats from current PR

🟢 1 improvement

Metric Canary PR Change Trend
node_modules Size 510 MB 509 MB 🟢 470 kB (0%) ▁▁▁▁█
📊 All Metrics
📖 Metrics Glossary

Dev Server Metrics:

  • Listen = TCP port starts accepting connections
  • First Request = HTTP server returns successful response
  • Cold = Fresh build (no cache)
  • Warm = With cached build artifacts

Build Metrics:

  • Fresh = Clean build (no .next directory)
  • Cached = With existing .next directory

Change Thresholds:

  • Time: Changes < 50ms AND < 10%, OR < 2% are insignificant
  • Size: Changes < 1KB AND < 1% are insignificant
  • All other changes are flagged to catch regressions

⚡ Dev Server

Metric Canary PR Change Trend
Cold (Listen) 811ms 811ms ▂█▂▅▂
Cold (Ready in log) 795ms 791ms ▃█▅▄▄
Cold (First Request) 1.230s 1.222s ▃█▃▄▃
Warm (Listen) 811ms 811ms ▃█▃▅▁
Warm (Ready in log) 792ms 792ms ▄█▆▇▄
Warm (First Request) 613ms 611ms ▄█▆█▅
📦 Dev Server (Webpack) (Legacy)

📦 Dev Server (Webpack)

Metric Canary PR Change Trend
Cold (Listen) 810ms 810ms █████
Cold (Ready in log) 792ms 794ms █▇███
Cold (First Request) 3.269s 3.308s ▇▅▅▇▆
Warm (Listen) 809ms 810ms █████
Warm (Ready in log) 791ms 793ms █▇███
Warm (First Request) 3.296s 3.332s ▇▆▆▇▇

⚡ Production Builds

Metric Canary PR Change Trend
Fresh Build 5.048s 5.130s ▁█▁▃▂
Cached Build 5.073s 5.092s ▁█▁▃▂
📦 Production Builds (Webpack) (Legacy)

📦 Production Builds (Webpack)

Metric Canary PR Change Trend
Fresh Build 24.109s 24.163s ▆▆▆▇▆
Cached Build 24.305s 24.234s ▇▆▆▇▇
node_modules Size 510 MB 509 MB 🟢 470 kB (0%) ▁▁▁▁█
📦 Bundle Sizes

Bundle Sizes

⚡ Turbopack

Client

Main Bundles
Canary PR Change
0_mmk-hpeq702.js gzip 13.6 kB N/A -
01n68n8hhw5d_.js gzip 13.1 kB N/A -
07hsc72npcvsw.js gzip 154 B N/A -
091lqj7mftx18.js gzip 9.44 kB N/A -
0bb0vtp8dwsez.js gzip 450 B N/A -
0cz1d0mv5g_q7.js gzip 39.4 kB 39.4 kB
0et1w1og4qe-w.js gzip 8.77 kB N/A -
0g142pw4-v76e.js gzip 157 B N/A -
0gka33kfmylew.js gzip 5.73 kB N/A -
0kg6p17bcw-mj.js gzip 156 B N/A -
0nxqz3q165m2l.js gzip 220 B N/A -
0q9arq1x6yhfz.js gzip 153 B N/A -
0qtv-fy3-6-d2.js gzip 156 B N/A -
106k2vntvwwyq.js gzip 8.71 kB N/A -
15wrg2j5h1l9b.js gzip 157 B N/A -
17h18zxt1915l.js gzip 1.46 kB N/A -
19f-38ee1pg1h.js gzip 10.6 kB N/A -
1elt1qium-r2m.css gzip 115 B 115 B
1kbf9bipowi5-.js gzip 157 B N/A -
1kxsy26us_qta.js gzip 159 B N/A -
1mpn9k5m63u2c.js gzip 8.82 kB N/A -
1zdvc4dx3ssq2.js gzip 65.6 kB N/A -
1zuz4qcpo6_iw.js gzip 233 B N/A -
2-dajsb75845z.js gzip 71 kB N/A -
26xvqnur0qzto.js gzip 2.29 kB N/A -
2cig-t0drayp7.js gzip 8.78 kB N/A -
2f26of93kwaq2.js gzip 8.71 kB N/A -
2mzq6p8iw2eux.js gzip 157 B N/A -
2nygym_m0g8q8.js gzip 155 B N/A -
2puwozfp1b6z1.js gzip 8.77 kB N/A -
2taxvkjo_52n1.js gzip 13 kB N/A -
2tta7i8-pm3y1.js gzip 169 B N/A -
2ugfqzz6jnlhg.js gzip 10.3 kB N/A -
2w_tv3ndkgyn0.js gzip 8.79 kB N/A -
2yv4z4rbdjeyl.js gzip 157 B N/A -
34sv9yx51zdzn.js gzip 10 kB N/A -
3832jw9t4kfff.js gzip 8.79 kB N/A -
3l16lwwz30mdq.js gzip 52.1 kB N/A -
3l2-s66d983z-.js gzip 7.61 kB N/A -
3vj9u_3qs3jqn.js gzip 13.9 kB N/A -
434o2iahatt03.js gzip 163 B N/A -
turbopack-00.._wa6.js gzip 3.51 kB N/A -
turbopack-0k..itvk.js gzip 3.52 kB N/A -
turbopack-0l..-p8_.js gzip 3.51 kB N/A -
turbopack-1-..6c2m.js gzip 3.53 kB N/A -
turbopack-1o..y-z6.js gzip 3.51 kB N/A -
turbopack-1o..va5i.js gzip 3.51 kB N/A -
turbopack-1u..-b1t.js gzip 3.52 kB N/A -
turbopack-23..c9ws.js gzip 3.51 kB N/A -
turbopack-24..ztg0.js gzip 3.51 kB N/A -
turbopack-36..p6ho.js gzip 3.51 kB N/A -
turbopack-3p..a8i9.js gzip 3.51 kB N/A -
turbopack-3r..uewq.js gzip 3.51 kB N/A -
turbopack-3r..uztm.js gzip 3.5 kB N/A -
turbopack-3y..e197.js gzip 3.51 kB N/A -
0_jduth67n2x7.js gzip N/A 158 B -
00tc16j5-eos9.js gzip N/A 1.46 kB -
042_tc716h8nr.js gzip N/A 155 B -
09zhpvggrs_ox.js gzip N/A 13.9 kB -
0ftf54asi5ajl.js gzip N/A 9.43 kB -
0j-dsdvj1vveo.js gzip N/A 8.76 kB -
0k6he6bdlhbhw.js gzip N/A 168 B -
0r_s07d76zskm.js gzip N/A 65.6 kB -
0y1lf4y5sa0bs.js gzip N/A 152 B -
13_72k_9edikr.js gzip N/A 158 B -
170t10ub162wk.js gzip N/A 5.74 kB -
1bphlaoc0oqjw.js gzip N/A 8.78 kB -
1e5p7c65hhjno.js gzip N/A 8.71 kB -
1fnkwntx2ib4c.js gzip N/A 156 B -
1kon91zndf-mv.js gzip N/A 8.79 kB -
1lft2ycaowfp8.js gzip N/A 161 B -
1llqf2qgac_-d.js gzip N/A 8.7 kB -
1lphldovrrilz.js gzip N/A 8.77 kB -
1pcl05db7ju7l.js gzip N/A 156 B -
1ro4uyx4xttfe.js gzip N/A 155 B -
1wm4l58y3tgrj.js gzip N/A 156 B -
1ytehjj6sa09e.js gzip N/A 158 B -
22geiudvi1dh6.js gzip N/A 2.29 kB -
23p7wy6zx0abb.js gzip N/A 10.3 kB -
2dwxs6nb7x820.js gzip N/A 71 kB -
2ee39g8qw0kdi.js gzip N/A 10 kB -
2ijnnfte3p-n6.js gzip N/A 51.8 kB -
2j007o3k44oed.js gzip N/A 8.76 kB -
2n266fk2lkrnp.js gzip N/A 155 B -
2uczrmaydhz57.js gzip N/A 159 B -
2vhmp74csw0ql.js gzip N/A 232 B -
2vn0l6vcof8b6.js gzip N/A 8.81 kB -
30h_m0irj-q6p.js gzip N/A 221 B -
38c63ct4i1cy7.js gzip N/A 449 B -
39kshdfkb5_dc.js gzip N/A 12.9 kB -
3bxtggll2q95f.js gzip N/A 13.1 kB -
3kzc9t3aw7glu.js gzip N/A 7.61 kB -
3xjvvbpazem9e.js gzip N/A 10.6 kB -
438815bhbzqil.js gzip N/A 13.6 kB -
turbopack-01..yid2.js gzip N/A 3.63 kB -
turbopack-0n..5bll.js gzip N/A 3.63 kB -
turbopack-18..1kjr.js gzip N/A 3.63 kB -
turbopack-1q..lc9a.js gzip N/A 3.61 kB -
turbopack-2-..kq19.js gzip N/A 3.63 kB -
turbopack-20..7qij.js gzip N/A 3.63 kB -
turbopack-2q..x4cx.js gzip N/A 3.63 kB -
turbopack-2r..g14u.js gzip N/A 3.63 kB -
turbopack-2x..8f06.js gzip N/A 3.63 kB -
turbopack-2y..be6q.js gzip N/A 3.63 kB -
turbopack-2z..yr3f.js gzip N/A 3.63 kB -
turbopack-3q.._n30.js gzip N/A 3.63 kB -
turbopack-3v..b1kt.js gzip N/A 3.63 kB -
turbopack-3w..4pb9.js gzip N/A 3.64 kB -
Total 461 kB 463 kB ⚠️ +1.22 kB

Server

Middleware
Canary PR Change
middleware-b..fest.js gzip 728 B 726 B
Total 728 B 726 B ✅ -2 B
Build Details
Build Manifests
Canary PR Change
_buildManifest.js gzip 432 B 429 B
Total 432 B 429 B ✅ -3 B

📦 Webpack

Client

Main Bundles
Canary PR Change
2486.HASH.js gzip 169 B N/A -
3146-HASH.js gzip 62 kB N/A -
39fcf99b-HASH.js gzip 62.8 kB N/A -
8443-HASH.js gzip 4.69 kB N/A -
9431-HASH.js gzip 5.63 kB N/A -
framework-HASH.js gzip 59.8 kB 59.8 kB
main-app-HASH.js gzip 255 B 254 B
main-HASH.js gzip 39.4 kB 39.8 kB 🔴 +432 B (+1%)
webpack-HASH.js gzip 1.68 kB 1.68 kB
6105-HASH.js gzip N/A 5.64 kB -
764.HASH.js gzip N/A 169 B -
8898-HASH.js gzip N/A 61.3 kB -
9597-HASH.js gzip N/A 4.67 kB -
e1ccab69-HASH.js gzip N/A 62.8 kB -
Total 236 kB 236 kB ✅ -264 B
Polyfills
Canary PR Change
polyfills-HASH.js gzip 39.4 kB 39.4 kB
Total 39.4 kB 39.4 kB
Pages
Canary PR Change
_app-HASH.js gzip 194 B 194 B
_error-HASH.js gzip 183 B 182 B
css-HASH.js gzip 335 B 335 B
dynamic-HASH.js gzip 1.8 kB 1.8 kB
edge-ssr-HASH.js gzip 255 B 254 B
head-HASH.js gzip 351 B 349 B
hooks-HASH.js gzip 384 B 384 B
image-HASH.js gzip 580 B 581 B
index-HASH.js gzip 259 B 259 B
link-HASH.js gzip 2.53 kB 2.52 kB
routerDirect..HASH.js gzip 319 B 319 B
script-HASH.js gzip 386 B 386 B
withRouter-HASH.js gzip 315 B 313 B
1afbb74e6ecf..834.css gzip 106 B 106 B
Total 7.99 kB 7.99 kB ✅ -7 B

Server

Edge SSR
Canary PR Change
edge-ssr.js gzip 126 kB 126 kB
page.js gzip 280 kB 280 kB
Total 406 kB 407 kB ⚠️ +578 B
Middleware
Canary PR Change
middleware-b..fest.js gzip 616 B 615 B
middleware-r..fest.js gzip 156 B 155 B
middleware.js gzip 45 kB 44.8 kB
edge-runtime..pack.js gzip 842 B 842 B
Total 46.6 kB 46.4 kB ✅ -273 B
Build Details
Build Manifests
Canary PR Change
_buildManifest.js gzip 719 B 717 B
Total 719 B 717 B ✅ -2 B
Build Cache
Canary PR Change
0.pack gzip 4.56 MB 4.54 MB 🟢 14.6 kB (0%)
index.pack gzip 114 kB 115 kB 🔴 +1.89 kB (+2%)
index.pack.old gzip 114 kB 116 kB 🔴 +2.18 kB (+2%)
Total 4.78 MB 4.77 MB ✅ -10.6 kB

🔄 Shared (bundler-independent)

Runtimes
Canary PR Change
app-page-exp...dev.js gzip 356 kB 356 kB
app-page-exp..prod.js gzip 197 kB 197 kB
app-page-tur...dev.js gzip 356 kB 355 kB
app-page-tur..prod.js gzip 197 kB 197 kB
app-page-tur...dev.js gzip 353 kB 352 kB
app-page-tur..prod.js gzip 195 kB 195 kB
app-page.run...dev.js gzip 353 kB 352 kB
app-page.run..prod.js gzip 195 kB 195 kB
app-route-ex...dev.js gzip 77.8 kB 77.8 kB
app-route-ex..prod.js gzip 53.1 kB 53.1 kB
app-route-tu...dev.js gzip 77.8 kB 77.8 kB
app-route-tu..prod.js gzip 53.1 kB 53.1 kB
app-route-tu...dev.js gzip 77.4 kB 77.4 kB
app-route-tu..prod.js gzip 52.9 kB 52.9 kB
app-route.ru...dev.js gzip 77.4 kB 77.4 kB
app-route.ru..prod.js gzip 52.8 kB 52.8 kB
dist_client_...dev.js gzip 324 B 324 B
dist_client_...dev.js gzip 326 B 326 B
dist_client_...dev.js gzip 318 B 318 B
dist_client_...dev.js gzip 317 B 317 B
pages-api-tu...dev.js gzip 44.3 kB 44.3 kB
pages-api-tu..prod.js gzip 33.8 kB 33.8 kB
pages-api.ru...dev.js gzip 44.3 kB 44.3 kB
pages-api.ru..prod.js gzip 33.7 kB 33.7 kB
pages-turbo....dev.js gzip 53.7 kB 53.7 kB
pages-turbo...prod.js gzip 39.4 kB 39.4 kB
pages.runtim...dev.js gzip 53.6 kB 53.6 kB
pages.runtim..prod.js gzip 39.4 kB 39.4 kB
server.runti..prod.js gzip 63.4 kB 63.4 kB
use-cache-pr...dev.js gzip 70.6 kB 70.6 kB
use-cache-pr...dev.js gzip 70.7 kB 70.7 kB
use-cache-pr...dev.js gzip 69 kB 69 kB
use-cache-pr...dev.js gzip 69 kB 69 kB
Total 3.41 MB 3.41 MB ✅ -4.26 kB
📝 Changed Files (8 files)

Files with changes:

  • app-page-exp..ntime.dev.js
  • app-page-exp..time.prod.js
  • app-page-tur..ntime.dev.js
  • app-page-tur..time.prod.js
  • app-page-tur..ntime.dev.js
  • app-page-tur..time.prod.js
  • app-page.runtime.dev.js
  • app-page.runtime.prod.js
View diffs
app-page-exp..ntime.dev.js
failed to diff
app-page-exp..time.prod.js
failed to diff
app-page-tur..ntime.dev.js
failed to diff
app-page-tur..time.prod.js
failed to diff
app-page-tur..ntime.dev.js
failed to diff
app-page-tur..time.prod.js
failed to diff
app-page.runtime.dev.js
failed to diff
app-page.runtime.prod.js
failed to diff
📎 Tarball URL
https://vercel-packages.vercel.app/next/commits/ab16b363d1604cabce40b5c37537e88a215be5d5/next

Commit: ab16b36

@mischnic

mischnic commented Jun 9, 2026

Copy link
Copy Markdown
Member

Doing this unlocks a codegen optimization to avoid arrow functions, this special case was needed to support export default ... which is a kind of declaration all on its own.

Can we add a test case for that? Or does this just unblock some future change?

…nd `visit_export_default_decl()`

Registering an assignment scope for the synthetic default-export binding marks
it as constant-after-eval, so codegen can emit `export default <expr>` as a
plain value ("default", 0, ...) instead of a live getter arrow (() => ...) for
non-circuit-breaker modules. This avoids an arrow closure per default export.

Adds a snapshot test covering the value-form codegen.
@sampoder sampoder force-pushed the sp/turbopack/register_assignment_scope branch from 5cd2411 to ab16b36 Compare June 9, 2026 18:08
@sampoder

sampoder commented Jun 9, 2026

Copy link
Copy Markdown
Member Author

Can we add a test case for that?

Added a snapshot test that demonstrates it!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants