目次

ARMの非境界整列アドレスのデータアクセス

RISC系CPUでは、16bitや32bitのデータアクセスを行う時に、それぞれ2バイト境界や4バイト境界アドレスでないと例外トラップする仕様も多いが、ARMv6T2以降では非境界アドレスのデータアクセスもサポートされている。

ARMv7の場合

ARMv7 以降では、非境界整列アクセスが義務付けられてる(デフォルト)。

ARMv7-A と ARMv7-R では、システム制御レジスタ、SCTLR の A ビット(bit1)によって、境界調整チェックのイネーブル(A=1)またはディセーブル(A=0)を制御できる。又、ARMv7-M では、コンフィギュレーションおよび制御レジスタ(CCR)の UNALIGN_TRP ビット、ビット 3 がこれを制御する。

境界調整チェックがイネーブルされている場合、すべての非境界整列ワードまたはハーフワード転送が行われると、境界調整例外が発生する。

ディセーブルされている場合は、LDR、LDRH、STR、STRH、LDRSH、LDRT、STRT、LDRSHT、LDRHT、STRHT、および TBH 命令で非境界整列アクセスを行うことができる。他のデータアクセス命令が行われると常に非境界整列データの境界調整例外が発生する。 STRD および LDRD では、指定されたアドレスはワード境界で整列させる必要がある。

Abitは、デフォルトで「0」となっており、非境界アドレスのアクセスも可能となっているが、システム起動初期化時 A=1 として非境界アクセスをチェックし、安定して起動したら A=0 とするような使い方もできる。

    asm("	mrc	p15, 0, r0, c1, c0, 0");
    asm("	bic	r0, r0, #0x00000002"); /* clear A bit (disable alignment fault) */
    asm("	mcr	p15, 0, r0, c1, c0, 0");

GCCコンパイラの動作

GCCは、デフォルトで -munaligned-access が有効となっているらしく、データアクセスが境界調整されている場合、コマンドラインオプション -mno_unaligned_access を使って非境界整列アクセスをサポートするライブラリ関数へのリンクを避けることもできる。

補足

参考