@fairyhunter13/opentui-core 0.1.112 → 0.1.114
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dev/keypress-debug-renderer.ts +148 -0
- package/dev/keypress-debug.ts +43 -0
- package/dev/print-env-vars.ts +32 -0
- package/dev/test-tmux-graphics-334.sh +68 -0
- package/dev/thai-debug-test.ts +68 -0
- package/docs/development.md +144 -0
- package/package.json +63 -51
- package/scripts/build.ts +400 -0
- package/scripts/publish.ts +60 -0
- package/src/3d/SpriteResourceManager.ts +286 -0
- package/src/3d/SpriteUtils.ts +70 -0
- package/src/3d/TextureUtils.ts +196 -0
- package/src/3d/ThreeRenderable.ts +197 -0
- package/src/3d/WGPURenderer.ts +294 -0
- package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
- package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
- package/src/3d/animation/SpriteAnimator.ts +633 -0
- package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
- package/src/3d/canvas.ts +464 -0
- package/src/3d/index.ts +12 -0
- package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
- package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
- package/src/3d/physics/physics-interface.ts +31 -0
- package/src/3d/shaders/supersampling.wgsl +201 -0
- package/src/3d.ts +3 -0
- package/src/NativeSpanFeed.ts +300 -0
- package/src/Renderable.ts +1704 -0
- package/src/__snapshots__/buffer.test.ts.snap +28 -0
- package/src/animation/Timeline.test.ts +2709 -0
- package/src/animation/Timeline.ts +598 -0
- package/src/ansi.ts +18 -0
- package/src/benchmark/attenuation-benchmark.ts +81 -0
- package/src/benchmark/colormatrix-benchmark.ts +128 -0
- package/src/benchmark/gain-benchmark.ts +80 -0
- package/src/benchmark/latest-all-bench-run.json +707 -0
- package/src/benchmark/latest-async-bench-run.json +336 -0
- package/src/benchmark/latest-default-bench-run.json +657 -0
- package/src/benchmark/latest-large-bench-run.json +707 -0
- package/src/benchmark/latest-quick-bench-run.json +207 -0
- package/src/benchmark/markdown-benchmark.ts +1796 -0
- package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
- package/src/benchmark/native-span-feed-benchmark.md +56 -0
- package/src/benchmark/native-span-feed-benchmark.ts +596 -0
- package/src/benchmark/native-span-feed-compare.ts +280 -0
- package/src/benchmark/renderer-benchmark.ts +754 -0
- package/src/benchmark/text-table-benchmark.ts +948 -0
- package/src/buffer.test.ts +291 -0
- package/src/buffer.ts +554 -0
- package/src/console.test.ts +612 -0
- package/src/console.ts +1254 -0
- package/src/edit-buffer.test.ts +1769 -0
- package/src/edit-buffer.ts +411 -0
- package/src/editor-view.test.ts +1032 -0
- package/src/editor-view.ts +284 -0
- package/src/examples/ascii-font-selection-demo.ts +245 -0
- package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
- package/src/examples/assets/concrete.png +0 -0
- package/src/examples/assets/crate.png +0 -0
- package/src/examples/assets/crate_emissive.png +0 -0
- package/src/examples/assets/forrest_background.png +0 -0
- package/src/examples/assets/hast-example.json +1018 -0
- package/src/examples/assets/heart.png +0 -0
- package/src/examples/assets/main_char_heavy_attack.png +0 -0
- package/src/examples/assets/main_char_idle.png +0 -0
- package/src/examples/assets/main_char_jump_end.png +0 -0
- package/src/examples/assets/main_char_jump_landing.png +0 -0
- package/src/examples/assets/main_char_jump_start.png +0 -0
- package/src/examples/assets/main_char_run_loop.png +0 -0
- package/src/examples/assets/roughness_map.jpg +0 -0
- package/src/examples/build.ts +115 -0
- package/src/examples/code-demo.ts +924 -0
- package/src/examples/console-demo.ts +358 -0
- package/src/examples/core-plugin-slots-demo.ts +759 -0
- package/src/examples/diff-demo.ts +701 -0
- package/src/examples/draggable-three-demo.ts +259 -0
- package/src/examples/editor-demo.ts +322 -0
- package/src/examples/extmarks-demo.ts +196 -0
- package/src/examples/focus-restore-demo.ts +310 -0
- package/src/examples/fonts.ts +245 -0
- package/src/examples/fractal-shader-demo.ts +268 -0
- package/src/examples/framebuffer-demo.ts +674 -0
- package/src/examples/full-unicode-demo.ts +241 -0
- package/src/examples/golden-star-demo.ts +933 -0
- package/src/examples/grayscale-buffer-demo.ts +249 -0
- package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
- package/src/examples/index.ts +926 -0
- package/src/examples/input-demo.ts +377 -0
- package/src/examples/input-select-layout-demo.ts +425 -0
- package/src/examples/install.sh +143 -0
- package/src/examples/keypress-debug-demo.ts +452 -0
- package/src/examples/lib/HexList.ts +122 -0
- package/src/examples/lib/PaletteGrid.ts +125 -0
- package/src/examples/lib/standalone-keys.ts +25 -0
- package/src/examples/lib/tab-controller.ts +243 -0
- package/src/examples/lights-phong-demo.ts +290 -0
- package/src/examples/link-demo.ts +220 -0
- package/src/examples/live-state-demo.ts +480 -0
- package/src/examples/markdown-demo.ts +725 -0
- package/src/examples/mouse-interaction-demo.ts +428 -0
- package/src/examples/nested-zindex-demo.ts +357 -0
- package/src/examples/opacity-example.ts +235 -0
- package/src/examples/opentui-demo.ts +1057 -0
- package/src/examples/physx-planck-2d-demo.ts +623 -0
- package/src/examples/physx-rapier-2d-demo.ts +655 -0
- package/src/examples/relative-positioning-demo.ts +323 -0
- package/src/examples/scroll-example.ts +214 -0
- package/src/examples/scrollbox-mouse-test.ts +112 -0
- package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
- package/src/examples/select-demo.ts +237 -0
- package/src/examples/shader-cube-demo.ts +1015 -0
- package/src/examples/simple-layout-example.ts +591 -0
- package/src/examples/slider-demo.ts +617 -0
- package/src/examples/split-mode-demo.ts +453 -0
- package/src/examples/sprite-animation-demo.ts +443 -0
- package/src/examples/sprite-particle-generator-demo.ts +486 -0
- package/src/examples/static-sprite-demo.ts +193 -0
- package/src/examples/sticky-scroll-example.ts +308 -0
- package/src/examples/styled-text-demo.ts +282 -0
- package/src/examples/tab-select-demo.ts +219 -0
- package/src/examples/terminal-title.ts +29 -0
- package/src/examples/terminal.ts +305 -0
- package/src/examples/text-node-demo.ts +416 -0
- package/src/examples/text-selection-demo.ts +377 -0
- package/src/examples/text-table-demo.ts +503 -0
- package/src/examples/text-truncation-demo.ts +481 -0
- package/src/examples/text-wrap.ts +757 -0
- package/src/examples/texture-loading-demo.ts +259 -0
- package/src/examples/timeline-example.ts +670 -0
- package/src/examples/transparency-demo.ts +400 -0
- package/src/examples/vnode-composition-demo.ts +404 -0
- package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
- package/src/index.ts +24 -0
- package/src/lib/KeyHandler.integration.test.ts +292 -0
- package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
- package/src/lib/KeyHandler.test.ts +662 -0
- package/src/lib/KeyHandler.ts +222 -0
- package/src/lib/RGBA.test.ts +984 -0
- package/src/lib/RGBA.ts +204 -0
- package/src/lib/ascii.font.ts +330 -0
- package/src/lib/border.test.ts +83 -0
- package/src/lib/border.ts +170 -0
- package/src/lib/bunfs.test.ts +27 -0
- package/src/lib/bunfs.ts +18 -0
- package/src/lib/clipboard.test.ts +41 -0
- package/src/lib/clipboard.ts +47 -0
- package/src/lib/clock.ts +35 -0
- package/src/lib/data-paths.test.ts +133 -0
- package/src/lib/data-paths.ts +109 -0
- package/src/lib/debounce.ts +106 -0
- package/src/lib/detect-links.test.ts +98 -0
- package/src/lib/detect-links.ts +56 -0
- package/src/lib/env.test.ts +228 -0
- package/src/lib/env.ts +209 -0
- package/src/lib/extmarks-history.ts +51 -0
- package/src/lib/extmarks-multiwidth.test.ts +322 -0
- package/src/lib/extmarks.test.ts +3457 -0
- package/src/lib/extmarks.ts +843 -0
- package/src/lib/fonts/block.json +405 -0
- package/src/lib/fonts/grid.json +265 -0
- package/src/lib/fonts/huge.json +741 -0
- package/src/lib/fonts/pallet.json +314 -0
- package/src/lib/fonts/shade.json +591 -0
- package/src/lib/fonts/slick.json +321 -0
- package/src/lib/fonts/tiny.json +69 -0
- package/src/lib/hast-styled-text.ts +59 -0
- package/src/lib/index.ts +21 -0
- package/src/lib/keymapping.test.ts +317 -0
- package/src/lib/keymapping.ts +115 -0
- package/src/lib/objects-in-viewport.test.ts +787 -0
- package/src/lib/objects-in-viewport.ts +153 -0
- package/src/lib/output.capture.ts +58 -0
- package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
- package/src/lib/parse.keypress-kitty.test.ts +663 -0
- package/src/lib/parse.keypress-kitty.ts +439 -0
- package/src/lib/parse.keypress.test.ts +1849 -0
- package/src/lib/parse.keypress.ts +397 -0
- package/src/lib/parse.mouse.test.ts +552 -0
- package/src/lib/parse.mouse.ts +232 -0
- package/src/lib/paste.ts +16 -0
- package/src/lib/queue.ts +65 -0
- package/src/lib/renderable.validations.test.ts +87 -0
- package/src/lib/renderable.validations.ts +83 -0
- package/src/lib/scroll-acceleration.ts +98 -0
- package/src/lib/selection.ts +240 -0
- package/src/lib/singleton.ts +28 -0
- package/src/lib/stdin-parser.test.ts +2290 -0
- package/src/lib/stdin-parser.ts +1810 -0
- package/src/lib/styled-text.ts +178 -0
- package/src/lib/terminal-capability-detection.test.ts +202 -0
- package/src/lib/terminal-capability-detection.ts +79 -0
- package/src/lib/terminal-palette.test.ts +878 -0
- package/src/lib/terminal-palette.ts +383 -0
- package/src/lib/tree-sitter/assets/README.md +118 -0
- package/src/lib/tree-sitter/assets/update.ts +334 -0
- package/src/lib/tree-sitter/assets.d.ts +9 -0
- package/src/lib/tree-sitter/cache.test.ts +273 -0
- package/src/lib/tree-sitter/client.test.ts +1165 -0
- package/src/lib/tree-sitter/client.ts +607 -0
- package/src/lib/tree-sitter/default-parsers.ts +86 -0
- package/src/lib/tree-sitter/download-utils.ts +148 -0
- package/src/lib/tree-sitter/index.ts +28 -0
- package/src/lib/tree-sitter/parser.worker.ts +1042 -0
- package/src/lib/tree-sitter/parsers-config.ts +81 -0
- package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
- package/src/lib/tree-sitter/resolve-ft.ts +189 -0
- package/src/lib/tree-sitter/types.ts +82 -0
- package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
- package/src/lib/tree-sitter-styled-text.ts +306 -0
- package/src/lib/validate-dir-name.ts +55 -0
- package/src/lib/yoga.options.test.ts +628 -0
- package/src/lib/yoga.options.ts +346 -0
- package/src/plugins/core-slot.ts +579 -0
- package/src/plugins/registry.ts +402 -0
- package/src/plugins/types.ts +46 -0
- package/src/post/effects.ts +930 -0
- package/src/post/filters.ts +489 -0
- package/src/post/matrices.ts +288 -0
- package/src/renderables/ASCIIFont.ts +219 -0
- package/src/renderables/Box.test.ts +205 -0
- package/src/renderables/Box.ts +326 -0
- package/src/renderables/Code.test.ts +2062 -0
- package/src/renderables/Code.ts +357 -0
- package/src/renderables/Diff.regression.test.ts +226 -0
- package/src/renderables/Diff.test.ts +3101 -0
- package/src/renderables/Diff.ts +1211 -0
- package/src/renderables/EditBufferRenderable.test.ts +288 -0
- package/src/renderables/EditBufferRenderable.ts +1166 -0
- package/src/renderables/FrameBuffer.ts +47 -0
- package/src/renderables/Input.test.ts +1228 -0
- package/src/renderables/Input.ts +247 -0
- package/src/renderables/LineNumberRenderable.ts +724 -0
- package/src/renderables/Markdown.ts +1393 -0
- package/src/renderables/ScrollBar.ts +422 -0
- package/src/renderables/ScrollBox.ts +883 -0
- package/src/renderables/Select.test.ts +1033 -0
- package/src/renderables/Select.ts +524 -0
- package/src/renderables/Slider.test.ts +456 -0
- package/src/renderables/Slider.ts +342 -0
- package/src/renderables/TabSelect.test.ts +197 -0
- package/src/renderables/TabSelect.ts +455 -0
- package/src/renderables/Text.selection-buffer.test.ts +123 -0
- package/src/renderables/Text.test.ts +2660 -0
- package/src/renderables/Text.ts +147 -0
- package/src/renderables/TextBufferRenderable.ts +518 -0
- package/src/renderables/TextNode.test.ts +1058 -0
- package/src/renderables/TextNode.ts +325 -0
- package/src/renderables/TextTable.test.ts +1421 -0
- package/src/renderables/TextTable.ts +1344 -0
- package/src/renderables/Textarea.ts +430 -0
- package/src/renderables/TimeToFirstDraw.ts +89 -0
- package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
- package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
- package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
- package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
- package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
- package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
- package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
- package/src/renderables/__tests__/Markdown.test.ts +2518 -0
- package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
- package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
- package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
- package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
- package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
- package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
- package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
- package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
- package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
- package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
- package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
- package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
- package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
- package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
- package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
- package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
- package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
- package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
- package/src/renderables/composition/README.md +8 -0
- package/src/renderables/composition/VRenderable.ts +32 -0
- package/src/renderables/composition/constructs.ts +127 -0
- package/src/renderables/composition/vnode.ts +289 -0
- package/src/renderables/index.ts +23 -0
- package/src/renderables/markdown-parser.ts +66 -0
- package/src/renderer.ts +2681 -0
- package/src/runtime-plugin-support.ts +39 -0
- package/src/runtime-plugin.ts +615 -0
- package/src/syntax-style.test.ts +841 -0
- package/src/syntax-style.ts +257 -0
- package/src/testing/README.md +210 -0
- package/src/testing/capture-spans.test.ts +194 -0
- package/src/testing/integration.test.ts +276 -0
- package/src/testing/manual-clock.ts +117 -0
- package/src/testing/mock-keys.test.ts +1378 -0
- package/src/testing/mock-keys.ts +457 -0
- package/src/testing/mock-mouse.test.ts +218 -0
- package/src/testing/mock-mouse.ts +247 -0
- package/src/testing/mock-tree-sitter-client.ts +73 -0
- package/src/testing/spy.ts +13 -0
- package/src/testing/test-recorder.test.ts +415 -0
- package/src/testing/test-recorder.ts +145 -0
- package/src/testing/test-renderer.ts +132 -0
- package/src/testing.ts +7 -0
- package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
- package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
- package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
- package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
- package/src/tests/allocator-stats.test.ts +38 -0
- package/src/tests/destroy-during-render.test.ts +200 -0
- package/src/tests/destroy-on-exit.fixture.ts +36 -0
- package/src/tests/destroy-on-exit.test.ts +41 -0
- package/src/tests/hover-cursor.test.ts +98 -0
- package/src/tests/native-span-feed-async.test.ts +173 -0
- package/src/tests/native-span-feed-close.test.ts +120 -0
- package/src/tests/native-span-feed-coverage.test.ts +227 -0
- package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
- package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
- package/src/tests/opacity.test.ts +123 -0
- package/src/tests/renderable.snapshot.test.ts +524 -0
- package/src/tests/renderable.test.ts +1281 -0
- package/src/tests/renderer.clock.test.ts +158 -0
- package/src/tests/renderer.console-startup.test.ts +185 -0
- package/src/tests/renderer.control.test.ts +425 -0
- package/src/tests/renderer.core-slot-binding.test.ts +952 -0
- package/src/tests/renderer.cursor.test.ts +26 -0
- package/src/tests/renderer.destroy-during-render.test.ts +147 -0
- package/src/tests/renderer.focus-restore.test.ts +257 -0
- package/src/tests/renderer.focus.test.ts +294 -0
- package/src/tests/renderer.idle.test.ts +219 -0
- package/src/tests/renderer.input.test.ts +2237 -0
- package/src/tests/renderer.kitty-flags.test.ts +195 -0
- package/src/tests/renderer.mouse.test.ts +1274 -0
- package/src/tests/renderer.palette.test.ts +629 -0
- package/src/tests/renderer.selection.test.ts +49 -0
- package/src/tests/renderer.slot-registry.test.ts +684 -0
- package/src/tests/renderer.useMouse.test.ts +47 -0
- package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
- package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
- package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
- package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
- package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
- package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
- package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
- package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
- package/src/tests/runtime-plugin-support.fixture.ts +11 -0
- package/src/tests/runtime-plugin-support.test.ts +19 -0
- package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
- package/src/tests/runtime-plugin.fixture.ts +40 -0
- package/src/tests/runtime-plugin.test.ts +354 -0
- package/src/tests/scrollbox-culling-bug.test.ts +114 -0
- package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
- package/src/tests/scrollbox-hitgrid.test.ts +909 -0
- package/src/tests/scrollbox.test.ts +1530 -0
- package/src/tests/wrap-resize-perf.test.ts +276 -0
- package/src/tests/yoga-setters.test.ts +921 -0
- package/src/text-buffer-view.test.ts +705 -0
- package/src/text-buffer-view.ts +189 -0
- package/src/text-buffer.test.ts +347 -0
- package/src/text-buffer.ts +250 -0
- package/src/types.ts +161 -0
- package/src/utils.ts +88 -0
- package/src/zig/ansi.zig +268 -0
- package/src/zig/bench/README.md +50 -0
- package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
- package/src/zig/bench/edit-buffer_bench.zig +476 -0
- package/src/zig/bench/native-span-feed_bench.zig +100 -0
- package/src/zig/bench/rope-markers_bench.zig +713 -0
- package/src/zig/bench/rope_bench.zig +514 -0
- package/src/zig/bench/styled-text_bench.zig +470 -0
- package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
- package/src/zig/bench/text-buffer-view_bench.zig +459 -0
- package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
- package/src/zig/bench/utf8_bench.zig +799 -0
- package/src/zig/bench-utils.zig +431 -0
- package/src/zig/bench.zig +217 -0
- package/src/zig/buffer-methods.zig +211 -0
- package/src/zig/buffer.zig +2281 -0
- package/src/zig/build.zig +289 -0
- package/src/zig/build.zig.zon +16 -0
- package/src/zig/edit-buffer.zig +825 -0
- package/src/zig/editor-view.zig +802 -0
- package/src/zig/event-bus.zig +13 -0
- package/src/zig/event-emitter.zig +65 -0
- package/src/zig/file-logger.zig +92 -0
- package/src/zig/grapheme.zig +599 -0
- package/src/zig/lib.zig +1854 -0
- package/src/zig/link.zig +333 -0
- package/src/zig/logger.zig +43 -0
- package/src/zig/mem-registry.zig +125 -0
- package/src/zig/native-span-feed-bench-lib.zig +7 -0
- package/src/zig/native-span-feed.zig +708 -0
- package/src/zig/renderer.zig +1393 -0
- package/src/zig/rope.zig +1220 -0
- package/src/zig/syntax-style.zig +161 -0
- package/src/zig/terminal.zig +987 -0
- package/src/zig/test.zig +72 -0
- package/src/zig/tests/README.md +18 -0
- package/src/zig/tests/buffer-methods_test.zig +1109 -0
- package/src/zig/tests/buffer_test.zig +2557 -0
- package/src/zig/tests/edit-buffer-history_test.zig +271 -0
- package/src/zig/tests/edit-buffer_test.zig +1689 -0
- package/src/zig/tests/editor-view_test.zig +3299 -0
- package/src/zig/tests/event-emitter_test.zig +249 -0
- package/src/zig/tests/grapheme_test.zig +1304 -0
- package/src/zig/tests/link_test.zig +190 -0
- package/src/zig/tests/mem-registry_test.zig +473 -0
- package/src/zig/tests/memory_leak_regression_test.zig +159 -0
- package/src/zig/tests/native-span-feed_test.zig +1264 -0
- package/src/zig/tests/renderer_test.zig +1017 -0
- package/src/zig/tests/rope-nested_test.zig +712 -0
- package/src/zig/tests/rope_fuzz_test.zig +238 -0
- package/src/zig/tests/rope_test.zig +2362 -0
- package/src/zig/tests/segment-merge.test.zig +148 -0
- package/src/zig/tests/syntax-style_test.zig +557 -0
- package/src/zig/tests/terminal_test.zig +754 -0
- package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
- package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
- package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
- package/src/zig/tests/text-buffer-segment_test.zig +320 -0
- package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
- package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
- package/src/zig/tests/text-buffer-view_test.zig +3649 -0
- package/src/zig/tests/text-buffer_test.zig +2191 -0
- package/src/zig/tests/unicode-width-map.zon +3909 -0
- package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
- package/src/zig/tests/utf8_test.zig +4057 -0
- package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
- package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
- package/src/zig/tests/word-wrap-editing_test.zig +498 -0
- package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
- package/src/zig/text-buffer-iterators.zig +499 -0
- package/src/zig/text-buffer-segment.zig +404 -0
- package/src/zig/text-buffer-view.zig +1371 -0
- package/src/zig/text-buffer.zig +1180 -0
- package/src/zig/utf8.zig +1948 -0
- package/src/zig/utils.zig +9 -0
- package/src/zig-structs.ts +261 -0
- package/src/zig.ts +3884 -0
- package/tsconfig.build.json +24 -0
- package/tsconfig.json +27 -0
- package/3d/SpriteResourceManager.d.ts +0 -74
- package/3d/SpriteUtils.d.ts +0 -13
- package/3d/TextureUtils.d.ts +0 -24
- package/3d/ThreeRenderable.d.ts +0 -40
- package/3d/WGPURenderer.d.ts +0 -61
- package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
- package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
- package/3d/animation/SpriteAnimator.d.ts +0 -124
- package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
- package/3d/canvas.d.ts +0 -44
- package/3d/index.d.ts +0 -12
- package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
- package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
- package/3d/physics/physics-interface.d.ts +0 -27
- package/3d.d.ts +0 -2
- package/3d.js +0 -34041
- package/3d.js.map +0 -155
- package/LICENSE +0 -21
- package/NativeSpanFeed.d.ts +0 -41
- package/Renderable.d.ts +0 -334
- package/animation/Timeline.d.ts +0 -126
- package/ansi.d.ts +0 -13
- package/buffer.d.ts +0 -111
- package/console.d.ts +0 -144
- package/edit-buffer.d.ts +0 -98
- package/editor-view.d.ts +0 -73
- package/index-8fks7yv1.js +0 -411
- package/index-8fks7yv1.js.map +0 -10
- package/index-egy5e2rs.js +0 -12267
- package/index-egy5e2rs.js.map +0 -42
- package/index-tse8gzh0.js +0 -20614
- package/index-tse8gzh0.js.map +0 -67
- package/index.d.ts +0 -23
- package/index.js +0 -478
- package/index.js.map +0 -9
- package/lib/KeyHandler.d.ts +0 -61
- package/lib/RGBA.d.ts +0 -25
- package/lib/ascii.font.d.ts +0 -508
- package/lib/border.d.ts +0 -51
- package/lib/bunfs.d.ts +0 -7
- package/lib/clipboard.d.ts +0 -17
- package/lib/clock.d.ts +0 -15
- package/lib/data-paths.d.ts +0 -26
- package/lib/debounce.d.ts +0 -42
- package/lib/detect-links.d.ts +0 -6
- package/lib/env.d.ts +0 -42
- package/lib/extmarks-history.d.ts +0 -17
- package/lib/extmarks.d.ts +0 -89
- package/lib/hast-styled-text.d.ts +0 -17
- package/lib/index.d.ts +0 -21
- package/lib/keymapping.d.ts +0 -25
- package/lib/objects-in-viewport.d.ts +0 -24
- package/lib/output.capture.d.ts +0 -24
- package/lib/parse.keypress-kitty.d.ts +0 -2
- package/lib/parse.keypress.d.ts +0 -26
- package/lib/parse.mouse.d.ts +0 -30
- package/lib/paste.d.ts +0 -7
- package/lib/queue.d.ts +0 -15
- package/lib/renderable.validations.d.ts +0 -12
- package/lib/scroll-acceleration.d.ts +0 -43
- package/lib/selection.d.ts +0 -63
- package/lib/singleton.d.ts +0 -7
- package/lib/stdin-parser.d.ts +0 -87
- package/lib/styled-text.d.ts +0 -63
- package/lib/terminal-capability-detection.d.ts +0 -30
- package/lib/terminal-palette.d.ts +0 -50
- package/lib/tree-sitter/assets/update.d.ts +0 -11
- package/lib/tree-sitter/client.d.ts +0 -47
- package/lib/tree-sitter/default-parsers.d.ts +0 -2
- package/lib/tree-sitter/download-utils.d.ts +0 -21
- package/lib/tree-sitter/index.d.ts +0 -8
- package/lib/tree-sitter/parser.worker.d.ts +0 -1
- package/lib/tree-sitter/parsers-config.d.ts +0 -53
- package/lib/tree-sitter/resolve-ft.d.ts +0 -5
- package/lib/tree-sitter/types.d.ts +0 -82
- package/lib/tree-sitter-styled-text.d.ts +0 -14
- package/lib/validate-dir-name.d.ts +0 -1
- package/lib/yoga.options.d.ts +0 -32
- package/parser.worker.js +0 -899
- package/parser.worker.js.map +0 -12
- package/plugins/core-slot.d.ts +0 -72
- package/plugins/registry.d.ts +0 -42
- package/plugins/types.d.ts +0 -34
- package/post/effects.d.ts +0 -147
- package/post/filters.d.ts +0 -65
- package/post/matrices.d.ts +0 -20
- package/renderables/ASCIIFont.d.ts +0 -52
- package/renderables/Box.d.ts +0 -81
- package/renderables/Code.d.ts +0 -78
- package/renderables/Diff.d.ts +0 -142
- package/renderables/EditBufferRenderable.d.ts +0 -237
- package/renderables/FrameBuffer.d.ts +0 -16
- package/renderables/Input.d.ts +0 -67
- package/renderables/LineNumberRenderable.d.ts +0 -78
- package/renderables/Markdown.d.ts +0 -185
- package/renderables/ScrollBar.d.ts +0 -77
- package/renderables/ScrollBox.d.ts +0 -124
- package/renderables/Select.d.ts +0 -115
- package/renderables/Slider.d.ts +0 -47
- package/renderables/TabSelect.d.ts +0 -96
- package/renderables/Text.d.ts +0 -36
- package/renderables/TextBufferRenderable.d.ts +0 -105
- package/renderables/TextNode.d.ts +0 -91
- package/renderables/TextTable.d.ts +0 -140
- package/renderables/Textarea.d.ts +0 -63
- package/renderables/TimeToFirstDraw.d.ts +0 -24
- package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
- package/renderables/composition/VRenderable.d.ts +0 -16
- package/renderables/composition/constructs.d.ts +0 -35
- package/renderables/composition/vnode.d.ts +0 -46
- package/renderables/index.d.ts +0 -23
- package/renderables/markdown-parser.d.ts +0 -10
- package/renderer.d.ts +0 -419
- package/runtime-plugin-support.d.ts +0 -3
- package/runtime-plugin-support.js +0 -29
- package/runtime-plugin-support.js.map +0 -10
- package/runtime-plugin.d.ts +0 -16
- package/runtime-plugin.js +0 -16
- package/runtime-plugin.js.map +0 -9
- package/syntax-style.d.ts +0 -54
- package/testing/manual-clock.d.ts +0 -17
- package/testing/mock-keys.d.ts +0 -81
- package/testing/mock-mouse.d.ts +0 -38
- package/testing/mock-tree-sitter-client.d.ts +0 -23
- package/testing/spy.d.ts +0 -7
- package/testing/test-recorder.d.ts +0 -61
- package/testing/test-renderer.d.ts +0 -23
- package/testing.d.ts +0 -6
- package/testing.js +0 -697
- package/testing.js.map +0 -15
- package/text-buffer-view.d.ts +0 -42
- package/text-buffer.d.ts +0 -67
- package/types.d.ts +0 -139
- package/utils.d.ts +0 -14
- package/zig-structs.d.ts +0 -155
- package/zig.d.ts +0 -353
- /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
|
@@ -0,0 +1,599 @@
|
|
|
1
|
+
const std = @import("std");
|
|
2
|
+
|
|
3
|
+
pub const GraphemePoolError = error{
|
|
4
|
+
OutOfMemory,
|
|
5
|
+
InvalidId,
|
|
6
|
+
WrongGeneration,
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
// Encoding flags for char buffer entries (u32)
|
|
10
|
+
// Bits 31-30: encoding type
|
|
11
|
+
// 00xxxxxxxx: direct unicode scalar value (30 bits, as-is)
|
|
12
|
+
// 10xxxxxxxx: grapheme start cell with pool ID (26 bits total payload)
|
|
13
|
+
// 11xxxxxxxx: continuation cell marker for wide/grapheme rendering
|
|
14
|
+
pub const CHAR_FLAG_GRAPHEME: u32 = 0x8000_0000;
|
|
15
|
+
pub const CHAR_FLAG_CONTINUATION: u32 = 0xC000_0000;
|
|
16
|
+
|
|
17
|
+
// For grapheme start and continuation cells:
|
|
18
|
+
// Bits 29..28: right extent (u2), Bits 27..26: left extent (u2)
|
|
19
|
+
pub const CHAR_EXT_RIGHT_SHIFT: u5 = 28;
|
|
20
|
+
pub const CHAR_EXT_LEFT_SHIFT: u5 = 26;
|
|
21
|
+
pub const CHAR_EXT_MASK: u32 = 0x3;
|
|
22
|
+
|
|
23
|
+
// Grapheme ID payload layout (26 bits total):
|
|
24
|
+
// [ class (3 bits) | generation (7 bits) | slot_index (16 bits) ]
|
|
25
|
+
pub const GRAPHEME_ID_MASK: u32 = 0x03FF_FFFF;
|
|
26
|
+
pub const CLASS_BITS: u5 = 3;
|
|
27
|
+
pub const GENERATION_BITS: u5 = 7;
|
|
28
|
+
pub const SLOT_BITS: u5 = 16;
|
|
29
|
+
pub const CLASS_MASK: u32 = (@as(u32, 1) << CLASS_BITS) - 1; // 0b111
|
|
30
|
+
pub const GENERATION_MASK: u32 = (@as(u32, 1) << GENERATION_BITS) - 1; // 0b1111111
|
|
31
|
+
pub const SLOT_MASK: u32 = (@as(u32, 1) << SLOT_BITS) - 1; // 0xFFFF
|
|
32
|
+
|
|
33
|
+
/// Global slab-allocated pool for grapheme clusters (byte slices)
|
|
34
|
+
/// This is total overkill probably, but fun
|
|
35
|
+
/// ID layout (26-bit payload):
|
|
36
|
+
/// [ class (3 bits) | generation (7 bits) | slot_index (16 bits) ]
|
|
37
|
+
pub const GraphemePool = struct {
|
|
38
|
+
const MAX_CLASSES: u5 = 5; // 0..4 => 8,16,32,64,128
|
|
39
|
+
const CLASS_SIZES = [_]u32{ 8, 16, 32, 64, 128 };
|
|
40
|
+
const DEFAULT_SLOTS_PER_PAGE = [_]u32{ 256, 128, 64, 16, 8 };
|
|
41
|
+
|
|
42
|
+
pub const IdPayload = u32;
|
|
43
|
+
|
|
44
|
+
pub const InitOptions = struct {
|
|
45
|
+
/// Slots per page for each size class. If null, uses DEFAULT_SLOTS_PER_PAGE.
|
|
46
|
+
/// Used to limit pool size for testing.
|
|
47
|
+
slots_per_page: ?[MAX_CLASSES]u32 = null,
|
|
48
|
+
};
|
|
49
|
+
|
|
50
|
+
allocator: std.mem.Allocator,
|
|
51
|
+
classes: [MAX_CLASSES]ClassPool,
|
|
52
|
+
interned_live_ids: std.StringHashMapUnmanaged(IdPayload),
|
|
53
|
+
|
|
54
|
+
const SlotHeader = extern struct {
|
|
55
|
+
len: u16,
|
|
56
|
+
refcount: u32,
|
|
57
|
+
generation: u32,
|
|
58
|
+
is_owned: u32, // 0 = unowned (external memory), 1 = owned (copied into pool)
|
|
59
|
+
};
|
|
60
|
+
|
|
61
|
+
pub fn init(allocator: std.mem.Allocator) GraphemePool {
|
|
62
|
+
return initWithOptions(allocator, .{});
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
pub fn initWithOptions(allocator: std.mem.Allocator, options: InitOptions) GraphemePool {
|
|
66
|
+
const slots_per_page = options.slots_per_page orelse DEFAULT_SLOTS_PER_PAGE;
|
|
67
|
+
var classes: [MAX_CLASSES]ClassPool = undefined;
|
|
68
|
+
var i: usize = 0;
|
|
69
|
+
while (i < MAX_CLASSES) : (i += 1) {
|
|
70
|
+
classes[i] = ClassPool.init(allocator, CLASS_SIZES[i], slots_per_page[i]);
|
|
71
|
+
}
|
|
72
|
+
return .{ .allocator = allocator, .classes = classes, .interned_live_ids = .{} };
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
pub fn deinit(self: *GraphemePool) void {
|
|
76
|
+
var key_it = self.interned_live_ids.keyIterator();
|
|
77
|
+
while (key_it.next()) |key_ptr| {
|
|
78
|
+
self.allocator.free(@constCast(key_ptr.*));
|
|
79
|
+
}
|
|
80
|
+
self.interned_live_ids.deinit(self.allocator);
|
|
81
|
+
|
|
82
|
+
var i: usize = 0;
|
|
83
|
+
while (i < MAX_CLASSES) : (i += 1) {
|
|
84
|
+
self.classes[i].deinit();
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/// removeInternedLiveId removes an interned ID from the live set if it
|
|
89
|
+
/// matches the expected ID.
|
|
90
|
+
fn removeInternedLiveId(self: *GraphemePool, bytes: []const u8, expected_id: IdPayload) void {
|
|
91
|
+
const live_id = self.interned_live_ids.get(bytes) orelse return;
|
|
92
|
+
if (live_id != expected_id) return;
|
|
93
|
+
if (self.interned_live_ids.fetchRemove(bytes)) |removed| {
|
|
94
|
+
self.allocator.free(@constCast(removed.key));
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/// lookupOrInvalidate checks if the given bytes are already interned and live, returning the existing ID if so.
|
|
99
|
+
fn lookupOrInvalidate(self: *GraphemePool, bytes: []const u8) ?IdPayload {
|
|
100
|
+
const live_id = self.interned_live_ids.get(bytes) orelse return null;
|
|
101
|
+
|
|
102
|
+
// Verify that the live ID is still valid and matches the bytes. If get
|
|
103
|
+
// fails, the ID is no longer valid, so remove it from the interned map.
|
|
104
|
+
const live_bytes = self.get(live_id) catch {
|
|
105
|
+
self.removeInternedLiveId(bytes, live_id);
|
|
106
|
+
return null;
|
|
107
|
+
};
|
|
108
|
+
|
|
109
|
+
// If the bytes don't match, this means the ID was recycled and now points
|
|
110
|
+
// to different data. Invalidate the interned ID.
|
|
111
|
+
if (!std.mem.eql(u8, live_bytes, bytes)) {
|
|
112
|
+
self.removeInternedLiveId(bytes, live_id);
|
|
113
|
+
return null;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
// check refcount > 0 to ensure the ID is still live. If refcount is 0,
|
|
117
|
+
// the slot is free but hasn't been reused yet, so we can treat it as
|
|
118
|
+
// not found.
|
|
119
|
+
const live_refcount = self.getRefcount(live_id) catch {
|
|
120
|
+
self.removeInternedLiveId(bytes, live_id);
|
|
121
|
+
return null;
|
|
122
|
+
};
|
|
123
|
+
if (live_refcount == 0) {
|
|
124
|
+
self.removeInternedLiveId(bytes, live_id);
|
|
125
|
+
return null;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
return live_id;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/// internLiveId interns the grapheme bytes.
|
|
132
|
+
fn internLiveId(self: *GraphemePool, id: IdPayload, bytes: []const u8) GraphemePoolError!void {
|
|
133
|
+
if (self.lookupOrInvalidate(bytes) != null) {
|
|
134
|
+
// Keep existing interned ID if it's still valid.
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
const owned_key = self.allocator.dupe(u8, bytes) catch return GraphemePoolError.OutOfMemory;
|
|
139
|
+
errdefer self.allocator.free(owned_key);
|
|
140
|
+
|
|
141
|
+
if (self.interned_live_ids.fetchPut(self.allocator, owned_key, id) catch return GraphemePoolError.OutOfMemory) |replaced| {
|
|
142
|
+
// A previous key allocation was replaced.
|
|
143
|
+
self.allocator.free(@constCast(replaced.key));
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
fn classForSize(size: usize) u32 {
|
|
148
|
+
if (size <= 8) return 0;
|
|
149
|
+
if (size <= 16) return 1;
|
|
150
|
+
if (size <= 32) return 2;
|
|
151
|
+
if (size <= 64) return 3;
|
|
152
|
+
return 4; // up to 128
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
fn packId(class_id: u32, slot_index: u32, generation: u32) GraphemePoolError!IdPayload {
|
|
156
|
+
if (slot_index > SLOT_MASK) return GraphemePoolError.OutOfMemory;
|
|
157
|
+
return (class_id << (GENERATION_BITS + SLOT_BITS)) |
|
|
158
|
+
((generation & GENERATION_MASK) << SLOT_BITS) |
|
|
159
|
+
(slot_index & SLOT_MASK);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
pub fn alloc(self: *GraphemePool, bytes: []const u8) GraphemePoolError!IdPayload {
|
|
163
|
+
if (self.lookupOrInvalidate(bytes)) |live_id| {
|
|
164
|
+
return live_id;
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
const class_id: u32 = classForSize(bytes.len);
|
|
168
|
+
const slot_index = try self.classes[class_id].allocInternal(bytes, true);
|
|
169
|
+
const generation = self.classes[class_id].getGeneration(slot_index);
|
|
170
|
+
return try packId(class_id, slot_index, generation);
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/// Allocate an ID for externally managed memory (no copy, just reference)
|
|
174
|
+
/// The caller is responsible for keeping the memory valid while the ID is in use
|
|
175
|
+
pub fn allocUnowned(self: *GraphemePool, bytes: []const u8) GraphemePoolError!IdPayload {
|
|
176
|
+
// For unowned allocations, we need space for a pointer
|
|
177
|
+
const ptr_size = @sizeOf(usize);
|
|
178
|
+
const class_id: u32 = classForSize(ptr_size);
|
|
179
|
+
const slot_index = try self.classes[class_id].allocInternal(bytes, false);
|
|
180
|
+
const generation = self.classes[class_id].getGeneration(slot_index);
|
|
181
|
+
return try packId(class_id, slot_index, generation);
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
pub fn incref(self: *GraphemePool, id: IdPayload) GraphemePoolError!void {
|
|
185
|
+
const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
|
|
186
|
+
if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
|
|
187
|
+
const slot_index: u32 = id & SLOT_MASK;
|
|
188
|
+
const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
|
|
189
|
+
const old_refcount = try self.classes[class_id].getRefcount(slot_index, generation);
|
|
190
|
+
try self.classes[class_id].incref(slot_index, generation);
|
|
191
|
+
|
|
192
|
+
if (old_refcount == 0) {
|
|
193
|
+
const is_owned = try self.classes[class_id].isOwned(slot_index, generation);
|
|
194
|
+
if (is_owned) {
|
|
195
|
+
// This is a transition from 0 to 1 for owned bytes, so intern it.
|
|
196
|
+
const bytes = try self.classes[class_id].get(slot_index, generation);
|
|
197
|
+
try self.internLiveId(id, bytes);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
pub fn decref(self: *GraphemePool, id: IdPayload) GraphemePoolError!void {
|
|
203
|
+
const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
|
|
204
|
+
if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
|
|
205
|
+
const slot_index: u32 = id & SLOT_MASK;
|
|
206
|
+
const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
|
|
207
|
+
|
|
208
|
+
const old_refcount = try self.classes[class_id].getRefcount(slot_index, generation);
|
|
209
|
+
if (old_refcount == 1) {
|
|
210
|
+
const is_owned = try self.classes[class_id].isOwned(slot_index, generation);
|
|
211
|
+
if (is_owned) {
|
|
212
|
+
// This is a transition from 1 to 0 for owned bytes, remove map entry.
|
|
213
|
+
const bytes = try self.classes[class_id].get(slot_index, generation);
|
|
214
|
+
self.removeInternedLiveId(bytes, id);
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
try self.classes[class_id].decref(slot_index, generation);
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
/// Free a freshly allocated slot that was never incref'd (refcount=0).
|
|
222
|
+
/// Use this for cleanup when allocation succeeded but the slot was never used.
|
|
223
|
+
/// This prevents slot leaks when an error occurs between alloc and incref.
|
|
224
|
+
pub fn freeUnreferenced(self: *GraphemePool, id: IdPayload) GraphemePoolError!void {
|
|
225
|
+
const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
|
|
226
|
+
if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
|
|
227
|
+
const slot_index: u32 = id & SLOT_MASK;
|
|
228
|
+
const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
|
|
229
|
+
|
|
230
|
+
const is_owned = try self.classes[class_id].isOwned(slot_index, generation);
|
|
231
|
+
if (is_owned) {
|
|
232
|
+
const bytes = try self.classes[class_id].get(slot_index, generation);
|
|
233
|
+
self.removeInternedLiveId(bytes, id);
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
try self.classes[class_id].freeUnreferenced(slot_index, generation);
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
pub fn get(self: *GraphemePool, id: IdPayload) GraphemePoolError![]const u8 {
|
|
240
|
+
const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
|
|
241
|
+
if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
|
|
242
|
+
const slot_index: u32 = id & SLOT_MASK;
|
|
243
|
+
const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
|
|
244
|
+
return self.classes[class_id].get(slot_index, generation);
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
pub fn getRefcount(self: *GraphemePool, id: IdPayload) GraphemePoolError!u32 {
|
|
248
|
+
const class_id: u32 = (id >> (GENERATION_BITS + SLOT_BITS)) & CLASS_MASK;
|
|
249
|
+
if (class_id >= MAX_CLASSES) return GraphemePoolError.InvalidId;
|
|
250
|
+
const slot_index: u32 = id & SLOT_MASK;
|
|
251
|
+
const generation: u32 = (id >> SLOT_BITS) & GENERATION_MASK;
|
|
252
|
+
return self.classes[class_id].getRefcount(slot_index, generation);
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
const ClassPool = struct {
|
|
256
|
+
allocator: std.mem.Allocator,
|
|
257
|
+
slot_capacity: u32,
|
|
258
|
+
slots_per_page: u32,
|
|
259
|
+
slot_size_bytes: usize,
|
|
260
|
+
slots: std.ArrayListUnmanaged(u8),
|
|
261
|
+
free_list: std.ArrayListUnmanaged(u32),
|
|
262
|
+
num_slots: u32,
|
|
263
|
+
|
|
264
|
+
pub fn init(allocator: std.mem.Allocator, slot_capacity: u32, slots_per_page: u32) ClassPool {
|
|
265
|
+
// Align slot size to SlotHeader alignment to prevent UB from misaligned access
|
|
266
|
+
const raw_slot_size = @sizeOf(SlotHeader) + slot_capacity;
|
|
267
|
+
const slot_size_bytes = std.mem.alignForward(usize, raw_slot_size, @alignOf(SlotHeader));
|
|
268
|
+
return .{
|
|
269
|
+
.allocator = allocator,
|
|
270
|
+
.slot_capacity = slot_capacity,
|
|
271
|
+
.slots_per_page = slots_per_page,
|
|
272
|
+
.slot_size_bytes = slot_size_bytes,
|
|
273
|
+
.slots = .{},
|
|
274
|
+
.free_list = .{},
|
|
275
|
+
.num_slots = 0,
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
pub fn deinit(self: *ClassPool) void {
|
|
280
|
+
self.slots.deinit(self.allocator);
|
|
281
|
+
self.free_list.deinit(self.allocator);
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
fn grow(self: *ClassPool) GraphemePoolError!void {
|
|
285
|
+
const add_bytes = self.slot_size_bytes * self.slots_per_page;
|
|
286
|
+
|
|
287
|
+
try self.slots.ensureTotalCapacity(self.allocator, self.slots.items.len + add_bytes);
|
|
288
|
+
try self.slots.appendNTimes(self.allocator, 0, add_bytes);
|
|
289
|
+
|
|
290
|
+
var i: u32 = 0;
|
|
291
|
+
while (i < self.slots_per_page) : (i += 1) {
|
|
292
|
+
try self.free_list.append(self.allocator, self.num_slots + i);
|
|
293
|
+
}
|
|
294
|
+
self.num_slots += self.slots_per_page;
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
fn slotPtr(self: *ClassPool, slot_index: u32) *u8 {
|
|
298
|
+
const offset: usize = @as(usize, slot_index) * self.slot_size_bytes;
|
|
299
|
+
return &self.slots.items[offset];
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
pub fn allocInternal(self: *ClassPool, bytes: []const u8, is_owned: bool) GraphemePoolError!u32 {
|
|
303
|
+
// Validate size for owned allocations
|
|
304
|
+
if (is_owned and bytes.len > self.slot_capacity) {
|
|
305
|
+
@panic("ClassPool.allocInternal: bytes.len > slot_capacity");
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
if (self.free_list.items.len == 0) try self.grow();
|
|
309
|
+
|
|
310
|
+
const slot_index = self.free_list.pop().?;
|
|
311
|
+
const p = self.slotPtr(slot_index);
|
|
312
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
313
|
+
|
|
314
|
+
// Increment generation when reusing a slot, wrapping at 7 bits (128 values)
|
|
315
|
+
const new_generation = (header_ptr.generation + 1) & GENERATION_MASK;
|
|
316
|
+
|
|
317
|
+
// Calculate length based on ownership
|
|
318
|
+
const len: u16 = if (is_owned) @intCast(@min(bytes.len, self.slot_capacity)) else @intCast(bytes.len);
|
|
319
|
+
|
|
320
|
+
header_ptr.* = .{
|
|
321
|
+
.len = len,
|
|
322
|
+
.refcount = 0,
|
|
323
|
+
.generation = new_generation,
|
|
324
|
+
.is_owned = if (is_owned) 1 else 0,
|
|
325
|
+
};
|
|
326
|
+
|
|
327
|
+
const data_ptr = @as([*]u8, @ptrCast(p)) + @sizeOf(SlotHeader);
|
|
328
|
+
|
|
329
|
+
if (is_owned) {
|
|
330
|
+
// Owned: copy bytes into our storage
|
|
331
|
+
@memcpy(data_ptr[0..header_ptr.len], bytes[0..header_ptr.len]);
|
|
332
|
+
} else {
|
|
333
|
+
// Unowned: store pointer to external memory
|
|
334
|
+
const ptr_storage = @as(*[*]const u8, @ptrCast(@alignCast(data_ptr)));
|
|
335
|
+
ptr_storage.* = bytes.ptr;
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
return slot_index;
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
pub fn getGeneration(self: *ClassPool, slot_index: u32) u32 {
|
|
342
|
+
if (slot_index >= self.num_slots) return 0;
|
|
343
|
+
const p = self.slotPtr(slot_index);
|
|
344
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
345
|
+
return header_ptr.generation;
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
pub fn incref(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!void {
|
|
349
|
+
const p = self.slotPtr(slot_index);
|
|
350
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
351
|
+
if (header_ptr.generation != expected_generation) {
|
|
352
|
+
// Generation mismatch - this is a stale reference
|
|
353
|
+
return GraphemePoolError.WrongGeneration;
|
|
354
|
+
}
|
|
355
|
+
header_ptr.refcount +%= 1;
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
pub fn decref(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!void {
|
|
359
|
+
const p = self.slotPtr(slot_index);
|
|
360
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
361
|
+
|
|
362
|
+
if (header_ptr.refcount == 0) return GraphemePoolError.InvalidId;
|
|
363
|
+
if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
|
|
364
|
+
|
|
365
|
+
header_ptr.refcount -%= 1;
|
|
366
|
+
|
|
367
|
+
if (header_ptr.refcount == 0) {
|
|
368
|
+
try self.free_list.append(self.allocator, slot_index);
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
/// Free a slot that has refcount=0 (freshly allocated, never incref'd).
|
|
373
|
+
/// This is used for cleanup when allocation succeeded but the caller
|
|
374
|
+
/// needs to abort before taking ownership via incref.
|
|
375
|
+
pub fn freeUnreferenced(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!void {
|
|
376
|
+
if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
|
|
377
|
+
const p = self.slotPtr(slot_index);
|
|
378
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
379
|
+
|
|
380
|
+
if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
|
|
381
|
+
if (header_ptr.refcount != 0) return GraphemePoolError.InvalidId; // Not unreferenced
|
|
382
|
+
|
|
383
|
+
try self.free_list.append(self.allocator, slot_index);
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
pub fn get(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError![]const u8 {
|
|
387
|
+
if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
|
|
388
|
+
|
|
389
|
+
const p = self.slotPtr(slot_index);
|
|
390
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
391
|
+
// Validate generation to prevent accessing stale data
|
|
392
|
+
if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
|
|
393
|
+
|
|
394
|
+
const data_ptr = @as([*]u8, @ptrCast(p)) + @sizeOf(SlotHeader);
|
|
395
|
+
|
|
396
|
+
if (header_ptr.is_owned == 1) {
|
|
397
|
+
// Owned memory: return slice from our storage
|
|
398
|
+
return data_ptr[0..header_ptr.len];
|
|
399
|
+
} else {
|
|
400
|
+
// Unowned memory: dereference stored pointer
|
|
401
|
+
const ptr_storage = @as(*[*]const u8, @ptrCast(@alignCast(data_ptr)));
|
|
402
|
+
const external_ptr = ptr_storage.*;
|
|
403
|
+
return external_ptr[0..header_ptr.len];
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
pub fn getRefcount(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!u32 {
|
|
408
|
+
if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
|
|
409
|
+
const p = self.slotPtr(slot_index);
|
|
410
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
411
|
+
if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
|
|
412
|
+
return header_ptr.refcount;
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
pub fn isOwned(self: *ClassPool, slot_index: u32, expected_generation: u32) GraphemePoolError!bool {
|
|
416
|
+
if (slot_index >= self.num_slots) return GraphemePoolError.InvalidId;
|
|
417
|
+
const p = self.slotPtr(slot_index);
|
|
418
|
+
const header_ptr = @as(*SlotHeader, @ptrCast(@alignCast(p)));
|
|
419
|
+
if (header_ptr.generation != expected_generation) return GraphemePoolError.WrongGeneration;
|
|
420
|
+
return header_ptr.is_owned == 1;
|
|
421
|
+
}
|
|
422
|
+
};
|
|
423
|
+
};
|
|
424
|
+
|
|
425
|
+
// Bit manipulation functions for encoded char values
|
|
426
|
+
|
|
427
|
+
pub fn isGraphemeChar(c: u32) bool {
|
|
428
|
+
return (c & 0xC000_0000) == CHAR_FLAG_GRAPHEME;
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
pub fn isContinuationChar(c: u32) bool {
|
|
432
|
+
return (c & 0xC000_0000) == CHAR_FLAG_CONTINUATION;
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
pub fn isClusterChar(c: u32) bool {
|
|
436
|
+
return (c & 0x8000_0000) == 0x8000_0000;
|
|
437
|
+
}
|
|
438
|
+
|
|
439
|
+
pub fn graphemeIdFromChar(c: u32) u32 {
|
|
440
|
+
return c & GRAPHEME_ID_MASK;
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
pub fn charRightExtent(c: u32) u32 {
|
|
444
|
+
return (c >> CHAR_EXT_RIGHT_SHIFT) & CHAR_EXT_MASK;
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
pub fn charLeftExtent(c: u32) u32 {
|
|
448
|
+
return (c >> CHAR_EXT_LEFT_SHIFT) & CHAR_EXT_MASK;
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
pub fn packGraphemeStart(gid: u32, total_width: u32) u32 {
|
|
452
|
+
const width_minus_one: u32 = if (total_width == 0) 0 else @intCast(@min(total_width - 1, 3));
|
|
453
|
+
const right: u32 = width_minus_one;
|
|
454
|
+
const left: u32 = 0;
|
|
455
|
+
return CHAR_FLAG_GRAPHEME |
|
|
456
|
+
((right & CHAR_EXT_MASK) << CHAR_EXT_RIGHT_SHIFT) |
|
|
457
|
+
((left & CHAR_EXT_MASK) << CHAR_EXT_LEFT_SHIFT) |
|
|
458
|
+
(gid & GRAPHEME_ID_MASK);
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
pub fn packContinuation(left: u32, right: u32, gid: u32) u32 {
|
|
462
|
+
return CHAR_FLAG_CONTINUATION |
|
|
463
|
+
((@min(left, 3) & CHAR_EXT_MASK) << CHAR_EXT_LEFT_SHIFT) |
|
|
464
|
+
((@min(right, 3) & CHAR_EXT_MASK) << CHAR_EXT_RIGHT_SHIFT) |
|
|
465
|
+
(gid & GRAPHEME_ID_MASK);
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
pub fn encodedCharWidth(c: u32) u32 {
|
|
469
|
+
if (isContinuationChar(c)) {
|
|
470
|
+
const left = charLeftExtent(c);
|
|
471
|
+
const right = charRightExtent(c);
|
|
472
|
+
return left + 1 + right;
|
|
473
|
+
} else if (isGraphemeChar(c)) {
|
|
474
|
+
return charRightExtent(c) + 1;
|
|
475
|
+
} else {
|
|
476
|
+
return 1;
|
|
477
|
+
}
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
var GLOBAL_POOL_STORAGE: ?GraphemePool = null;
|
|
481
|
+
|
|
482
|
+
pub fn initGlobalPool(allocator: std.mem.Allocator) *GraphemePool {
|
|
483
|
+
return initGlobalPoolWithOptions(allocator, .{});
|
|
484
|
+
}
|
|
485
|
+
|
|
486
|
+
pub fn initGlobalPoolWithOptions(allocator: std.mem.Allocator, options: GraphemePool.InitOptions) *GraphemePool {
|
|
487
|
+
if (GLOBAL_POOL_STORAGE == null) {
|
|
488
|
+
GLOBAL_POOL_STORAGE = GraphemePool.initWithOptions(allocator, options);
|
|
489
|
+
}
|
|
490
|
+
return &GLOBAL_POOL_STORAGE.?;
|
|
491
|
+
}
|
|
492
|
+
|
|
493
|
+
pub fn deinitGlobalPool() void {
|
|
494
|
+
if (GLOBAL_POOL_STORAGE) |*p| {
|
|
495
|
+
p.deinit();
|
|
496
|
+
GLOBAL_POOL_STORAGE = null;
|
|
497
|
+
}
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
pub const GraphemeTracker = struct {
|
|
501
|
+
pool: *GraphemePool,
|
|
502
|
+
used_ids: std.AutoHashMap(u32, u32), // id -> number of cells in this buffer
|
|
503
|
+
|
|
504
|
+
pub fn init(allocator: std.mem.Allocator, pool: *GraphemePool) GraphemeTracker {
|
|
505
|
+
return .{
|
|
506
|
+
.pool = pool,
|
|
507
|
+
.used_ids = std.AutoHashMap(u32, u32).init(allocator),
|
|
508
|
+
};
|
|
509
|
+
}
|
|
510
|
+
|
|
511
|
+
fn decRefAll(self: *GraphemeTracker) void {
|
|
512
|
+
var it = self.used_ids.keyIterator();
|
|
513
|
+
while (it.next()) |idp| {
|
|
514
|
+
// Pool refs are tracked per ID (first/last cell transition), so clear
|
|
515
|
+
// decrefs once per tracked ID, not once per per-buffer cell count.
|
|
516
|
+
self.pool.decref(idp.*) catch {};
|
|
517
|
+
}
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
pub fn deinit(self: *GraphemeTracker) void {
|
|
521
|
+
self.decRefAll();
|
|
522
|
+
self.used_ids.deinit();
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
pub fn clear(self: *GraphemeTracker) void {
|
|
526
|
+
self.decRefAll();
|
|
527
|
+
self.used_ids.clearRetainingCapacity();
|
|
528
|
+
}
|
|
529
|
+
|
|
530
|
+
pub fn add(self: *GraphemeTracker, id: u32) void {
|
|
531
|
+
const res = self.used_ids.getOrPut(id) catch |err| {
|
|
532
|
+
std.debug.panic("GraphemeTracker.add failed: {}\n", .{err});
|
|
533
|
+
};
|
|
534
|
+
if (!res.found_existing) {
|
|
535
|
+
res.value_ptr.* = 1;
|
|
536
|
+
self.pool.incref(id) catch |err| {
|
|
537
|
+
std.debug.panic("GraphemeTracker.add incref failed: {}\n", .{err});
|
|
538
|
+
};
|
|
539
|
+
} else {
|
|
540
|
+
res.value_ptr.* += 1;
|
|
541
|
+
}
|
|
542
|
+
}
|
|
543
|
+
|
|
544
|
+
pub fn remove(self: *GraphemeTracker, id: u32) void {
|
|
545
|
+
const count_ptr = self.used_ids.getPtr(id) orelse return;
|
|
546
|
+
if (count_ptr.* > 1) {
|
|
547
|
+
count_ptr.* -= 1;
|
|
548
|
+
return;
|
|
549
|
+
}
|
|
550
|
+
|
|
551
|
+
if (self.used_ids.remove(id)) {
|
|
552
|
+
self.pool.decref(id) catch {};
|
|
553
|
+
}
|
|
554
|
+
}
|
|
555
|
+
|
|
556
|
+
pub fn replace(self: *GraphemeTracker, old_id: ?u32, new_id: ?u32) void {
|
|
557
|
+
if (old_id != null and new_id != null and old_id.? == new_id.?) return;
|
|
558
|
+
|
|
559
|
+
if (new_id) |id| self.add(id);
|
|
560
|
+
if (old_id) |id| self.remove(id);
|
|
561
|
+
}
|
|
562
|
+
|
|
563
|
+
pub fn contains(self: *const GraphemeTracker, id: u32) bool {
|
|
564
|
+
return self.used_ids.contains(id);
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
pub fn hasAny(self: *const GraphemeTracker) bool {
|
|
568
|
+
return self.used_ids.count() > 0;
|
|
569
|
+
}
|
|
570
|
+
|
|
571
|
+
pub fn getGraphemeCount(self: *const GraphemeTracker) u32 {
|
|
572
|
+
return @intCast(self.used_ids.count());
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
pub fn getGraphemeCellCount(self: *const GraphemeTracker) u32 {
|
|
576
|
+
var total: u32 = 0;
|
|
577
|
+
var it = self.used_ids.valueIterator();
|
|
578
|
+
while (it.next()) |count_ptr| {
|
|
579
|
+
total += count_ptr.*;
|
|
580
|
+
}
|
|
581
|
+
return total;
|
|
582
|
+
}
|
|
583
|
+
|
|
584
|
+
pub fn getTotalGraphemeBytes(self: *const GraphemeTracker) u32 {
|
|
585
|
+
var total_bytes: u32 = 0;
|
|
586
|
+
var it = self.used_ids.iterator();
|
|
587
|
+
while (it.next()) |entry| {
|
|
588
|
+
const id = entry.key_ptr.*;
|
|
589
|
+
const count = entry.value_ptr.*;
|
|
590
|
+
if (self.pool.get(id)) |bytes| {
|
|
591
|
+
total_bytes += @as(u32, @intCast(bytes.len)) * count;
|
|
592
|
+
} else |_| {
|
|
593
|
+
// If we can't get the bytes, this shouldn't happen but handle gracefully
|
|
594
|
+
continue;
|
|
595
|
+
}
|
|
596
|
+
}
|
|
597
|
+
return total_bytes;
|
|
598
|
+
}
|
|
599
|
+
};
|