前言:

  • 最近接到个需求,我们新产品上的外包侧APP需要使用硬件唯一ID(不管怎么升级怎么操作,ID始终不变和硬件绑定),用来做权限校验。
  • 由于了解到安卓ID或序列号都会在擦除升级后重新随机生成,所以这里使用硬件上的ID来作为唯一ID,接下来进入正题
  • 此篇以安卓7.1系统为例

一,常用硬件信息ID

这里列举一些常用的一芯一码ID查询获取方式

1. CPU ID

我们当前所使用的主芯片RK3128上没有CPU id,此处也举个例

shell命令:

cat /proc/cpuinfo | grep Serial

结果如下:(rk3128上没有固定ID,所以显示的为0)

Serial          : 0000000000000000

2. eMMC/Flash ID

使用存储芯片eMMC(Embedded Multi Media Card)/Flash的cid

shell命令:

 cat /sys/bus/mmc/devices/mmc0:0001/cid

这里的mmc0:0001可能为其他地址,请按实际来,结果如下:

150100424a5444345203e977be8f4963

此处的Cid的32字节的字串,格式如下:

MID: [127:120]   —— 8bit(1Byte)Manufacturer ID,由MMCA分配,比如Sandisk为0x02,Kingston为0x37,Samsung为0x15。

OID: [119:104]   —— 16bit OEN/Application ID,OEM/应用ID号,也由MMCA分配。

PNM: [103:64]    —— 40bit Product Name,产品名称。

PRV: [63:56]     —— 8bit Product revision,产品版本,前4bit fw版本,后4bit hw版本。

PSN: [55:24]     —— 32bit Product serial number,产品序列号。

MDT: [19:8]     —— 12bit Manufacturing date,生产日期,前4bit是月份,后8bit为年份,0对应2000年。

CRC: [7:1]       —— 7bit CRC7 checksum,循环冗余校验。

3. 其他ID

因为每个平台所配置的外设不一样,实际还需根据情况获取。

二,应用硬件ID

在应用硬件ID之前,我们先把安卓framework层中的随机生成安卓ID的部分修改了

1. 修改framework安卓ID生成源数据

  1. 进到android系统源码里,目录:

frameworks/base/packages/SettingsProvider/src/com/android/providers/settings

  1. 打开SettingsProvider.java

先improt相关的包:

import android.os.SystemProperties;

然后修改private void ensureSecureSettingAndroidIdSetLocked(SettingsState secureSettings)方法中的 String androidId

//建议把后面的默认值改为固定一个ID,个人感觉固定比随机好
String androidId = SystemProperties.get("ro.serialno", Long.toHexString(new SecureRandom().nextLong()));

PS:原理就是通过序列号的固定值替换生成的随机值。当然,具体用哪个属性值去替换,由咱们自己决定。此处以ro.serialno为例

2. 获取硬件值ID应用到属性

通过上个步骤我们已经把安卓ID给固定到了ro.serialno属性值上,下面我们就修改这个属性值

ps:关于安卓序列号产生的流程,可参考我另一篇笔记

  1. 让ro.serialno不再从cmdline上获取

a. 打开安卓系统源码: system/core/init/init.cpp

b. 找到export_kernel_boot_props这个函数

c. 注释掉prop_map结构体数组中的这一组值

static void export_kernel_boot_props() {
char cmdline[1024];
char* s1;
char* s2;
char* s3;
char* s4; struct {
const char *src_prop;
const char *dst_prop;
const char *default_value;
} prop_map[] = {
//{ "ro.boot.serialno", "ro.serialno", "", },注释掉
{ "ro.boot.mode", "ro.bootmode", "unknown", },
{ "ro.boot.baseband", "ro.baseband", "unknown", },
{ "ro.boot.bootloader", "ro.bootloader", "unknown", },
{ "ro.boot.hardware", "ro.hardware", "unknown", },
{ "ro.boot.revision", "ro.revision", "0", },
}; //if storagemedia is emmc, so we will wait emmc init finish
for (int i = 0; i < EMMC_RETRY_COUNT; i++) {
proc_read( "/proc/cmdline", cmdline, sizeof(cmdline) );
s1 = strstr(cmdline, STORAGE_MEDIA);
s2 = strstr(cmdline, "androidboot.mode=emmc");
s3 = strstr(cmdline, "storagemedia=nvme");
s4 = strstr(cmdline, "androidboot.mode=nvme");
....

如上即可

  1. 将硬件ID应用到属性

    这里主要是将获取到的ID使用property_set设进sys.serialno这个属性里,然后再init.rc里通过如下设进ro.serialno这个属性里

    # set ro.serialno
    on property:sys.serialno=*
    setprop ro.serialno ${sys.serialno}

    主要修改drmservice服务,路径:system/core/drmservice/drmservice.c

    diff patch如下:

    diff --git a/drmservice/drmservice.c b/drmservice/drmservice.c
    old mode 100644
    new mode 100755
    index 86c8e32..cdad0d4
    --- a/drmservice/drmservice.c
    +++ b/drmservice/drmservice.c
    @@ -29,6 +29,7 @@
    #define DEVICE_SERIALNO "/data/misc/wifi/serialno"
    #define USB_SERIAL_PATH "/sys/class/android_usb/android0/iSerial"
    #define USB_SERIAL_PATH1 "/config/usb_gadget/g1/strings/0x409/serialnumber"
    +#define EMMC_CID_PATH "/sys/bus/mmc/devices/mmc0:0001/cid" extern int init_module(void *, unsigned long, const char *);
    extern int delete_module(const char *, unsigned int);
    @@ -703,12 +704,13 @@ void generate_device_serialno(int len,char*result)
    {
    int temp=0,rand_bit=0,times =0;
    int fd,type;
    - char buf[32];
    + char buf[33];
    char value[6][2];
    const char *bufp;
    ssize_t nbytes;
    char path[64];
    unsigned int seed[2]={0,0};
    + len=len>32?32:len; #ifdef DEBUG_RANDOM
    SLOGE("-------DEBUG_RANDOM mode-------");
    @@ -720,7 +722,70 @@ void generate_device_serialno(int len,char*result)
    SLOGE("----------serianno =%s",result);
    return;
    }
    + #if 1
    + //通过CPU ID应用为安卓ID
    + char cpuinfobuf[256] = {0};
    + char *buf_pos = cpuinfobuf;
    + char *result_pos = result;
    + FILE *fp = fopen("/proc/cpuinfo", "r");
    + if(NULL != fp)
    + {
    + while(!feof(fp))
    + {
    + memset(cpuinfobuf,0, sizeof(cpuinfobuf));
    + fgets(cpuinfobuf,sizeof(cpuinfobuf)-1, fp);
    + if(strstr(cpuinfobuf,"Serial"))//找到包含Serial这一行
    + {
    + while(*(buf_pos++) != ':');//找到:这一个字符
    + while(*(++buf_pos))
    + {
    + *(result_pos++) = *buf_pos;
    + }
    + *result_pos = '\0';
    + break;
    + }
    + }
    + fclose(fp);
    + }
    + else
    + {
    + SLOGE("failed to open cpuinfo\n");
    + } + #else
    //通过eMMC ID应用为安卓ID
    + fd = open(EMMC_CID_PATH, O_RDONLY);
    + if(fd<0)
    + {
    + srand(time(0));
    + if(DEBUG_LOG)
    + SLOGE("------------emmc cid has been cached ,but open failed,SLOGE=%s\n",strerror(errno));
    + goto mac_gen;
    + }
    + nbytes = read(fd, buf, 32);//max length 32 byte
    + close(fd);
    +
    + if (nbytes < 0) {
    + srand(time(0));
    + if(DEBUG_LOG)
    + SLOGE("-------------read fd failed\n");
    + goto mac_gen;
    + }
    + buf[nbytes] = '\0';
    + bufp = buf;
    + if(DEBUG_LOG)
    + SLOGE("---------read %s =%s,len=%d",EMMC_CID_PATH,bufp,nbytes);
    + //优先取后 len长的字节,因为cid前16位基本一致,容易造成多个机器一个id的情况
    + if(nbytes>=len){
    + memcpy(result,bufp+(nbytes-len),len);
    + result[len]='\0';
    + } else {
    + memcpy(result,bufp,nbytes);
    + result[nbytes]='\0';
    + }
    + #endif
    + store_serialno(result);//存到data目录某个文件当中,这样再启动时就不会再走一次这个流程
    + SLOGE("-------------generate_device_serialno,len =%d,result=%s-------------",len,result);
    + return;
    +mac_gen:
    if(check_wlan_mac()<0)//not buffered in data,do it
    {
    fd = open(WIFI_MAC_FILENAME, O_RDONLY);//read form buffered file
    @@ -1101,8 +1166,10 @@ int main( int argc, char *argv[] )
    }
    else//auto generate serialno
    {
    - generate_device_serialno(10,sn_buf_auto);
    - property_set("sys.serialno", sn_buf_auto[0] ? sn_buf_auto : "");
    + generate_device_serialno(16,sn_buf_auto);
    +
    + //SLOGE("----------------sn_buf_auto:%s ---------------",sn_buf_auto);
    + property_set("sys.serialno", strlen(sn_buf_auto)>0 ? sn_buf_auto : "");
    write_serialno2kernel(sn_buf_auto);
    SLOGE("auto generate serialno,serialno = %s",sn_buf_auto);
    }

    3. 查看结果

    将上面patch应用到项目当中,通过如下命令可查看是否生效:

    adb shell settings get secure android_id #查看安卓ID
    adb get-serialno #查看序列号

    例:

    PS E:\> adb get-serialno
    5203e977be8e4975
    PS E:\> adb shell settings get secure android_id
    5203e977be8e4975
    PS E:\>

    end

    感谢阅读~

    希望能帮到你~

    see you~

    码字不易,转载请注明原作者 ~ (from:https://erdong.work

修改安卓ID为硬件唯一ID的更多相关文章

  1. 分布式全局唯一ID生成策略

    为什么分布式系统需要用到ID生成系统 在复杂分布式系统中,往往需要对大量的数据和消息进行唯一标识.如在美团点评的金融.支付.餐饮.酒店.猫眼电影等产品的系统中,数据日渐增长,对数据库的分库分表后需要有 ...

  2. 只要单片机具有真正唯一ID,就可以让加密坚不可摧(转)

    源:只要单片机具有真正唯一ID,就可以让加密坚不可摧 http://www.amobbs.com/thread-5518980-1-1.html 第一环:ID-->F1(ID) -----> ...

  3. 分布式唯一ID生成服务

    SNService是一款基于分布式的唯一ID生成服务,主要用于提供大数量业务数据建立唯一ID的需要;服务提供最低10K/s的唯一ID请求处理.如果你部署服务的CPU资源达到4核的情况下那该服务最低可以 ...

  4. 高并发分布式系统中生成全局唯一Id汇总

    数据在分片时,典型的是分库分表,就有一个全局ID生成的问题.单纯的生成全局ID并不是什么难题,但是生成的ID通常要满足分片的一些要求:   1 不能有单点故障.   2 以时间为序,或者ID里包含时间 ...

  5. 分布式系统唯一ID生成方案汇总

    系统唯一ID是我们在设计一个系统的时候常常会遇见的问题,也常常为这个问题而纠结.生成ID的方法有很多,适应不同的场景.需求以及性能要求.所以有些比较复杂的系统会有多个ID生成的策略.下面就介绍一些常见 ...

  6. STM32全球唯一ID读取方法

    产品唯一的身份标识非常适合:● 用来作为序列号(例如USB字符序列号或者其他的终端应用)● 用来作为密码,在编写闪存时,将此唯一标识与软件加解密算法结合使用,提高代码在闪存存储器内的安全性.● 用来激 ...

  7. 如何检索Android设备的唯一ID

    关于本文档 Android的开发者在一些特定情况下都需要知道手机中的唯一设备ID.例如,跟踪应用程序的安装,生成用于复制保护的DRM时需要使用设备的唯一ID.在本文档结尾处提供了作为参考的示例代码片段 ...

  8. snowflake 分布式唯一ID生成器

    本文来自我的github pages博客http://galengao.github.io/ 即www.gaohuirong.cn 摘要: 原文参考运维生存和开源中国上的代码整理 我的环境是pytho ...

  9. 全局唯一ID发号器的几个思路

    标识(ID / Identifier)是无处不在的,生成标识的主体是人,那么它就是一个命名过程,如果是计算机,那么它就是一个生成过程.如何保证分布式系统下,并行生成标识的唯一与标识的命名空间有着密不可 ...

  10. 分布式系统唯一ID的生成方案讨论

    在分布式系统下唯一id问题,就是id咋生成?比如分表分库,因为要是一个表分成多个表之后,每个表的id都是从1开始累加自增长,那是不对的.举个例子,一个表拆分为了2张表,每个表的id都从1开始累加,这个 ...

随机推荐

  1. 协程Part1-boost.Coroutine.md

    首先,在计算机科学中 routine 被定义为一系列的操作,多个 routine 的执行形成一个父子关系,并且子 routine 一定会在父 routine 结束前结束,也就是一个个的函数执行和嵌套执 ...

  2. 【算法题型总结】--6、BFS

    // 计算从起点 start 到终点 target 的最近距离 int BFS(Node start, Node target) { Queue<Node> q; // 核心数据结构 Se ...

  3. 自研分布式高性能RPC框架及服务注册中心实践笔记【原创】【开源】

    痛点 1. bsf底层依赖springcloud,影响bsf更新springboot新版本和整体最新技术版本升级. 2. eureka已经闭源,且框架设计较重,同时引入eureka会自行引入较多spr ...

  4. Prometheus高可用架构介绍

    Prometheus作为新生代的开源监控系统,慢慢成为了云原生体系的监控事实标准,也证明了其设计得到业界认可.但在多集群,大集群等场景下,Prometheus由于没有分片能力和多集群支持,还有Prom ...

  5. Clickhouse表引擎探究-ReplacingMergeTree

    作者:耿宏宇 1 表引擎简述 1.1 官方描述 MergeTree 系列的引擎被设计用于插入极大量的数据到一张表当中.数据可以以数据片段的形式一个接着一个的快速写入,数据片段在后台按照一定的规则进行合 ...

  6. JavaScript:控制跳转:break、continue与标签

    在循环结构中,经常需要使用关键字break和continue来控制跳转: 遇到break,就会跳出循环结构,执行循环体后面的代码: 遇到continue,就会跳出本次循环,进入下一次循环: 那么,假如 ...

  7. day14-功能实现13

    家居网购项目实现013 以下皆为部分代码,详见 https://github.com/liyuelian/furniture_mall.git 32.功能30-会员不能登录后台管理 32.1需求分析/ ...

  8. [python]《Python编程快速上手:让繁琐工作自动化》学习笔记2

    1. 读写文件笔记(第8章)(代码下载) 1.1 文件与文件路径 通过import os调用os模块操作目录,常用函数如下: 函数 用途 os.getcwd() 取得当前工作路径 os.chdir() ...

  9. 【大型软件开发】浅谈大型Qt软件开发(二)面向未来开发——来自未来的技术:COM组件。我如何做到让我们的教学模块像插件一样即插即用,以及为什么这么做。

    前言 最近我们项目部的核心产品正在进行重构,然后又是年底了,除了开发工作之外项目并不紧急,加上加班时间混不够了....所以就忙里偷闲把整个项目的开发思路聊一下,以供参考. 鉴于接下来的一年我要进行这个 ...

  10. JS逆向之浏览器补环境详解

    JS逆向之浏览器补环境详解 "补浏览器环境"是JS逆向者升职加薪的必备技能,也是工作中不可避免的操作. 为了让大家彻底搞懂 "补浏览器环境"的缘由及原理,本文将 ...