应用层代码:sourceliblibnvram
vram_env.cmtd层代码:sourcelinux-2.6.36.xarchmips
alink
vram.c数据结构typedef struct block_s {
char *name;
env_t env; //env block
cache_t cache[MAX_CACHE_ENTRY]; //env cache entry by entry
unsigned long flash_offset;
unsigned long flash_max_len; //ENV_BLK_SIZE
char valid;
char dirty;
} block_t;
typedef struct cache_environment_s {
char *name;
char *value;
} cache_t;
typedef struct environment_s {
unsigned long crc; //CRC32 over data bytes
char *data;
} env_t;
流程应用 -----------------------------------------内核 -----------------------flashralink_init.c-----nvram_env.c----------------nvram.c--------------------flash内核函数:
ioctlint ralink_nvram_ioctl(struct inode *inode, struct file *file, unsigned int req,
unsigned long arg)
{
switch (req) {
case RALINK_NVRAM_IOCTL_GET:
nvr = (nvram_ioctl_t __user *)arg;
p = nvram_get(nvr->index, nvr->name);
if (p == NULL)
p = "";
if (copy_to_user(nvr->value, p, strlen(p) + 1))
return -EFAULT;
break;
case RALINK_NVRAM_IOCTL_GETALL:
nvr = (nvram_ioctl_t __user *)arg;
index = nvr->index;
len = fb[index].flash_max_len - sizeof(fb[index].env.crc);
if (nvram_getall(index, fb[index].env.data) == 0) {
if (copy_to_user(nvr->value, fb[index].env.data, len))
return -EFAULT;
}
break;
case RALINK_NVRAM_IOCTL_SET:
nvr = (nvram_ioctl_t *)arg;
value = (char *)kmalloc(MAX_VALUE_LEN, GFP_KERNEL);
if (!value)
return -ENOMEM;
if (copy_from_user(value, nvr->value, strlen(nvr->value) + 1)) {
kfree(value);
return -EFAULT;
}
nvram_set(nvr->index, nvr->name, value);
kfree(value);
break;
case RALINK_NVRAM_IOCTL_COMMIT:
nvr = (nvram_ioctl_t __user *)arg;
nvram_commit(nvr->index);
break;
case RALINK_NVRAM_IOCTL_CLEAR:
nvr = (nvram_ioctl_t __user *)arg;
nvram_clear(nvr->index);
default:
break;
}
}
应用调用相应的命令后会进行到ioctl这个函数中。在ioctl这个函数中,内核调用copy_to_user()和copy_from_user()这两个函数来完成数据在用户态和内核态之间的交互。 应用调用这些命令前,都会进行一个初始化nvram_init(),在这个初始化函数中,会获取所有的nvram信息1.nvram_show 初始化后会调用nvram_buflist显示所有信息。2.nvram_get 初始化后,调用nvram_bufget,进入内核的ioctl,通过name找到对应的值的,用copy_to_user()将值返回给应用,然后应用在内存中找到相应的名字,将值赋给对应name的值3.nvram_set 初始化后,调用nvram_set,然后进入内核的ioctl,调用内核的nvram_set(),然后用copy_from_user()将值从应用传到内核,最后标记为dirty。接着在应用中调用nvram_commit,通过ioctl进入内核,调用内核的nvram_commit(),将crc和数据分别写入config分区后返回应用,应用再用新的config覆盖kernel的最后的一个块的旧的备份数据。Question: 1. 什么时候会启用备用机制里面的数据? 初始化nvram块设备的时候,会读取conifg分区里面的crc和数据,然后通过数据算出新的crc和读出来的crc做比较,如果不一致,然后再读取kernel里的备份数据和crc,通过数据算出新的crc与读出来的scr做比较,如果一致说明,备份数据是好的,config里面的数据是坏的。最后读取备份数据,覆盖config中的数据。