The OpenNET Project / Index page

[ новости /+++ | форум | теги | ]

Изменение номера 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...
 
30.09.2009 , Автор: Victor Leschuk
Ключи: linux, module, kernel, module, inode, fs / Лицензия: CC-BY
Раздел:    Корень / Программисту и web-разработчику / C/C++, сборка, отладка

Обсуждение [ RSS ]
  • 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

     

     Добавить комментарий
    Имя:
    E-Mail:
    Заголовок:
    Текст:




    Партнёры:
    PostgresPro
    Inferno Solutions
    Hosting by Hoster.ru
    Хостинг:

    Закладки на сайте
    Проследить за страницей
    Created 1996-2024 by Maxim Chirkov
    Добавить, Поддержать, Вебмастеру