修改安卓ID为硬件唯一ID

前言:

  • 最近接到个需求,我们新产品上的外包侧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:关于安卓序列号产生的流程,可参考我另一篇笔记安卓ro.serialno产生的整个流程

  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

原文链接:https://www.cnblogs.com/erdongsir/p/17148263.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:修改安卓ID为硬件唯一ID - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • AR Engine毫秒级平面检测,带来更准确的呈现效果

    近年来,AR版块成为时下大热,这是一种将现实环境中不存在的虚拟物体融合到真实环境里的技术,用户借助显示设备可以拥有真实的感官体验。AR的应用场景十分广泛,涉及娱乐、社交、广告、购物、教育等领域:AR可以让游戏更具互动性;商品通过AR展示更真实;使用AR进行教育教学让抽象事物更形象等,可以说AR技术已经渗透人们生活的方方面面。 为了让人们产生强烈的视觉真实感,…

    Android 2023年4月18日
    00
  • Android报”NullPointerException”如何解决?

    当Android应用程序运行时,如果遇到无法解析的引用或访问已被销毁的对象时,会报 NullPointerException 异常。通常NullPointerException是由于一些常见的编程错误导致的。例如,尝试在null对象上调用方法或访问属性,或者使用未初始化的变量。 在Android应用程序的开发过程中,NullPointerException异…

    Android 2023年4月3日
    00
  • 【FAQ】关于JavaScript版本的华为地图服务Map的点击事件与Marker的点击事件存在冲突的解决方案

    一. 问题描述 创建地图对象,并添加marker标记,对map和marker均添加了点击事件; <body> <script> function initMap() { // 创建地图对象 const map = new HWMapJsSDK.HWMap(document.getElementById(‘map’), { center…

    Android 2023年4月27日
    00
  • 投放视频广告时,如何快速与第三方播放器兼容?

    新媒体时代,广告样式越来越丰富。相较于传统的图文信息,视频类广告更具有直观性,能够让消费者在了解产品知识和功能的同时加深对产品的印象。 因此在各类网站或App上投放视频类广告是个很好的宣传方式,但广告商们如果想在网站上展示视频广告,必须确保视频广告投放协议与发布渠道的播放器兼容;如果不能兼容,广告商或者视频广告投放平台必须创建不同的响应,来与特定视频播放器兼…

    Android 2023年4月17日
    00
  • 实现自有App上架小游戏其实并非难事

    近年来小程序游戏迎来了爆发式增长。微信、支付宝、抖音等各大平台小程序游戏愈加丰富,你是否也让自己的App也拥有运行丰富的小游戏的能力?今天就来带大家看看如何实现。 我们先来看看各互联网巨头关于小游戏生态的特征: 「微信」 率先推出小程序,具有先发优势的微信平台毫无疑问走在了市场前列。根据2021阿拉丁发布的小游戏行业年度报告显示,微信小游戏已超过10w的开发…

    Android 2023年4月18日
    00
  • scrollView 嵌套 recyclerview 时 BaseQuickAdapter 九宫格图片拖拽到底部删除

    九宫格图片布局,长按直接拖拽图片,长按时显示底部删除布局,拖拽到删除布局处松手可删除布局,最后添加按钮不可拖拽,基于 BaseQuickAdapter 基础上实现 BaseQuickAdapter 确实很好用,简化我们的实现代码,它本身也集成了一套拖拽实现,不过目前无法完美的满足上面的需求,需要做一些修改 1、首先自定义好九宫格布局,末尾是一个 + 号,这个…

    Android 2023年4月18日
    00
  • 跑步课程导入能力,助力科学训练

    HUAWEI Health Kit为开发者提供用户自定义的跑步课程导入接口,便于用户在华为运动健康App和华为智能穿戴设备上查看来自生态应用的训练课表,开启科学、适度的运动训练。 跑步课程导入能力支持生态应用在获取用户的华为帐号授权后,将跑步课程数据写入至华为运动健康App,并在已有的华为智能穿戴设备连接并支持课程导入时,直接将课表推送到设备上,用户可以轻松…

    Android 2023年5月11日
    00
  • HMS Core 6.10.0版本发布公告

    分析服务 ◆ 事件分析下新增商品订阅分析报告,帮助开发者了解应用内用户付费订阅概况,评估订阅付费价值; ◆ 营销分析、用户质量、转化分析以及过滤器中,新增广告系列/广告任务通过ID进行搜索的功能,通过更便捷高效的数据分析体验,帮助开发者合理评估广告投放的后端转化效果。 查看详情>> 运动健康服务 ◆ 血糖数据类型在原有指尖血糖数据开放的基础上,新…

    Android 2023年5月6日
    00
合作推广
合作推广
分享本页
返回顶部