| Изменение номера inode файла в Linux | [исправить] | 
| Для некоторых специфических целей может понадобиться изменить номер inode у существующего файла, 
либо создать файл с заранее заданным номером. Штатными средствами сделать это -
задача нетривиальная,
однако с помощью модуля ядра это несложно. 
Создаем файл inode_modify.c следующего содержания: 
 #include <linux/module.h>
 #include <linux/kernel.h>
 #include <linux/version.h>
 #include <linux/fs.h>
 #include <linux/namei.h>
 #ifndef BUF_LEN
 #define BUF_LEN 256
 #endif
 char file[BUF_LEN];
 unsigned long new_num=0;
 module_param_string( name, file, BUF_LEN, 0);
 module_param(new_num, ulong, 0);
 struct nameidata nd;
 unsigned long get_number() {
	int error;
	error = path_lookup( file, 0, &nd);
	printk( KERN_ALERT "name = %s\n", file);
	if(error) {
		printk( KERN_ALERT "Can't access file\n");
		return -1;
	}
 #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,25)
	return nd.path.dentry->d_inode->i_ino;
 #else 
	return nd.dentry->d_inode->i_ino;
 #endif
 }
 unsigned long set_number(unsigned long new_num) {
 #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,25)
	nd.path.dentry->d_inode->i_ino = new_num;
	return nd.path.dentry->d_inode->i_ino;
 #else 
	nd.dentry->d_inode->i_ino = new_num;
	return nd.dentry->d_inode->i_ino;
 #endif
 }
 int inode_modify_init(){
	unsigned long inode_num;
	inode_num = get_number();
	printk ( KERN_ALERT "Inode number is %lu\n", inode_num);
	printk ( KERN_ALERT "New inode number is %lu\n",  set_number(new_num));
	return 0;
 }
 void inode_modify_exit(){
	printk(KERN_ALERT "Exiting...\n"); 
 }
 module_init(inode_modify_init);
 module_exit(inode_modify_exit);
 MODULE_LICENSE("GPL");
 MODULE_AUTHOR("Victor Leschuk <Victor.Leschuk@ebanat.com>");
И простой Makefile: 
 obj-m := inode_modify.o
После чего в директории с модулем: 
 $ make -C  /path/to/kernel/sources SUBDIRS=$PWD modules
Здесь нужно помнить, что исходники и версия gcc должны соответствовать тем, 
которые были использованы при сборке используемого ядра.
Далее тестируем модуль: 
 $ touch /dev/shm/test
 $ ls -i /dev/shm/test
 172461 /dev/shm/test
 $ sudo insmod ./inode_modify.ko name=/dev/shm/test new_num=12345
 $ ls -i /dev/shm/test
 12345 /dev/shm/test
 $ sudo rmmod inode_modify 
 $ dmesg |tail
 name = /dev/shm/test
 Inode number is 172461
 New inode number is 12345
 Exiting...
 | 
|  | 
| 
 | 
 | | Раздел:    Корень / Программисту и web-разработчику / C/C++, сборка, отладка | 
| | 1.1, dffd (?), 22:54, 30/09/2009  [ответить] | +/– |  | а в каких целях это можно использовать? ничего на ум не идёт. 
 |  |  | 
 
|  | | 2.2, Рус (?), 23:55, 30/09/2009 [^] [^^] [^^^] [ответить] | +/– |  | Не в обиду автору - в целях обмана rkhunter и прочих tripwire. 
 |  |  | 
 |  | | 3.3, Victor (??), 12:04, 01/10/2009 [^] [^^] [^^^] [ответить] | +/– |  | Я это использовал в целях репродакшна специфической ситуации для тестирования. /* Тестер я */
 |  |  | 
 | 
 | 
 
 | 1.4, Breg (??), 13:59, 01/10/2009  [ответить] | +/– |  | Ээээ... хорошего о себе мнения автор ))) См. MODULE_AUTHOR
 
 |  |  | 
 
 | 1.5, Breg (??), 14:32, 01/10/2009  [ответить] | +/– |  | А физически файл перемещается на диске? Можно это использовать чтобы все файлы, нужные для readahead, переместить, чтобы ускорить доступ к ним?
 
 |  |  | 
 
|  | | 2.6, Victor (??), 14:50, 01/10/2009 [^] [^^] [^^^] [ответить] | +/– |  | Нет, физически файл не перемещается. Изменяется только одно поле структуры struct inode. 
 |  |  | 
 | 
 
 | 1.7, pavlinux (ok), 21:40, 01/10/2009  [ответить] | +/– |  | pavel@amd64:/work/tmp> > ino1 pavel@amd64:/work/tmp> > ino2
 pavel@amd64:/work/tmp> ls -i1 ino[0-9]
 2924453 ino1
2924454 ino2
 pavel@amd64:/work/tmp> sudo /sbin/insmod ./ino.ko name=/work/tmp/ino2 new_num=2924453
 pavel@amd64:/work/tmp> dmesg | grep ino
 name = /work/tmp/ino2
New inode number is 2924453
 pavel@amd64:/work/tmp> ls -i1 ino[0-9]
 2924453 ino1
2924454 ino2
 Тo есть хрен.
 |  |  | 
 
|  | | 2.8, Victor (??), 14:01, 02/10/2009 [^] [^^] [^^^] [ответить] | +/– |  | Какое у вас ядро и ФС? Такой сценарий у меня на 2.6.30.8-64.fc11.i586 и ext3 вполне работает: [del@leshuk inode_modify]$ >ino1
[del@leshuk inode_modify]$ >ino2
 [del@leshuk inode_modify]$ ls -i1 ino[0-9]
 1142265 ino1
 1142266 ino2
 [del@leshuk inode_modify]$ sudo insmod ./inode_modify.ko name=ino2 new_num=1142265
 [del@leshuk inode_modify]$ ls -i1 ino[0-9]
 1142265 ino1
 1142265 ino2
 
 |  |  | 
 |  | | 3.9, pavlinux (ok), 00:23, 03/10/2009 [^] [^^] [^^^] [ответить] | +/– |  | >Какое у вас ядро и ФС? Такой сценарий у меня на 2.6.30.8-64.fc11.i586 2.6.32-rc1
 |  |  | 
 | 
 | 
 
 | 1.10, dmitry (??), 12:15, 06/10/2009  [ответить] | +/– |  |  крайне бесполезный и к томуже крайне ОПАСНЫЙ хак. Бесполезный потому сразу-же после изменения номера инода его могут выкинуть из памяти (если его не держит какой нибудь файл)
 Опасный потому что  в лучшем случае приведет к срабатыванию  какого нибудь BUG_ON(), а в хучшем разнесет вашу ФС на клочки. Например при попытке записи в файл мы пытаясь выделить блок попадем сюда. ext3_get_inode_block(inode->i_sb, inode->i_ino, iloc)
 Так что если уж очень хочется использовать этот способ то строго запретить запись через immutable
 или еще лучше перевести ФС в RO режим.
 А если нужен нормальный а не временный эффект с возможностью записи то стоит использовать e2fslib
(в случае etx234)  где можно выполнять любые операции с инодом
 А точнее выделить инод с нужным номером и скопировать в него i_blocks с исходного инода. Это будет работать по факту такой подхот используется online-defrag тулзой для etx4
 -dmonakhov
 |  |  | 
 
 
 |