<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:content="http://purl.org/rss/1.0/modules/content/"><channel><title>Posts on Xerxes II's Blog</title><link>/posts/</link><description>Recent content in Posts on Xerxes II's Blog</description><generator>Hugo</generator><language>zh-cn</language><lastBuildDate>Sun, 03 May 2026 00:00:00 +1000</lastBuildDate><atom:link href="/posts/index.xml" rel="self" type="application/rss+xml"/><item><title>用 Scheme 给 Helix 写 Wakatime 插件</title><link>/posts/helix-wakatime-plugin-in-scheme/</link><pubDate>Sun, 03 May 2026 00:00:00 +1000</pubDate><guid>/posts/helix-wakatime-plugin-in-scheme/</guid><description>&lt;p&gt;我日常用的编辑器是 &lt;a href="https://helix-editor.com/"&gt;Helix&lt;/a&gt;。我从来没深入用过 Vim，所以转到 Helix 并没有什么适应成本——反而更自然：Helix 开箱即用的体验很好，而且&amp;quot;先选择再操作&amp;quot;（select-then-act）的编辑模型比 Vim 的&amp;quot;先动词再名词&amp;quot;对我来说更直觉。&lt;/p&gt;
&lt;p&gt;Helix 主线目前没有插件系统，这一直是社区里最大的痛点之一。不过 &lt;a href="https://github.com/mattwparas"&gt;mattwparas&lt;/a&gt; 维护了一个 fork（&lt;a href="https://github.com/mattwparas/helix/tree/steel-event-system"&gt;&lt;code&gt;steel-event-system&lt;/code&gt; 分支&lt;/a&gt;），集成了 &lt;a href="https://github.com/mattwparas/steel"&gt;Steel&lt;/a&gt;——一个嵌入式 Scheme 实现——作为插件运行时。Scheme 是 Lisp 的一个方言，而 Steel 是 Scheme 的一个方言，所以你大概能猜到写插件的体验是什么画风：很多括号。&lt;/p&gt;
&lt;p&gt;我在 AUR 上打包了这个 fork：&lt;a href="https://aur.archlinux.org/packages/helix-steel-git"&gt;helix-steel-git&lt;/a&gt;，Arch 用户可以直接装。&lt;/p&gt;
&lt;p&gt;其实我很早就知道这个 fork 的存在，但一直没有真正去用——原因很简单，我是真的怕括号。然而最近我彻底抛弃了 VSCode，全面转向 Helix 作为唯一的编辑器，随之而来的问题是：对插件的需求越来越强烈。而 Steel 生态目前能用的插件屈指可数，想要什么功能基本只能自己动手。于是我终于硬着头皮开始写 Scheme 了。&lt;/p&gt;
&lt;h2 id="第一版让-llm-来"&gt;第一版：让 LLM 来&lt;/h2&gt;
&lt;p&gt;第一步是从社区已有的插件里复制了一套模板，建好 &lt;a href="https://github.com/Xerxes-2/wakatime.hx"&gt;wakatime.hx&lt;/a&gt; 仓库，然后直接把需求丢给 Claude，让它生成第一版代码。毕竟我对 Lisp 一无所知，连基本语法都得现查。&lt;/p&gt;
&lt;p&gt;Claude 确实给了我一个能跑的版本。问题是——LLM 生成的 Lisp 代码，人类读起来比手写的还要痛苦。比如获取编辑器版本号这段：&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-scheme" data-lang="scheme"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(&lt;span style="color:#66d9ef"&gt;define &lt;/span&gt;(&lt;span style="color:#a6e22e"&gt;editor-version&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;if &lt;/span&gt;*wakatime-editor-version*
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; *wakatime-editor-version*
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;begin&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;let &lt;/span&gt;([spawned (&lt;span style="color:#a6e22e"&gt;spawn-process&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;with-stdout-piped&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;command&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;hx&amp;#34;&lt;/span&gt; (list &lt;span style="color:#e6db74"&gt;&amp;#34;--version&amp;#34;&lt;/span&gt;))))])
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;if &lt;/span&gt;(&lt;span style="color:#a6e22e"&gt;Ok?&lt;/span&gt; spawned)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;let &lt;/span&gt;([output (&lt;span style="color:#a6e22e"&gt;wait-&amp;gt;stdout&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;Ok-&amp;gt;value&lt;/span&gt; spawned))])
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;if &lt;/span&gt;(&lt;span style="color:#a6e22e"&gt;Ok?&lt;/span&gt; output)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;let &lt;/span&gt;([version (&lt;span style="color:#a6e22e"&gt;extract-version&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;Ok-&amp;gt;value&lt;/span&gt; output))])
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;set! &lt;/span&gt;*wakatime-editor-version* version)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; version)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;unknown&amp;#34;&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;unknown&amp;#34;&lt;/span&gt;)))))
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;嵌套的 &lt;code&gt;if&lt;/code&gt;-&lt;code&gt;let&lt;/code&gt;-&lt;code&gt;if&lt;/code&gt;-&lt;code&gt;let&lt;/code&gt;-&lt;code&gt;if&lt;/code&gt;，层层剥洋葱式地处理 &lt;code&gt;Ok?&lt;/code&gt; 结果——典型的&amp;quot;LLM 不知道这门语言的惯用写法，只好用最保守的方式堆逻辑&amp;quot;。能跑，但读起来让人头大。&lt;/p&gt;</description></item><item><title>Oracle Cloud Ubuntu VM 从 ext4 原地转换到 Btrfs</title><link>/posts/ext4-to-btrfs-oracle-cloud/</link><pubDate>Mon, 06 Apr 2026 00:00:00 +0000</pubDate><guid>/posts/ext4-to-btrfs-oracle-cloud/</guid><description>&lt;p&gt;记录一下把 Oracle Cloud 上的 Ubuntu 26.04 ARM VM 从 ext4 原地转换为 Btrfs 的过程。&lt;/p&gt;
&lt;blockquote&gt;
&lt;p&gt;&lt;strong&gt;&lt;span style="color:red"&gt;WARNING&lt;/span&gt;: 本文操作涉及文件系统级别的原地转换，一旦出错数据将无法恢复。我的 VM 是一台空白机器，没有任何重要数据，所以才敢这么操作。如果你的机器上有任何你不想丢的东西，请务必先做好完整备份（Oracle Cloud 的 Boot Volume Backup、rsync 到远端、dd 镜像等），确认备份可恢复之后再动手。文件系统转换失败 + 没有备份 = 数据全灭，没有后悔药。&lt;/strong&gt;&lt;/p&gt;
&lt;/blockquote&gt;
&lt;h2 id="背景"&gt;背景&lt;/h2&gt;
&lt;p&gt;Oracle Cloud 免费 ARM 实例，150G 磁盘，系统盘只有一个 ext4 根分区。想用上 Btrfs 的透明压缩、快照和子卷管理，但不想重装系统。&lt;/p&gt;
&lt;h2 id="方案initramfs-钩子--btrfs-convert"&gt;方案：initramfs 钩子 + btrfs-convert&lt;/h2&gt;
&lt;p&gt;核心思路：根分区在挂载状态下无法转换，所以利用 initramfs 的 &lt;code&gt;local-premount&lt;/code&gt; 阶段——此时根分区尚未挂载——执行 &lt;code&gt;btrfs-convert&lt;/code&gt;。&lt;/p&gt;
&lt;h3 id="1-安装-btrfs-progs"&gt;1. 安装 btrfs-progs&lt;/h3&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-bash" data-lang="bash"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;sudo apt install btrfs-progs
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;h3 id="2-创建-initramfs-钩子"&gt;2. 创建 initramfs 钩子&lt;/h3&gt;
&lt;p&gt;将 &lt;code&gt;btrfs-convert&lt;/code&gt; 和 &lt;code&gt;btrfs&lt;/code&gt; 二进制打包进 initramfs：&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-bash" data-lang="bash"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#75715e"&gt;# /etc/initramfs-tools/hooks/btrfs-convert&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#75715e"&gt;#!/bin/sh&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;set -e
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;PREREQ&lt;span style="color:#f92672"&gt;=&lt;/span&gt;&lt;span style="color:#e6db74"&gt;&amp;#34;&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;prereqs&lt;span style="color:#f92672"&gt;()&lt;/span&gt; &lt;span style="color:#f92672"&gt;{&lt;/span&gt; echo &lt;span style="color:#e6db74"&gt;&amp;#34;&lt;/span&gt;$PREREQ&lt;span style="color:#e6db74"&gt;&amp;#34;&lt;/span&gt;; &lt;span style="color:#f92672"&gt;}&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#66d9ef"&gt;case&lt;/span&gt; $1 in prereqs&lt;span style="color:#f92672"&gt;)&lt;/span&gt; prereqs; exit 0;; &lt;span style="color:#66d9ef"&gt;esac&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;. /usr/share/initramfs-tools/hook-functions
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;copy_exec /usr/bin/btrfs-convert /usr/bin
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;copy_exec /usr/bin/btrfs /usr/bin
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;在 root 挂载前执行转换。注意 &lt;code&gt;btrfs-convert&lt;/code&gt; 会生成新的 UUID，所以脚本里需要同时更新 GRUB 配置，否则重启后内核找不到 root 会掉进 initramfs shell：&lt;/p&gt;</description></item><item><title>我是怎么改善 Rust 编译体验的</title><link>/posts/optimize-rust-compile-time-space/</link><pubDate>Mon, 06 Apr 2026 00:00:00 +0000</pubDate><guid>/posts/optimize-rust-compile-time-space/</guid><description>&lt;p&gt;Rust 编译慢、&lt;code&gt;target/&lt;/code&gt; 目录大，这是老生常谈了。这篇记录一下我目前在用的几个改善手段，以我的项目 &lt;a href="https://github.com/Xerxes-2/clewdr"&gt;ClewdR&lt;/a&gt;（394 个 crate 依赖的异步 Web 服务）为例。&lt;/p&gt;
&lt;p&gt;环境：Rust 1.94.1，CachyOS (Arch-based)，NVMe SSD，Btrfs。&lt;/p&gt;
&lt;h2 id="rust-lld"&gt;rust-lld&lt;/h2&gt;
&lt;p&gt;链接是 Rust 编译的最后一步，也是传统上最慢的一步。GNU ld 在这里表现很差，特别是开了 LTO 的时候。&lt;/p&gt;
&lt;p&gt;以前的做法是手动装 &lt;code&gt;lld&lt;/code&gt; 或 &lt;code&gt;mold&lt;/code&gt;，然后在 &lt;code&gt;.cargo/config.toml&lt;/code&gt; 里配：&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-toml" data-lang="toml"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;[&lt;span style="color:#a6e22e"&gt;target&lt;/span&gt;.&lt;span style="color:#a6e22e"&gt;x86_64-unknown-linux-gnu&lt;/span&gt;]
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#a6e22e"&gt;linker&lt;/span&gt; = &lt;span style="color:#e6db74"&gt;&amp;#34;clang&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#a6e22e"&gt;rustflags&lt;/span&gt; = [&lt;span style="color:#e6db74"&gt;&amp;#34;-C&amp;#34;&lt;/span&gt;, &lt;span style="color:#e6db74"&gt;&amp;#34;link-arg=-fuse-ld=lld&amp;#34;&lt;/span&gt;]
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;但从 Rust 1.85（2025-02-20）开始，&lt;code&gt;rust-lld&lt;/code&gt; 在 &lt;code&gt;x86_64-unknown-linux-gnu&lt;/code&gt; 上已经是默认链接器了，不需要任何配置：&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-bash" data-lang="bash"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;$ readelf -p .comment target/release/clewdr
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;String dump of section &lt;span style="color:#e6db74"&gt;&amp;#39;.comment&amp;#39;&lt;/span&gt;:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f92672"&gt;[&lt;/span&gt; 1&lt;span style="color:#f92672"&gt;]&lt;/span&gt; Linker: LLD 21.1.8
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#f92672"&gt;[&lt;/span&gt; 5f&lt;span style="color:#f92672"&gt;]&lt;/span&gt; rustc version 1.94.1 &lt;span style="color:#f92672"&gt;(&lt;/span&gt;e408947bf 2026-03-25&lt;span style="color:#f92672"&gt;)&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;升级 Rust 就行，免费午餐。&lt;/p&gt;
&lt;p&gt;顺带一提，社区里还有两个值得关注的替代链接器：&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;a href="https://github.com/rui314/mold"&gt;mold&lt;/a&gt;：以速度为卖点的链接器，在非 LTO 场景下通常比 lld 更快，不过对 LTO 的支持有限。&lt;/li&gt;
&lt;li&gt;&lt;a href="https://github.com/wild-linker/wild"&gt;wild&lt;/a&gt;：一个用 Rust 写的实验性链接器，目标是成为最快的 Linux ELF 链接器，做了大量多线程优化。目前还在活跃开发中，有兴趣可以关注。&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;对于大多数人来说，默认的 rust-lld 已经够用了。&lt;/p&gt;</description></item><item><title>用 cargo-zigbuild 交叉编译 Rust 项目到 FreeBSD</title><link>/posts/cross-compile-rust-freebsd-zigbuild/</link><pubDate>Mon, 06 Apr 2026 00:00:00 +0000</pubDate><guid>/posts/cross-compile-rust-freebsd-zigbuild/</guid><description>&lt;p&gt;记录一下在 Linux (x86_64) 上用 &lt;code&gt;cargo-zigbuild&lt;/code&gt; 交叉编译 Rust 项目到 FreeBSD x86_64 的踩坑过程。项目是一个 Discord bot，依赖了 &lt;code&gt;sysinfo&lt;/code&gt;、&lt;code&gt;tikv-jemallocator&lt;/code&gt;、&lt;code&gt;sea-orm&lt;/code&gt;(SQLite) 等 crate。&lt;/p&gt;
&lt;h2 id="背景"&gt;背景&lt;/h2&gt;
&lt;p&gt;Zig 自带了多平台的 libc 和交叉编译工具链，&lt;code&gt;cargo-zigbuild&lt;/code&gt; 利用 Zig 作为 linker 来实现 Rust 的交叉编译，通常比自己配置交叉工具链方便很多。但 FreeBSD 作为目标平台时会遇到一些特殊问题。&lt;/p&gt;
&lt;h2 id="准备"&gt;准备&lt;/h2&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-bash" data-lang="bash"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#75715e"&gt;# 安装 FreeBSD 目标&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;rustup target add x86_64-unknown-freebsd
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#75715e"&gt;# 安装 cargo-zigbuild（如果还没有）&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;cargo install cargo-zigbuild
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;h2 id="问题一缺少-freebsd-系统库"&gt;问题一：缺少 FreeBSD 系统库&lt;/h2&gt;
&lt;p&gt;编译本身顺利通过，但链接阶段报错：&lt;/p&gt;
&lt;pre tabindex="0"&gt;&lt;code&gt;error: unable to find dynamic system library &amp;#39;geom&amp;#39; using strategy &amp;#39;no_fallback&amp;#39;
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;&lt;code&gt;sysinfo&lt;/code&gt; crate 在 FreeBSD 上需要链接 &lt;code&gt;libkvm&lt;/code&gt;、&lt;code&gt;libprocstat&lt;/code&gt;、&lt;code&gt;libgeom&lt;/code&gt;、&lt;code&gt;libdevstat&lt;/code&gt; 等 FreeBSD base system 的库。Zig 的工具链只提供 libc，不包含这些系统库。&lt;/p&gt;</description></item></channel></rss>