====== 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