高通 Android 12 调试产测NFC功能

1、在dev/nxpnfc节点添加对应的文件权限

on property:sys.boot_completed=1

# nfc add by zm

chmod 777 /dev/nxpnfc

2、在vendor/nxp/nfcdevice-nfc.mk文件中 修改NFC添加到编译路径如下所示,跟平时内置apk方式有点类似

PRODUCT_PACKAGES += NFCTestApp

-include \vendor\nxp\nfc\FactoryTestApp\Android.mk

3、然后执行make命令 这时候就会在out目录下面system/bin目录生成 NFCTestApp可执行文件

4、产测中执行shell命令 代码如下

val nfcShell = Shell.exe("/system/bin/NFCTestApp")

if(nfcShell.contains("xxx"){

requireActivity().runOnUiThread(Runnable {

textView!!.text = "识别NFC成功"

})

}

5、其他原生识别NFC的方式

6、Android原生识别NFC

package com.example.nfctest;

import androidx.appcompat.app.AppCompatActivity;

import android.nfc.tech.Ndef;

import android.os.Bundle;

import android.app.PendingIntent;

import android.content.Intent;

import android.nfc.NfcAdapter;

import android.nfc.Tag;

import android.os.Bundle;

import android.text.TextUtils;

import android.util.Log;

import android.widget.TextView;

import android.widget.Toast;

import java.util.List;

public class MainActivity extends AppCompatActivity {

private NfcAdapter nfcAdapter;

private PendingIntent pendingIntent;

private TextView tvUid;

String datas;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

tvUid = (TextView) findViewById(R.id.tv_uid);

nfcAdapter = NfcAdapter.getDefaultAdapter(this);

/*pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this,

getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);*/

if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.S) {

mPendingPollIntent = PendingIntent.getBroadcast(mContext, POLL_REQUEST, pollIntent, PendingIntent.FLAG_IMMUTABLE);

}else{

mPendingPollIntent = PendingIntent.getBroadcast(mContext, POLL_REQUEST, pollIntent, 0);

}

if (nfcAdapter == null) {

Toast.makeText(MainActivity.this,"设备不支持NFC",Toast.LENGTH_LONG).show();

return;

}

if (nfcAdapter != null && !nfcAdapter.isEnabled()) {

Toast.makeText(MainActivity.this, "请在系统设置中先启用NFC功能", Toast.LENGTH_LONG).show();

return;

}

//onNewIntent(getIntent());

}

@Override

protected void onNewIntent(Intent intent) {

super.onNewIntent(intent);

resolveIntent(intent);

// resolveIntent(intent);

}

// void resolveIntent(Intent intent) {

// Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);

// if (tag != null) {

// processTag(intent);

// }

// }

public void processTag(Intent intent) {//处理tag

String uid = "";

Tag tagFromIntent = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);

byte[] aa = tagFromIntent.getId();

uid += bytesToHexString(aa);//获取卡的UID

Log.e("processTag","uid="+uid);

tvUid.setText(uid);

}

// //字符序列转换为16进制字符串

// private String bytesToHexString(byte[] src) {

// StringBuilder stringBuilder = new StringBuilder("0x");

// if (src == null || src.length <= 0) {

// return null;

// }

// char[] buffer = new char[2];

// for (int i = 0; i < src.length; i++) {

// buffer[0] = Character.forDigit((src[i] >>> 4) & 0x0F, 16);

// buffer[1] = Character.forDigit(src[i] & 0x0F, 16);

// stringBuilder.append(buffer);

// }

// return stringBuilder.toString();

// }

/**

* 数组转换成十六进制字符串

*

* @param bArray

* @return

*/

public static String bytesToHexString(byte[] bArray) {

StringBuffer sb = new StringBuffer(bArray.length);

String sTemp;

for (int i = 0; i < bArray.length; i++) {

sTemp = Integer.toHexString(0xFF & bArray[i]);

if (sTemp.length() < 2)

sb.append(0);

sb.append(sTemp.toUpperCase());

}

return sb.toString();

}

@Override

protected void onPause() {

super.onPause();

if (nfcAdapter != null)

nfcAdapter.disableForegroundDispatch(this);

}

@Override

protected void onResume() {

super.onResume();

if (!this.nfcAdapter.isEnabled()) {

Toast.makeText(this,"请在系统设置中先启用NFC功能",Toast.LENGTH_SHORT).show();

}

if (nfcAdapter != null)

nfcAdapter.enableForegroundDispatch(this, pendingIntent,

null, null);

}

protected void resolveIntent(Intent intent) {

// 得到是否检测到TAG触发

if (NfcAdapter.ACTION_TECH_DISCOVERED.equals(intent.getAction())

|| NfcAdapter.ACTION_TAG_DISCOVERED.equals(intent.getAction())

|| NfcAdapter.ACTION_NDEF_DISCOVERED.equals(intent.getAction())) {

// 处理该intent

Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);

// 获取标签id数组

byte[] bytesId = tag.getId();

//获取消息内容

NfcMessageParser nfcMessageParser = new NfcMessageParser(intent);

List tagMessage = nfcMessageParser.getTagMessage();

if (tagMessage == null || tagMessage.size() == 0) {

//Toast.makeText(this, "NFC格式不支持...", Toast.LENGTH_LONG).show();

} else {

for (int i = 0; i < tagMessage.size(); i++) {

Log.e("tag", tagMessage.get(i));

}

datas = tagMessage.get(0);

}

String info = "";

if (datas != null) {

info += "内容:" + datas + "\n卡片ID:" + bytesToHexString(bytesId) + "\n";

} else {

info += "卡片ID:" + bytesToHexString(bytesId) + "\n";

}

String[] techList = tag.getTechList();

//分析NFC卡的类型: Mifare Classic/UltraLight Info

String cardType = "";

for (String aTechList : techList) {

if (TextUtils.equals(aTechList, "android.nfc.tech.Ndef")) {

Ndef ndef = Ndef.get(tag);

cardType += "最大数据尺寸:" + ndef.getMaxSize() + "字节";

}

}

info += cardType;

tvUid.setText("NFC信息如下:\n" + info);

}

}

}

7、NFC解析数据类

package com.example.nfctest;

import android.content.Intent;

import android.nfc.NdefMessage;

import android.nfc.NdefRecord;

import android.nfc.NfcAdapter;

import android.os.Parcelable;

import java.util.ArrayList;

import java.util.List;

/**

* 类名 NfcMessageParser

* 作者 dy

* 功能

* 创建日期 2017/3/14 15:54

* 修改日期 2017/3/14 15:54

*/

public class NfcMessageParser {

private Intent tagIntent;

private String TAG = "NfcMessageParser";

public NfcMessageParser() {

}

public NfcMessageParser(Intent intent) {

this.tagIntent = intent;

}

// 解析NFC信息,

public List getTagMessage() {

if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(tagIntent.getAction())) {

NdefMessage[] msgs = getTagNdef(tagIntent);

List ndefList = getNdefString(msgs);

if (ndefList != null && ndefList.size() != 0) {

return ndefList;

}

}

return null;

}

// 得到Intent中的NDEF数据

private NdefMessage[] getTagNdef(Intent intent) {

// TODO Auto-generated method stub

NdefMessage[] msgs = null;

Parcelable[] rawMsgs = intent

.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES);

//把序列化数据转成Messaeg对象

if (rawMsgs != null) {

msgs = new NdefMessage[rawMsgs.length];

for (int i = 0; i < rawMsgs.length; i++) {

msgs[i] = (NdefMessage) rawMsgs[i];

}

} else {

// Unknown tag type

byte[] empty = new byte[]{};

NdefRecord record = new NdefRecord(NdefRecord.TNF_UNKNOWN, empty,

empty, empty);

NdefMessage msg = new NdefMessage(new NdefRecord[]{record});

msgs = new NdefMessage[]{msg};

}

return msgs;

}

// 把Message转成List

private List getNdefString(NdefMessage[] msgs) {

// TODO Auto-generated method stub

if (msgs != null && msgs.length != 0) {

List tagMessage = parser(msgs[0]);

return tagMessage;

}

return null;

}

// 把NDEF中的信息系转化为Record,并最终转化为String

private List parser(NdefMessage ndefMessage) {

// TODO Auto-generated method stub

NdefRecord[] records = ndefMessage.getRecords();

List elements = new ArrayList<>();

for (NdefRecord ndefRecord : records) {

if (!TextRecord.isText(ndefRecord)) {

return null;

}

elements.add(TextRecord.parse(ndefRecord));

}

return elements;

}

// 字符序列转换为16进制字符串

private String bytesToHexString(byte[] src) {

return bytesToHexString(src, true);

}

private String bytesToHexString(byte[] src, boolean isPrefix) {

StringBuilder stringBuilder = new StringBuilder();

if (isPrefix) {

stringBuilder.append("0x");

}

if (src == null || src.length <= 0) {

return null;

}

char[] buffer = new char[2];

for (int i = 0; i < src.length; i++) {

buffer[0] = Character.toUpperCase(Character.forDigit(

(src[i] >>> 4) & 0x0F, 16));

buffer[1] = Character.toUpperCase(Character.forDigit(src[i] & 0x0F,

16));

System.out.println(buffer);

stringBuilder.append(buffer);

}

return stringBuilder.toString();

}

}

8、在xml中添加nfc_tech_filter.xml文件 内容如下

、在AndroidManifest.xml中添加以下权限以及NFC TAG标签

android:resource="@xml/nfc_tech_filter" />

9、TextRecord中代码如下所示

package com.example.nfctest;

import android.nfc.NdefRecord;

import java.io.UnsupportedEncodingException;

import java.util.Arrays;

/**

* 类名 TextRecord

* 作者 dy

* 功能 标签管理

* 创建日期 2017/3/14 15:55

* 修改日期 2017/3/14 15:55

*/

public class TextRecord {

public static boolean isText(NdefRecord ndefRecord) {

// TODO Auto-generated method stub

try {

parse(ndefRecord);

return true;

} catch (IllegalArgumentException e) {

return false;

}

}

/**

* 吧Ndef记录转为String,这里的格式是TNF_WELL_KNOWN with RTD_TEXT

* @param ndefRecord

* @return

*/

public static String parse(NdefRecord ndefRecord) {

// TODO Auto-generated method stub

if (ndefRecord.getTnf() != NdefRecord.TNF_WELL_KNOWN

|| !Arrays.equals(ndefRecord.getType(), NdefRecord.RTD_TEXT)) {

throw new IllegalArgumentException("NFC类型不正确...");

}

try {

byte[] payload = ndefRecord.getPayload();

//把byte转为String

String textEncoding = ((payload[0] & 0200) == 0) ? "UTF-8"

: "UTF-16";

int languageCodeLength = payload[0] & 0077;

String languageCode = new String(payload, 1, languageCodeLength,

"US-ASCII");

String text = new String(payload, languageCodeLength + 1,

payload.length - languageCodeLength - 1, textEncoding);

return text;

} catch (UnsupportedEncodingException e) {

// should never happen unless we get a malformed tag.

throw new IllegalArgumentException(e);

}

}

}

10、然后在手机界面打开NFC开关,不知道怎么打开或者手机设置搜索NFC哈,说明NFC功能正常了。

10、总结:

1、不管是上面二种那种方式 第一种如果使用这种.c执行shell脚本记得将系统nfc功能开关关闭 否则会冲突不生效哈

2、第二种系统原生nfc 需要注意Android10以后PendingIntent类型发生变化,否则会报错提示。Android12以上PendingIntent需要强制增加FLAG_IMMUTABLE或FLAG_MUTABLE

3、调试的时候可以多拿几种不同NFC卡进行调试。

到这里所有NFC调试流程基本结束了,转载请注明出处高通 Android 12 调试产测NFC功能_KdanMin的博客-CSDN博客。谢谢!