本页内容由 AI 自动翻译,所有内容以英文版本为准。查看英文版 →
Sqrt-price 表示
CLMM 将价格存储为sqrt_price_x64——token1 相对于 token0 的价格的平方根,采用 Q64.64 定点数格式:
其中 p = token1_amount / token0_amount。在 sqrt 空间而非 p 空间中工作使得交换数学线性化(代币数量增量相对于 Δsqrt_price 是线性的),而 x64 定点格式在多 tick 交换过程中保持精度。
Tick ↔ sqrt-price 转换通过 bit-by-bit 对数近似预计算:
在 tick_math::get_sqrt_price_at_tick 中实现为基于查表的指数运算。
流动性作为规范单位
在范围[sqrt_a, sqrt_b](其中 sqrt_a < sqrt_b)内,流动性 L 的头寸对应的代币数量如下。设 sqrt_c = sqrt_price_x64 为池当前价格。
| 情况 | amount0 | amount1 |
|---|---|---|
sqrt_c <= sqrt_a(池价格低于范围) | L · (sqrt_b - sqrt_a) / (sqrt_a · sqrt_b) | 0 |
sqrt_a < sqrt_c < sqrt_b(在范围内) | L · (sqrt_b - sqrt_c) / (sqrt_c · sqrt_b) | L · (sqrt_c - sqrt_a) |
sqrt_c >= sqrt_b(池价格高于范围) | 0 | L · (sqrt_b - sqrt_a) |
x = L / sqrt_p、y = L · sqrt_p。
集成者通常需要反向计算:给定 amount0 / amount1 的存款,计算能放入该范围的最大 L。SDK 的 LiquidityMath.getLiquidityFromTokenAmounts 就是做这个的。在范围内情况下的公式为:
哪一边受限决定了实际消耗的比例;另一边可能有剩余。
单个 tick 交换步骤
交换以步骤进行。每个步骤要么 (a) 在当前 tick 范围内消耗所有可用输入而不跨越 tick,要么 (b) 将价格精确移动到下一个已初始化的 tick。 给定当前状态(sqrt_c, L) 和向上的交换(token0 输入、token1 输出、sqrt_price 上升),到下一个已初始化 tick 的距离为 sqrt_t。在这个微小间隔内,输入与价格之间的关系为:
和
程序执行以下两个操作之一:
-
整个输入能否放下? 如果剩余输入(扣除费用后)小于到达
sqrt_t所需的Δamount0,则精确求解新的sqrt_c': (对于精确输入的token0 → token1交换)。交换在此步骤完成,不跨越 tick。 -
输入超过
Δamount0? 设置sqrt_c' = sqrt_t,跨越 tick(应用liquidity_net),将剩余输入减少Δamount0,将输出增加Δamount1,然后重复。
token1 → token0、价格下降),公式中 sqrt_c 和 sqrt_t 交换,另一处进行反演。
完整的 Rust 实现位于 raydium-clmm/programs/amm/src/libraries/swap_math.rs。该处的逻辑与 Uniswap v3 的 SwapMath.computeSwapStep 一一对应。
每个步骤的费用
交易费从输入金额扣除,与 CPMM 的约定相同:L_i 的头寸以后会读回 L_i · Δfee_growth_global / 2^{64} 欠款代币。
协议和基金部分分别累加到 PoolState.protocol_fees_token_{0,1} 和 PoolState.fund_fees_token_{0,1},与 CPMM 相同。它们由 CollectProtocolFee / CollectFundFee 扫除。
范围外和范围内的费用增长
CLMM 费用会计的难点在于:头寸只有在池价格在其范围内时才能获得费用。池跟踪全局累积费用;头寸需要知道在其特定范围内时的累积费用。 解决方案是基于 tick 的累加器。每个 tick 存储:- 如果池的价格高于该 tick(
tick_current >= this_tick),则fee_growth_outside = fee_growth_global。(到目前为止赚取的所有内容相对于当前价格都是”范围外”——即在该 tick 下方。) - 否则
fee_growth_outside = 0。
fee_growth_outside:
这个不变量保证了:对于任何 tick t,fee_growth_outside(t) 等于当 tick_current 在 t 的对侧时累积的费用。
范围 [tick_lower, tick_upper] 内的费用增长然后派生为:
头寸存储的内容及其读取方式
PersonalPositionState 存储 fee_growth_inside_0_last_x64 和 fee_growth_inside_1_last_x64:上次触及头寸时的 fee_growth_inside 值。
在任何后续触及(增加、减少、收集)时,程序:
- 使用上述公式计算当前的
fee_growth_inside_{0,1}_x64。 - 计算
Δ = fee_growth_inside_now − fee_growth_inside_last(u128 上的模运算)。 - 向
tokens_fees_owed_{0,1}加上Δ × position.liquidity / 2^{64}。 - 将
fee_growth_inside_last更新为新值。
CollectFees / DecreaseLiquidity 时从金库中移出,针对 tokens_fees_owed。
奖励
池最多 3 个奖励流中的每一个都使用相同的增长内机制,在其自己的reward_growth_global_x64 累加器中。在发放时:
——排放量与活跃流动性成反比,因此密集池中的每个头寸每秒支付的比例更低,但总位置数更多。每个头寸欠的奖励为
并通过 CollectReward 领取。见 products/clmm/fees。
工作示例:精确输入交换
假设:tick_spacing = 60sqrt_price_x64 = 1 × 2^{64}——价格 = 1.0,所以tick_current = 0。- 活跃流动性
L = 1_000_000 × 2^{64}。 - 上方下一个已初始化 tick:
t = 60(sqrt_price_b ≈1.003004 × 2^{64})。 - 交易费率:500(0.05%)。
SwapBaseInput 精确输入 1,000 token0。
步骤 1——费用:
999 < 2995.5,所以整个输入适合而无需跨越 tick。
步骤 3——新价格:
sqrt_c' 略低于 sqrt_c。请注意,上述公式是针对 token1 → token0 交换的。这里的例子是 token0 → token1,它驱动价格上升而不是下降——所以我们使用 token0 输入对应的形式:
token0 → token1 交换的预期交换方向:sqrt_c 随价格上升。)
步骤 4——输出数量:
trade_fee_rate × protocol_fee_rate / 1e6(基金也类似)在 LP、协议和基金之间分割;LP 部分流入 fee_growth_global_0_x64。
交换过程中的限价单匹配
当交换步骤跨越持有未平仓限价单的 tick 时,这些订单在 LP 曲线之前以该 tick 的精确价格消耗交换输入。在该 tick 内按order_phase 队列 FIFO 匹配。
TickState 上的每队列状态
orders_amount 并继承下一个 order_phase;它们直到前一队列完全消耗才能填充。
匹配步骤
在交换期间每个 tick 交叉处发生的匹配的伪代码:SettleLimitOrder(或 DecreaseLimitOrder)。池仅通过 unfilled_ratio_x64 跟踪队列现在有多少被填充。每个 LimitOrderState 在开设时存储其自己的 (order_phase, unfilled_ratio_x64) 快照,因此结算可简化为:
与 LP 曲线的交互
在交换步骤中,限价单匹配发生在 tick(零Δsqrt_price);LP 曲线消耗发生在 tick 之间。因此顺序为:
- 跨越 tick
t_cross(首先应用 LPliquidity_net变化,因为这是 Uniswap-V3 的做法)。 - 填充坐在
t_cross的任何限价单。 - 沿 LP 曲线继续到下一个已初始化 tick 或
swap_input耗尽。
动态费用派生
PoolState.dynamic_fee_info 携带波动率状态。每个交换步骤计算每步费率为:
其中:
- —
DYNAMIC_FEE_CONTROL_DENOMINATOR - —
VOLATILITY_ACCUMULATOR_SCALE vol_acc是更新规则后的每次交换累加器tick_spacing来自PoolState.tick_spacing
累加器更新
两个规则在每次交换中按顺序应用: 衰减。 参考底线根据自上次更新以来的时间衰减: 累积。 新累加器是参考加上自前一参考索引以来的 tick 距离:tick_spacing_index_reference()以 tick 间距单位表示,而非原始 tick:。
为什么在 tick 距离上是抛物线
对累加器平方意味着费用随价格离其参考点走过距离的平方而上升。经验上这与随机游走压力下价格方差缩放相匹配:2× tick 偏移意味着 4× 隐含波动率,因此收费 4× 增加。dynamic_fee_control 参数校准绝对水平。
filter_period 窗口防止微小的次秒级振荡(例如 MEV 机器人夹击)充气累加器。decay_period 窗口防止单个过去的峰值在市场已平静后无限期收费。
数值稳健性
- 所有中间乘积通过
u128或u256形状算术进行。CLMM 直接使用从 Uniswap v3 移植的U128Sqrt辅助工具和FullMath::mulDiv模式。 - 除法舍入按步选择以在本地强制不变量
k' ≥ k。SwapBaseInput将输出向下舍入;SwapBaseOutput将输入向上舍入。 - Tick 交叉使
PoolState.liquidity变为零被允许(价格可以遍历”流动性洞”)但交换仅推进到下一个已初始化 tick 而不消耗输入,不收费。 - 溢出防护:
sqrt_price_x64保持在包含范围[MIN_SQRT_PRICE_X64, MAX_SQRT_PRICE_X64]内,对应于[MIN_TICK, MAX_TICK]。会推过任一边界的交换以SqrtPriceLimitOverflow还原。
接下来去哪里
products/clmm/ticks-and-positions了解 tick 地图如何参与 walk。products/clmm/fees了解数学的费用/奖励方面的细节。algorithms/clmm-math了解L = sqrt(x · y)及范围 vs 流动性公式背后的推导。
raydium-io/raydium-clmm—libraries/swap_math.rs,libraries/tick_math.rs- “Uniswap v3 Core” 白皮书,§6–7

