バレットの還元アルゴリズム

提供: testwiki
2023年8月9日 (水) 00:47時点におけるimported>MSY-07による版 (参考文献: デフォルトソートの追加)
(差分) ← 古い版 | 最新版 (差分) | 新しい版 → (差分)
ナビゲーションに移動 検索に移動

合同算術において、バレットの還元アルゴリズムは P.D. バレットによって 1986 年に開発されたアルゴリズムである。[1]以下の式を計算することを考える。

c=amodn

愚直な方法は、高速な除算アルゴリズムを使用する方法である。バレットの還元アルゴリズムは、除算を乗算に置き換えることにより、n が定数で a<n2 であるときにこの計算を高速化する。

着想

n浮動小数点数としての逆数を s=1/n とおくと、x床関数として

amodn=aasn

となる。s が十分な精度で計算される限り、これは誤差なく成り立つ。

1ワード版のバレットの還元アルゴリズム

バレットが最初に考案したのは、上記のアルゴリズムの整数版であり、特に値がマシンのワードに収まる場合に適用できる。

上記の方法で amodn を整数の範囲内で計算する場合、自明に対応するのは n による除算を用いる方法である。

func reduce(a uint) uint {
    q := a / n  // 小数部分は暗黙に切り捨てられる
    return a - q * n
}

しかし、除算の計算には時間がかかり、CPUによっては定数時間で完了しない命令が使用されうる。そこで、バレットの方法では 2k による除算なら低コストな右シフトとして実現できることに注目し、1/nm/2k で近似する。

2k が与えられたとき、最良の m を見つけるために以下の式を考える。

m2k=1nm=2kn

m を整数にするために、2k/n をどうにか整数に丸める必要がある。最も近い整数に丸めるのが最適だが、m/2k1/n より大きくなるとアンダーフローを生じうるため、一般には m=2k/n を用いる。

すると、前述の関数は以下のように書き換えられる。

func reduce(a uint) uint {
    q := (a * m) >> k // ">> k" は k ビットの右シフト
    return a - q * n
}

ここで、m/2k1/n なので、この関数における q は小さめに評価される。これによりa[0,n) に必ずしも収まらず、[0,2n) の値を取りうる。そのため、値が n を超えた場合は n を引くことによってこれを補正する。

func reduce(a uint) uint {
    q := (a * m) >> k
    a -= q * n
    if n <= a {
        a -= n
    }
    return a
}

m/2k は近似にすぎないので、近似が正当となる a の範囲を考える必要がある。1/n の近似誤差は

e=1nm2k

なので、qの誤差はaeである。ae<1、すなわち a<1/e である限り、この還元は正当である。a1/e である場合も必ずしもこの関数が誤った値を返すわけではないが、a を先述の範囲内に収めることで一般の場合においての正当性を保証できる。

k を大きく取ると、還元が正当となる a の範囲は広がる一方、他の計算においてオーバーフローを生じる可能性がある。

16 ビット整数のシステム上で n=101 としたときを考える。

アルゴリズムが意味をなすためには、少なくとも k7 以上である必要がある。さもなくば、剰余を必要としない小さい値に対してしか還元が正しく動作しなくなってしまう。k=7 のときは、m=2k/n=128/101=1 であり、k=8 のときは m=256/101=2 である。後者においては 1/1012/256 で近似することになるので、前者における 1/128 による近似と全く等価になる。k=9 とすれば m=512/101=5 となり、よりよい近似を得る。

k=7k=9 において、アルゴリズムが正しい値を与える入力の範囲を考える。前者においては、e=1/nm/2k=1/1011/128=27/12928 なので、a<1/e すなわち a<478.81 である。a は整数なので、実質的な上限は 478 である。(実際は 504 まで正しく動く。)

k=9 においては、e=1/1015/512=7/51712 なので a の実質的な上限は 7387 である。(実際は 7473 まで正しく動く。)

近似がより良くなる次の k は 13 であり、81/8192 を与える。もっとも、計算途中に現れる ax810a でオーバーフローすることに注意すれば、この問題設定では k=7 とした方が良い。

特定の k に対する証明

2k0>n なる最小の整数 k0 をとり、kk0+1 とすると、これは上記の式で説明した「意味をなす」k である。前述のコードに対応して次の値を定義する。

  • q=ma2k
  • r=aqn.

床関数の定義から、q は整数で、ra(modn) である。また、a2k なら r<2n となる。このとき、前述のコードを数式で表すと

amodn={rif r<nrnotherwise

このとき、r<2n であることが以下のように示せる。

a2k なら

a2k(2kmodn)<n

である。nmamod2k2k<na にかかわらず成立するので、以下の式変形を得る。

a(2kmodn)2k+nmamod2k2k<2n
a(aa(2kmodn)2k)+n(mamod2k)2k<2n
aa2k(2k(2kmodn))+n(mamod2k)2k<2n
ana2k(2k(2kmodn)n)+n(mamod2k)2k<2n
ana2k2kn +n(mamod2k)2k<2n
anma2k+n(mamod2k)2k<2n
a(ma(mamod2k)2k)n<2n
ama2kn<2n
aqn<2n
r<2n

マルチワード版のバレットの還元アルゴリズム

バレットはこのアルゴリズムをRSA暗号の実装のために考案した。しかし、RSA暗号においては入力値は基本的にワードサイズを超える。そのためバレットは上記のアルゴリズムのマルチワード版も与えた。詳細については Handbook of Applied Cryptography を参照のこと。[2]

多項式に対するバレットのアルゴリズム

バレットのアルゴリズムは多項式除算に対しても適用できる。これは、多項式を反転して X 進数を用いることで計算される。[3]

関連項目

脚注

テンプレート:Reflist

参考文献

テンプレート:Refbegin

テンプレート:Refend