简析nvram的数据结构及流程

2019-07-13 05:53发布

应用层代码: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中的数据。