您现在的位置是:主页 > news > 马卡龙网站建设方案/360seo
马卡龙网站建设方案/360seo
admin2025/4/24 7:58:51【news】
简介马卡龙网站建设方案,360seo,做张家界旅游网站多少钱,在线翻墙前言: Vendor storage 被设计用来存储 SN,MAC,LAN,BT 等 vendor data。 特征: 唯一的访问 ID;可靠的数据验证;掉电不会丢失;系统启动各个阶段都可访问;PC 端可读写&…
前言:
Vendor storage 被设计用来存储 SN,MAC,LAN,BT 等 vendor data。
特征:
- 唯一的访问 ID;
- 可靠的数据验证;
- 掉电不会丢失;
- 系统启动各个阶段都可访问;
- PC 端可读写;
- Kernel 可读写;
- Linux Application 可读写;
如下图:
Data Layout:
也就是说 vendor storage 是从 eMMC 3.5M 之后开始存放,每个元素是 64K,一共 4 块总共 256K。
系统一共把 vendor 的存储块分成 4 个分区,vendor0、vendor1、vendor2、vendor3。每个 vendorX 的 hdr 里都有一个单调递增的 Version 字段用于表示 vendorX 被更新的时刻点。每次读操作只读取新的 vendorX(即 vendor 大),写操作的时候会更新 Version、并且把整个原有信息和新增信息搬移到下一个 vendor 分区里。例如当前 从 vendor2读取到信息,经过修改后再回写,此时写入的是 vendor3。这样做只是为了起到一个简单的安全防护作用。
存储数据结构:
在代码中用 struct vendor_info 表示:
struct vendor_info {u32 tag;u32 version;u16 next_index;u16 item_num;u16 free_offset;u16 free_size;struct vendor_item item[126]; /* 126 * 8*/u8 data[EMMC_VENDOR_PART_SIZE * 512 - 1024 - 8];u32 hash;u32 version2;
};
各个阶段的驱动文件及接口如下, 两个阶段的初始化以及读写接口的本质实现是一样的。
uboot:
驱动文件: storage.c
初始化接口: vendor_storage_init();
读写接口: vendor_storage_read()/vendor_storage_write();
接口会在rockusb也就是loader模式被使用,可查看rkusb_handle_datatx()以及rkusb_handle_datarx()两个函数。
kernel:
驱动文件: sdmmc_vendor_storage.c和rk_vendor_storage.c
初始化接口:emmc_vendor_storage_init(), 它被放在线程vendor_init_thread()中实现是因为
1.不能阻塞kernel驱动初始化过程
2.需要等待eMMC初始化完成。
接口一方面提供给用户空间使用,通过vendor_storage_ioctl()接口实现。
另外读写函数也被注册为公共接口(通过rk_vendor_register()实现),供kernel使用,比如wifi模块的get_wifi_addr_vendor()会去读写Wifi MAC ID.
用户空间:
在 /system/eyeknowproj/vendoroper/ 目录下,有我们的项目源码:
Android.mk:
LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES := vendorlib.c
LOCAL_SHARED_LIBRARIES := liblog libm libc libprocessgroup
LOCAL_CFLAGS := -Werror
LOCAL_MODULE := libvendoroper
LOCAL_C_INCLUDES := $(LOCAL_PATH)/
include $(BUILD_SHARED_LIBRARY)include $(CLEAR_VARS)
LOCAL_SRC_FILES := vendor_test.c
LOCAL_SHARED_LIBRARIES := libvendoroper liblog libm libc libprocessgroup
LOCAL_CFLAGS := -Werror
LOCAL_MODULE := vendor_test
LOCAL_C_INCLUDES := $(LOCAL_PATH)/
include $(BUILD_EXECUTABLE)
vendorlib.c:
#include "vendorlib.h"
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>int vendor_read(enum VENDOR_ID vid,char* buf,int len)
{int ret;uint8 p_buf[sizeof(struct rk_vendor_req)+64];struct rk_vendor_req *req;req = (struct rk_vendor_req *)p_buf;int sys_fd = open("/dev/vendor_storage",O_RDWR,0); if(sys_fd < 0){printf("vendor storage open fail");return -1;}req->tag = VENDOR_REQ_TAG;req->id = vid;req->len = len;ret = ioctl(sys_fd, VENDOR_READ_IO, req);if (!ret) {memcpy(buf, req->data, len);}close(sys_fd);return 0;
}
int vendor_write(enum VENDOR_ID vid,char* buf,int len)
{int ret;uint8 p_buf[sizeof(struct rk_vendor_req)+64];struct rk_vendor_req *req;int i;int sys_fd = open("/dev/vendor_storage",O_RDWR,0); req = (struct rk_vendor_req *)p_buf;if(sys_fd < 0){printf("vendor storage open fail");return -1;}req->tag = VENDOR_REQ_TAG;req->id = vid;req->len = len;for (int i = 0; i < len; i++)req->data[i] = buf[i];ret = ioctl(sys_fd, VENDOR_WRITE_IO, req);if (ret) {printf("vendor write fail\n");return -1;}close(sys_fd);return 0;
}
vendorlib.h:
#include <sys/ioctl.h>typedef unsigned short uint16;
typedef unsigned int uint32;
typedef unsigned char uint8;#define VENDOR_REQ_TAG 0x56524551
#define VENDOR_READ_IO _IOW('v', 0x01, unsigned int)
#define VENDOR_WRITE_IO _IOW('v', 0x02, unsigned int)enum VENDOR_ID
{VENDOR_SN_ID = 1,VENDOR_WIFI_MAC_ID = 2,VENDOR_LAN_MAC_ID = 3,VENDOR_BLUETOOTH_ID = 4,VENDOR_USER1 = 16,VENDOR_USER2 = 17,
};#define VENDOR_USER_LENGTH 32
struct rk_vendor_req {uint32 tag;uint16 id;uint16 len;uint8 data[1];
};
int vendor_read(enum VENDOR_ID vid,char* buf,int len);
int vendor_write(enum VENDOR_ID vid,char* buf,int len);
vendor_test.c:
#include "vendorlib.h"
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>int main()
{char temp[64] = {'k','e','l','l','a','n','d','s',};char temp1[64] = {};vendor_write(VENDOR_USER1,temp,VENDOR_USER_LENGTH);vendor_read(VENDOR_USER1,temp1,VENDOR_USER_LENGTH);printf("%c,%c,%c,%c,%c,%c,%c\n",temp1[0],temp1[1],temp1[2],temp1[3],temp1[4],temp1[5],temp1[6]);return 0;
}
小结:
- 由于 Vendor Storage 位于 eMMC 上,所以做擦除的时候也会一并丢失;
- 重新烧写固件或者擦除 idb 是不会丢失的。
- 工具中目前是不支持添加新 id 的,如果要添加,那就要改工具源代码,或者直接通过驱动提供给用户空间的接口来添加新 id。
参考:
Rockchip Vendor Storage Application Note.pdf
Rockchip-Developer-Guide-UBoot-nextdev.pdf
[RK3399][Android7.1] Vendor Storage区域知识及探讨_Kris Fei's blog-CSDN博客