前言:

  • 最近接到个需求,我们新产品上的外包侧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. goioc:一个使用 Go 写的简易的 ioc 框架

    目录 goioc 介绍 快速上手 接口介绍 使用 goioc 如何使用 生命周期 实例化 获取对象 结构体字段依赖注入 Dispose 接口 反射形式使用 goioc 如何使用 接口.结构体.结构体指 ...

  2. 如何在 K8S 集群范围使用 imagePullSecret?

    在这篇文章中,我将向你展示如何在 Kubernetes 中使用 imagePullSecrets. imagePullSecrets 简介 Kubernetes 在每个 Pod 或每个 Namespa ...

  3. 数电第8周周结_by_yc

    基本知识: 1.有限状态机的分类: Moore型:输出仅与电路的状态有关: Mealy型:输出与当前电路状态和当前电路输入有关. 2.有限状态机的描述方法: 状态转换图:节点:状态(Moore输出): ...

  4. MySQL 的 NULL 值是怎么存储的?

    大家好,我是小林. 之前有位读者在面字节的时候,被问到这么个问题: 如果你知道 MySQL 一行记录的存储结构,那么这个问题对你没什么难度. 如果你不知道也没关系,这次我跟大家聊聊 MySQL 一行记 ...

  5. openpyxl写数据

    import osimport openpyxlos.chdir(r'D:/openpyxl') wb = openpyxl.Workbook() sht = wb.create_sheet('dat ...

  6. python 运算优先级

    python 运算优先级,请见下图

  7. Spring面试点汇总

    Spring面试点汇总 我们会在这里介绍我所涉及到的Spring相关的面试点内容,本篇内容持续更新 我们会介绍下述Spring的相关面试点: Spring refresh Spring bean Sp ...

  8. @ApiImplicitParams注解的详细使用

    一.@ApiImplicitParams注解的详细使用 业务需求: 1.根据服务员类别id(单个id)+服务员星级id(id的list)查询对应的服务员列表 1.controller代码: 点击查看代 ...

  9. 安装node.js与webpack创建vue2项目

    本文为博主原创,转载请注明出处: 1.安装node.js 下载地址:http://nodejs.cn/download/ (可查看历史版本) node.js 中文网:http://nodejs.cn/ ...

  10. 一文读懂Go Http Server原理

    hello大家好呀,我是小楼,这是系列文<Go底层原理剖析>的第二篇,依旧是分析 Http 模块,话不多说,开始. 从一个 Demo 入手 俗话说万事开头难,但用 Go 实现一个 Http ...