Изменение номера 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 <[email protected]>");
И простой 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
| |
|