ビデオメモリにスワップ

提供: ArchWiki
2022年7月30日 (土) 23:57時点におけるAshMyzk (トーク | 投稿記録)による版 (同期)
ナビゲーションに移動 検索に移動

関連記事

RAM の容量が非常に小さいが、ビデオ RAM に余りがあるような稀なケースでは、ビデオ RAM をスワップとして使うことができます。

MTD カーネルサブシステム

メリット

グラフィックカードに搭載された GDDRX SDRAM や DDR SDRAM はカーネルの MTD サブシステムを使うことでスワップとして利用することができます。専用のグラフィックメモリが 256 MB 以上存在してシステムメモリ (DDRX SDRAM) の量が限られている場合、グラフィックメモリにスワップを設定する価値があります。

警告:
  • バイナリドライバでは上手く行きません。
  • グラフィックドライバーが使用するメモリを設定しないと、スワップとして使われてるメモリにテクスチャを保存しようとして Xorg がクラッシュする可能性があります。ビデオメモリを上書きすることで出来るビデオドライバーを使用することで安定性が増します。

事前設定

MTD モジュールを有効にしたカーネルを実行している場合、ビデオカードのメモリに対応する PCI アドレスの領域を指定してモジュールをロードしてください。

利用可能なメモリ領域を確認するには、以下のコマンドを実行して VGA compatible controller のセクションをチェックしてください:

$ lspci -vvv
01:00.0 VGA compatible controller: NVIDIA Corporation GK104 [GeForce GTX 670] (rev a1) (prog-if 00 [VGA controller])
	Subsystem: ASUSTeK Computer Inc. Device 8405
	Control: I/O+ Mem+ BusMaster+ SpecCycle- MemWINV- VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx-
	Status: Cap+ 66MHz- UDF- FastB2B- ParErr- DEVSEL=fast >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
	Latency: 0
	Interrupt: pin A routed to IRQ 57
	Region 0: Memory at f5000000 (32-bit, non-prefetchable) [size=16M]
	Region 1: Memory at e8000000 (64-bit, prefetchable) [size=128M]
	Region 3: Memory at f0000000 (64-bit, prefetchable) [size=32M]
	Region 5: I/O ports at e000 [size=128]
	[virtual] Expansion ROM at f6000000 [disabled] [size=512K]
	Capabilities: <access denied>
	Kernel driver in use: nvidia
	Kernel modules: nouveau, nvidia
ノート: 複数の GPU が搭載されている場合、複数のエントリが表示されます。

最も利用価値が高いのは prefetchable かつ64ビットかつ容量が一番大きな region です。

ノート: 使用するグラフィックカードに GDDR5 SDRAM が 2 GB 以上搭載されていても、上記のコマンドで全ての容量が出力されることはありません。

ビデオカードを機能させるにはメモリが多少必要なので、計算が必要です。オフセットは2の累乗で簡単に計算できます。グラフィックカードはアドレス領域の最初の部分をテクスチャなどに使います。しかし、この記事の冒頭で書いたように、2つのプログラムが同じセクタに書き込みを行うと、安定性に問題が生じます。

例: グラフィックメモリが合計で 256 MB ある場合、式は 2^28 になります (2の28乗)。グラフィックメモリに約 64 MB 残すとするとグラフィックメモリの中でスワップに使用する領域は 2^26 で計算されます。

上の値を使って、差を取ってスワップメモリに使用する容量を決定してください。通常の動作には 2^24 (32M) 残せば十分です。

セットアップ

slram モジュールを設定:

# /etc/modprobe.d/modprobe.conf
options phram phram=VRAM,0xStartRange,0xUsedAmount

起動時にモジュールをロード:

# /etc/modules-load.d/vramswap.conf
phram
mtdblock

systemd サービスを作成:

# /usr/lib/systemd/system/vramswap.service
[Unit]
Description=Swap on Video RAM

[Service]
Type=oneshot
ExecStart=/usr/bin/mkswap /dev/mtdblock0
ExecStart=/usr/bin/swapon /dev/mtdblock0 -p 10
ExecStop=/usr/bin/swapoff /dev/mtdblock0
RemainAfterExit=yes

[Install]
WantedBy=multi-user.target

Xorg ドライバーの設定

X を安定化させるために、検出されているビデオメモリよりも少ないメモリしか使用しないようにビデオドライバーを設定する必要があります:

# /etc/X11/xorg.conf.d/vramswap.conf
Section "Device"
    Driver "radeon" # 使用するドライバに変更してください
    VideoRam 32768
	#その他の部分
EndSection

上記の例ではグラフィックメモリの 32 MB だけを使用するように指定しています。

ノート: ドライバーによってはビデオメモリの値を MiB で指定します。man ページを参照してください。

トラブルシューティング

以下のコマンドを使うことでディスクパーティションやフラッシュディスク、あるいはこのページのビデオメモリのように、様々な領域で使われているスワップを確認できます:

$ swapon -s

参照

FUSE ファイルシステム

この方法は OpenCL をサポートするハードウェア上で FUSE ファイルシステムにスワップファイルを保存することにより機能します。さらなる情報については GPGPU をご覧ください。

セットアップ

まず vramfs-gitAUR をインストールしてください。次に、マウントポイント用の空ディレクトリを作成してください(例: /tmp/vram)。

以下のコマンドを実行し、vramfs とスワップファイルをセットアップします。

# vramfs /tmp/vram 256MB -f # 256M の部分は対象の vramfs サイズに変更してください
# dd if=/dev/zero of=/tmp/vram/swapfile bs=1M count=200 # 200 の部分は対象のスワップ領域のサイズを MiB で指定してください
# chmod 0600 /tmp/vram/swapfile
# mkswap -U clear /tmp/vram/swapfile

これでスワップが準備完了になったはずです。swapon を実行し確認してください。

さらなる情報についてはスワップ#スワップファイルをご覧ください。

ノート: この方法で作成したスワップは永続的なものではなく、システムを再起動すると消失します。
ヒント: /tmp/vramtmpfs のように一時的な保存場所としても利用可能です。

swappiness を設定

ノート: 以下の推奨事項は、あなたの環境では適用できないかもしれません。常に注意をはらい、あなたの特定の構成でどのように適用されるかを確認するべきです。

VRAM 上にスワップを配置する場合、swappiness を増やすのは良いアイデアかもしれません。特に、VRAM スワップファイルに対するランダム I/O がランダムディスク I/O よりも大幅に速い場合に効果を発揮します。ディスク読み込みをキャッシュするメリットが、スワップのコストを上回るためです。例えば、ランダムディスク I/O の速度が VRAM のスワップ I/O と等しい場合、swappiness は 100 に設定するべきです。VRAM のスワップ I/O がディスク I/O より2倍速い場合、swappiness は 133 に設定するべきです。swappiness の値を適切に計算する方法については カーネルドキュメント を見てください。

トラブルシューティング

swapon: /tmp/vram/swapfile: skipping - it appears to have holes.

作成したスワップファイルの領域が連続していないことを意味しています。ループデバイスをセットアップすることでこの問題を回避できます。

# cd /tmp/vram
# LOOPDEV=$(losetup -f)
# truncate -s 4G swapfile # 4G の部分は対象のスワップ領域のサイズに置き換えてください。この値は確保した vramfs のサイズよりも小さくなければなりません。
# losetup $LOOPDEV swapfile
# mkswap $LOOPDEV
# swapon $LOOPDEV

高メモリ負荷時にシステムが完全にフリーズする

高メモリ負荷時に vramfs プロセス自体が VRAM スワップ領域にスワップされてしまうことがあります。これにより、完全なデッドロックが発生します。解決策は、systemd ファイルで vramfs を起動することで、cgroups 経由でそのプロセスをスワップ不可にすることです:

/etc/systemd/system/vramswap.service
[Unit]
Description=Set up swap in VRAM
After=default.target

[Service]
Type=oneshot
RemainAfterExit=yes
# /root/vramswap.sh は、必要なセットアップを行うスクリプトへのパスに置き換えてください
ExecStart=/root/vramswap.sh
TimeoutStartSec=0
# スワップを無効化
MemorySwapMax=0

[Install]
WantedBy=default.target

参照

翻訳ステータス: このページは en:Swap on video RAM の翻訳バージョンです。最後の翻訳日は 2022-07-30 です。もし英語版に 変更 があれば、翻訳の同期を手伝うことができます。