====== Writable snapshot libovolného blokového zařízení bez LVM ======
**"Dmsetup"** je user interface k **"dm"** (device mapper), což je velmi mocný low-level nástroj na translační mapování mezi blokovými zařízeními.
Zjednodušeně řečeno se tím říká, co má kernel dělat při r/w přístupu k určitému LBA daného zařízení, od lineárního mapování do jiného zařízení, přes striping/mirroring po snapshoty a další. Device mapper je interní součástí kernelu, tedy "dmsetup" je podobný jako třeba příkazy "iptables" nebo "ip", nedělají přímo nic, jen zkonfigurují vnitřní tabulky v kernelu, který pro následující požadavky již dělá potřebná mapování.
Využívá to např. LVM na vytváření/úpravy mapování LV do VG a VG do PV - např. triviálním několikařádkovým souborem s mapováním "linear" lze dosáhnout toho, že blokové zařízení bude na jiných zařízeních v nesouvislých úsecích (typický případ LVM po větším množství lvremove a lvcreate). Taková tabulka by mohla vypadat např. takhle:
0 8000 linear /dev/mapper/vg-main 60000
8000 20000 linear /dev/mapper/vg-main 100000
(od začátku řádku to je "začátek v novém zařízení, počet sektorů, způsob mapování, původní zařízení (do nějž jsou přístupy k danému úseku mapovány), začátek na původním zařízení, jednotkou měření pozic je sektor - 512 byte)
Existující tabulky od všeho v /dev/mapper lze vypsat příkazem "dmsetup table /dev/mapper/" - pro studijní účely...
:!: POZOR je to velmi nebezpečné, lze klidně udělat mapování do již zmapované oblasti a něco rozbít. Prostě něco jako dd, nic to nekontroluje a udělá to, co se tomu řekne...
===== Příkazy =====
Tohle vypíše použitá loop zařízení (loop vyrábí blokové zařízení z úseku jiného blokového zařízení nebo souboru, jde o to, aby se v dalším postupu nepoužila již použitá loop zařízení).
losetup -l
Tohle vyrobí loop zařízení z daného úseku disku (tohle je potřeba, když je poškozená partition table a tedy systém nevidí např. /dev/sdb2). I tohle by se dalo udělat dmsetupem, tady to dělám kvůli "read only" mapování.
losetup -r -o 512006144 /dev/loop0 /dev/sdb
Tohle zjistí velikost blokového zařízení v sektorech (je to pravděpodobně to samé, co vypisuje fdisk na prvním řádku).
blockdev --getsize /dev/loop0
FIXME Tenhle řádek připraví soubor s definicí mapování typu "celé blokové zařízení". To "xtable" je jen jméno, může být libovolné.
xtable:
0 975773156 linear /dev/loop0 0
Tohle vytvoří blokové zařízení /dev/mapper/base s výšeuvedeným xtable mapováním:
dmsetup create base xtable
Tohle vytvoří soubor dané velikosti (bude to odkládací prostor pro writable snapshot), pokud jde jen o dočasnost a vejde se to, dávám to do /dev/shm, velikost by měla být tak dvojnásobek předpokládaných úprav ve snapshotu - velikost těch úprav se odhaduje těžko, i jeden změněný sektor způsobí copy-on-write většího úseku.
dd if=/dev/zero of=fmodify bs=1M count=64
Tohle ze souboru vyrobí blokové zařízení (tady by to nešlo dmsetupem, protože ten umí pracovat jen s blokovými zařízeními):
losetup /dev/loop1 /root/disk/fmodify
FIXME Tohle je největší magie, vytvoří mapování typu copy-on-write snapshot, v podstatě to říká "při přístupu k novému zařízení (to se vytvoří v dalším kroku pomocí dmsetup create) použij daný úsek zařízení /dev/mapper/base, když tam někdo zapíše, tak změny ukládej do zařízení /dev/loop1 a to s granularitou 8 sektorů, tedy jakákoliv změna způsobí snapshot-mapping 4 kB (8*512B) bloku, ve kterém je ta změna. Při čtení to pak používá data z /dev/loop1, nejsou-li tam, pak z originálu, při zápisu jdou data vždy do /dev/loop1, pokud tam příslušné LBA už je, jen se přepíše, pokud není, udělá se nejdřív kopie bloku z originálu.
mtable:
0 975773156 snapshot /dev/mapper/base /dev/loop1 p 8
Tohle vytvoří ten snapshot podle definiční tabulky:
dmsetup create snap mtable
Tohle už je jen příklad, co se s tím pak dá dělat - fsck a rw mount fungují, ale nezapíše to ani bajt na původní zařízení /dev/sdb. Pokud dojde ten "snapshot modify space", začne to hlásit "device is read only".
reiserfsck --check /dev/mapper/snap
mount -t reiserfs /dev/mapper/snap /x