Для некоторых специфических целей может понадобиться изменить номер 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
#endifchar 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...
URL:
Обсуждается: http://www.opennet.me/tips/info/2182.shtml
а в каких целях это можно использовать? ничего на ум не идёт.
Не в обиду автору - в целях обмана rkhunter и прочих tripwire.
Я это использовал в целях репродакшна специфической ситуации для тестирования./* Тестер я */
Ээээ... хорошего о себе мнения автор )))
См. MODULE_AUTHOR
А физически файл перемещается на диске?
Можно это использовать чтобы все файлы, нужные для readahead, переместить, чтобы ускорить доступ к ним?
Нет, физически файл не перемещается. Изменяется только одно поле структуры struct inode.
pavel@amd64:/work/tmp> > ino1
pavel@amd64:/work/tmp> > ino2
pavel@amd64:/work/tmp> ls -i1 ino[0-9]2924453 ino1
2924454 ino2pavel@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 2924453pavel@amd64:/work/tmp> ls -i1 ino[0-9]
2924453 ino1
2924454 ino2Тo есть хрен.
Какое у вас ядро и ФС? Такой сценарий у меня на 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
>Какое у вас ядро и ФС? Такой сценарий у меня на 2.6.30.8-64.fc11.i5862.6.32-rc1
крайне бесполезный и к томуже крайне ОПАСНЫЙ хак.
Бесполезный потому сразу-же после изменения номера инода его могут выкинуть из памяти (если его не держит какой нибудь файл)
Опасный потому что в лучшем случае приведет к срабатыванию какого нибудь BUG_ON(), а в хучшем разнесет вашу ФС на клочки. Например при попытке записи в файл мы пытаясь выделить блок попадем сюда. ext3_get_inode_block(inode->i_sb, inode->i_ino, iloc)
Так что если уж очень хочется использовать этот способ то строго запретить запись через immutable
или еще лучше перевести ФС в RO режим.А если нужен нормальный а не временный эффект с возможностью записи то стоит использовать e2fslib
(в случае etx234) где можно выполнять любые операции с инодом
А точнее выделить инод с нужным номером и скопировать в него i_blocks с исходного инода. Это будет работать по факту такой подхот используется online-defrag тулзой для etx4-dmonakhov