本示例使用的发卡器:https://item.taobao.com/item.htm?spm=a1z10.5-c-s.w4002-21818769070.11.6c46789elLwMzv&id=615391857885?
package com.usbreadertest;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.CheckBox;
import android.widget.EditText;
import android.widget.Spinner;
import android.widget.TextView;
import com.reader.ourmifare;
public class CpuCardActivity extends AppCompatActivity {
private TextView tv;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_cpu_card);
androidx.appcompat.widget.Toolbar toolbar=findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
tv = findViewById(R.id.sample_text);
tv.setText("操作结果");
Spinner spls = findViewById(R.id.spin_SelRWauthkey);
spls.setSelection(1);
}
@Override
public void onBackPressed(){
super.onBackPressed();
finish();
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
if(item.getItemId()==android.R.id.home){
finish();
return true;
}
return super.onOptionsItemSelected(item);
}
public void retmain(View view)
{
finish();
}
public void cpu_request(View view){
byte status; //存放返回值
byte[] mypiccserial = new byte[4]; //4字节设备编号
byte[] myparam = new byte[4]; //4字节卡参数
byte[] myver= new byte[1];
byte[] mycode= new byte[1];
TextView ctrcarduid;
ctrcarduid = findViewById(R.id.textcarduid);
ctrcarduid.setText("");
tv = findViewById(R.id.sample_text);
tv.setText("");
status =ourmifare.cpurequest(mypiccserial,myparam, myver, mycode);
if (status == 0 || status == 52) {
ourmifare.pcdbeep(38);
String serialnumber = "";
for (int i = 0; i < 4; i++) {
String bytestr = "00" + Integer.toHexString(mypiccserial[i] & 0xff);
serialnumber = serialnumber + bytestr.substring(bytestr.length() - 2, bytestr.length());
}
ctrcarduid.setText(serialnumber);
tv.setText("CPU卡已激活,卡号:"+serialnumber);
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_rsinit(View view){
byte status; //存放返回值
byte[] mypicckey = new byte[16]; //卡片认证密码,16进制,最好不要随便修改,此密码不影响数据安全
byte mykeylen = 16 ; //卡片密码长度
int spacesize; //应用空间字节大小(要大于所有文件总和),根据不同容量的卡设不同值
byte emptycard; //初始化前是否清空卡内数据,取值1将先清空卡,取值为0不清空现有数据
String cardkeystr;
CheckBox ctrcheck;
EditText ctrcardkey;
EditText ctrspace;
ctrcheck=findViewById(R.id.checkcleckcard) ;
if(ctrcheck.isChecked()) {
emptycard=1;
}else {emptycard=0;}
ctrspace=findViewById(R.id.textusespace) ;
spacesize=Integer.parseInt(ctrspace.getText().toString().trim());
ctrcardkey=findViewById(R.id.cardkey) ;
cardkeystr=ctrcardkey.getText().toString().trim();
if (cardkeystr.length() < 32)
{
tv.setText("卡密钥是32位16进制数据,请输入正确的卡片密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
mypicckey[i] = (byte) Integer.parseInt(cardkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
status =ourmifare.cpursinit(emptycard, mypicckey, mykeylen, spacesize);
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText("初始化卡成功!" );
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_rseasyfileadd(View view){
byte status; //存放返回值
byte RKLen = 16 ; //文件 只读权限密码 长度
byte[] RFilekey = new byte[RKLen]; //文件 只读权限密码
byte WKLen = 16 ; //文件 读写权限密码 长度
byte[] WFilekey = new byte[WKLen]; //文件 读写权限密码,此权限高,可认证此密码修改只读密码,要记住文件密码
Spinner spls = findViewById(R.id.spin_SelCreatefilenum);
byte FileIndex = (byte)(spls.getSelectedItemId()); //指定文件号
EditText ctrfilesize = findViewById(R.id.textfileszie); //文件大小
int FileSize=Integer.parseInt(ctrfilesize.getText().toString().trim());
EditText ctrreadkey=findViewById(R.id.readkey) ;
String readkeystr=ctrreadkey.getText().toString().trim();
if (readkeystr.length() < 32)
{
tv.setText("文件只读密钥是32位16进制数据,请输入正确的只读密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
RFilekey[i] = (byte) Integer.parseInt(readkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
EditText ctrwritekey=findViewById(R.id.writekey) ;
String writekeystr=ctrwritekey.getText().toString().trim();
if (writekeystr.length() < 32)
{
tv.setText("文件读写密钥是32位16进制数据,请输入正确的读写密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
WFilekey[i] = (byte) Integer.parseInt(writekeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
status =ourmifare.cpursfileadds(FileIndex, RFilekey, RKLen, WFilekey, WKLen, FileSize) ; //
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText(Integer.toHexString(FileIndex+1)+" 号文件创建成功!");
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_rsfileadd(View view){
byte status; //存放返回值
byte RKLen = 16 ; //文件 只读权限密码 长度
byte[] RFilekey = new byte[RKLen]; //文件 只读权限密码
byte WKLen = 16 ; //文件 读写权限密码 长度
byte[] WFilekey = new byte[WKLen]; //文件 读写权限密码,此权限高,可认证此密码修改只读密码,要记住文件密码
Spinner spls = findViewById(R.id.spin_SelCreatefilenum);
byte FileIndex = (byte)(spls.getSelectedItemId()); //指定文件号
EditText ctrfilesize = findViewById(R.id.textfileszie); //文件大小
int FileSize=Integer.parseInt(ctrfilesize.getText().toString().trim());
EditText ctrreadkey=findViewById(R.id.readkey) ;
String readkeystr=ctrreadkey.getText().toString().trim();
if (readkeystr.length() < 32)
{
tv.setText("文件只读密钥是32位16进制数据,请输入正确的只读密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
RFilekey[i] = (byte) Integer.parseInt(readkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
EditText ctrwritekey=findViewById(R.id.writekey) ;
String writekeystr=ctrwritekey.getText().toString().trim();
if (writekeystr.length() < 32)
{
tv.setText("文件读写密钥是32位16进制数据,请输入正确的读写密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
WFilekey[i] = (byte) Integer.parseInt(writekeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
status =ourmifare.cpursfileadd(FileIndex, RFilekey, RKLen, WFilekey, WKLen, FileSize) ; //
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText(Integer.toHexString(FileIndex+1)+" 号文件创建成功!");
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_rsfilekeychg(View view){
byte status; //存放返回值
byte OldKey;
byte AKLen = 16 ; //文件 认证密码 长度
byte[] Autkey = new byte[AKLen]; //文件 认证密码
byte NKLen = 16 ; //文件 新密码密码 长度
byte[] Newkey = new byte[NKLen]; //文件 新密码密码
Spinner spls = findViewById(R.id.spin_Selchangkeyfile);
byte FileIndex = (byte)(spls.getSelectedItemId()); //指定文件号
Spinner spcharw = findViewById(R.id.spin_SelchangRWkey);
byte ChaKey= (byte)(spcharw.getSelectedItemId()); //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
Spinner spsuthkey = findViewById(R.id.spin_Selauthkey);
if (spsuthkey.getSelectedItemId()==0) {
OldKey = 0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
}else {OldKey=2;}
byte ctrlCode=(byte)(ChaKey+OldKey);
EditText ctrauthkey=findViewById(R.id.authkey0) ;
String authkeystr=ctrauthkey.getText().toString().trim();
if (authkeystr.length() < 32)
{
tv.setText("文件认证密钥是32位16进制数据,请输入正确的认证密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
Autkey[i] = (byte) Integer.parseInt(authkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
EditText ctrnewkey=findViewById(R.id.newkey) ;
String newkeystr=ctrnewkey.getText().toString().trim();
if (newkeystr.length() < 32)
{
tv.setText("文件新密钥是32位16进制数据,请输入正确的新密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
Newkey[i] = (byte) Integer.parseInt(newkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
status =ourmifare.cpursfilekeychg(FileIndex, ctrlCode, Autkey, AKLen, Newkey, NKLen) ;//& 0xff用于转为无符号行数据
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText(Integer.toHexString(FileIndex+1)+" 号文件密码修改成功!" );
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_readfile(View view){
byte status; //存放返回值
byte AKLen = 16 ; //文件 认证密码 长度
byte[] Autkey = new byte[AKLen]; //文件 认证密码
byte DataStart=0; //读起始位置
byte DataLen = 0 ; //读长度最大不能超过255,如文件长度>255 要循环读取
byte[] DataBuf = new byte[300]; //读文件数据缓冲
byte ctrlCode=0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
EditText ctrrwdata = findViewById(R.id.editrwdata);
ctrrwdata.setText("");
Spinner spls = findViewById(R.id.spin_SelRWfile);
byte FileIndex = (byte)(spls.getSelectedItemId()); //指定文件号
Spinner spsuthkey = findViewById(R.id.spin_SelRWauthkey);
if (spsuthkey.getSelectedItemId()==0) {
ctrlCode = 0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
}else {ctrlCode=2;}
EditText ctrauthkey=findViewById(R.id.editrwauthkey) ;
String authkeystr=ctrauthkey.getText().toString().trim();
if (authkeystr.length() < 32)
{
tv.setText("文件认证密钥是32位16进制数据,请输入正确的认证密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
Autkey[i] = (byte) Integer.parseInt(authkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
EditText ctrbeginadd = findViewById(R.id.textrwbeginadd); //操作起始地址
DataStart=(byte)Integer.parseInt(ctrbeginadd.getText().toString().trim());
EditText ctrrwlen = findViewById(R.id.textrwlen); //读写长度
DataLen=(byte)Integer.parseInt(ctrrwlen.getText().toString().trim());
if (DataLen<1 || DataLen>255){
tv.setText("一次读取的数据长度应大于0,小于256!如要读取更多的数据请使用循环的方式读取。");
return;
}
status = ourmifare.cpursfiledataread(FileIndex, ctrlCode, Autkey, AKLen, DataStart, DataBuf,DataLen) ;//& 0xff用于转为无符号行数据
if (status == 0 ) {
ourmifare.pcdbeep(38);
String filedata="";
for (int i = 0; i < DataLen; i++) {
String bytestr = "00" + Integer.toHexString(DataBuf[i] & 0xff);
filedata = filedata + bytestr.substring(bytestr.length() - 2, bytestr.length()) +" ";
}
ctrrwdata.setText(filedata);
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_writeeasyfile(View view){
byte status; //存放返回值
byte AKLen = 16 ; //文件 认证密码 长度
byte[] Autkey = new byte[AKLen]; //文件 认证密码
byte DataStart=0; //写起始位置
byte DataLen ; //写长度最大不能超过247,如文件长度>247 要循环写入
byte[] DataBuf = new byte[300]; //写文件数据缓冲
byte ctrlCode=0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
Spinner spls = findViewById(R.id.spin_SelRWfile);
byte FileIndex = (byte)(spls.getSelectedItemId()); //指定文件号
Spinner spsuthkey = findViewById(R.id.spin_SelRWauthkey);
if (spsuthkey.getSelectedItemId()==0) {
ctrlCode = 0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
}else {ctrlCode=2;}
EditText ctrauthkey=findViewById(R.id.editrwauthkey) ;
String authkeystr=ctrauthkey.getText().toString().trim();
if (authkeystr.length() < 32)
{
tv.setText("文件认证密钥是32位16进制数据,请输入正确的认证密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
Autkey[i] = (byte) Integer.parseInt(authkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
EditText ctrbeginadd = findViewById(R.id.textrwbeginadd); //操作起始地址
DataStart=(byte)Integer.parseInt(ctrbeginadd.getText().toString().trim());
EditText ctrrwlen = findViewById(R.id.textrwlen); //读写长度
DataLen=(byte)Integer.parseInt(ctrrwlen.getText().toString().trim());
if (DataLen<1 || DataLen>247){
tv.setText("一次写入的数据长度应大于0,小于248,如要写入更多的数据可使用循环的方式写入。");
return;
}
EditText ctrrwdata = findViewById(R.id.editrwdata); //读写数据
String rwdatahex=ctrrwdata.getText().toString().trim();
String[] strArr = rwdatahex.split("\\ "); /*分割接收到的数据后再分析、处理、返回指令 */
if (strArr.length<DataLen){
tv.setText("写入数据不足,请输入"+Integer.toString(DataLen*2)+"位16进制写入数据!");
return;
}else{
for(int p=0;p<strArr.length;p++) {
DataBuf[p]=(byte)(Integer.parseInt(strArr[p],16));
}
}
status =ourmifare.cpursfiledatawrites(FileIndex, ctrlCode, Autkey, AKLen, DataStart, DataBuf,DataLen) ;//& 0xff用于转为无符号行数据
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText("写数据成功!" );
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_writefile(View view){
byte status; //存放返回值
byte AKLen = 16 ; //文件 认证密码 长度
byte[] Autkey = new byte[AKLen]; //文件 认证密码
byte DataStart=0; //写起始位置
byte DataLen ; //写长度最大不能超过247,如文件长度>247 要循环写入
byte[] DataBuf = new byte[300]; //写文件数据缓冲
byte ctrlCode=0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
Spinner spls = findViewById(R.id.spin_SelRWfile);
byte FileIndex = (byte)(spls.getSelectedItemId()); //指定文件号
Spinner spsuthkey = findViewById(R.id.spin_SelRWauthkey);
if (spsuthkey.getSelectedItemId()==0) {
ctrlCode = 0; //取值 0 表示用只读密码来认证,取值 2 表示用读写密码来认证
}else {ctrlCode=2;}
EditText ctrauthkey=findViewById(R.id.editrwauthkey) ;
String authkeystr=ctrauthkey.getText().toString().trim();
if (authkeystr.length() < 32)
{
tv.setText("文件认证密钥是32位16进制数据,请输入正确的认证密钥!");
return;
}else {
for (int i = 0; i < 16; i++) {
Autkey[i] = (byte) Integer.parseInt(authkeystr.substring(i * 2, i * 2 + 2), 16);//只有用Integer.parseInt才能杜绝大于128时的错误
}
}
EditText ctrbeginadd = findViewById(R.id.textrwbeginadd); //操作起始地址
DataStart=(byte)Integer.parseInt(ctrbeginadd.getText().toString().trim());
EditText ctrrwlen = findViewById(R.id.textrwlen); //读写长度
DataLen=(byte)Integer.parseInt(ctrrwlen.getText().toString().trim());
if (DataLen<1 || DataLen>247){
tv.setText("一次写入的数据长度应大于0,小于248,如要写入更多的数据可使用循环的方式写入。");
return;
}
EditText ctrrwdata = findViewById(R.id.editrwdata); //读写数据
String rwdatahex=ctrrwdata.getText().toString().trim();
String[] strArr = rwdatahex.split("\\ "); /*分割接收到的数据后再分析、处理、返回指令 */
if (strArr.length<DataLen){
tv.setText("写入数据不足,请输入"+Integer.toString(DataLen*2)+"位16进制写入数据!");
return;
}else{
for(int p=0;p<strArr.length;p++) {
DataBuf[p]=(byte)(Integer.parseInt(strArr[p],16));
}
}
status =ourmifare.cpursfiledatawrite(FileIndex, ctrlCode, Autkey, AKLen, DataStart, DataBuf,DataLen) ;//& 0xff用于转为无符号行数据
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText("写数据成功!" );
} else {
PrintErrInf(status); //返回代码提示
}
}
public void cpu_cardclr(View view){
byte status=ourmifare.cpurscardclr() ;//& 0xff用于转为无符号行数据
if (status == 0 ) {
ourmifare.pcdbeep(38);
tv.setText("强制清空卡成功!");
} else if(status == 64){
tv.setText("此卡为未经过我们cpursinit函数成功初始化的卡!");
}else{
PrintErrInf(status); //返回代码提示
}
}
public void PrintErrInf(byte errcode) {
String dispstr;
switch(errcode){
case 8:
dispstr="错误代码:8,未寻到卡,请重新拿开卡后再放到感应区!";
break;
case 21:
dispstr="错误代码:21,没有动态库!";
break;
case 22:
dispstr="错误代码:22,动态库或驱动程序异常!";
break;
case 23:
dispstr="错误代码:23,驱动程序错误或尚未安装!";
break;
case 24:
dispstr="错误代码:24,操作超时,一般是动态库没有反映!";
break;
case 25:
dispstr="错误代码:25,发送字数不够!";
break;
case 26:
dispstr="错误代码:26,发送的CRC错!";
break;
case 27:
dispstr="错误代码:27,接收的字数不够!";
break;
case 28:
dispstr="错误代码:28,接收的CRC错!";
break;
case 50:
dispstr="错误代码:50,RATS错误,厂家调试代码,用户不需理会!";
break;
case 51:
dispstr="错误代码:51,PPS错误,厂家调试代码,用户不需理会!";
break;
case 52:
dispstr="错误代码:52,已进入了14443-4协议状态,可进行CPU卡功能所有操作了!";
break;
case 53:
dispstr="错误代码:53,CPU卡功能通讯错误,请先激活卡片!";
break;
case 54:
dispstr="错误代码:54,数据不足,需要接着发送未完成的数据至卡上!";
break;
case 55:
dispstr="错误代码:55,发送ACK指令给卡,让卡接着发送数据回来!";
break;
case 56:
dispstr="错误代码:56,清空根目录失败!";
break;
case 57:
dispstr="错误代码:57,卡片不支持功能!";
break;
case 58:
dispstr="错误代码:58,卡片初始化失败!";
break;
case 59:
dispstr="错误代码:59,分配的空间不足!";
break;
case 60:
dispstr="错误代码:60,本次操作的实体已存在!";
break;
case 61:
dispstr="错误代码:61,无足够空间!";
break;
case 62:
dispstr="错误代码:62,文件不存在!";
break;
case 63:
dispstr="错误代码:63,权限不足,有可能是用只读密码认证,导致无法更改读写密码或无法写文件!";
break;
case 64:
dispstr="错误代码:64,密码不存在,或密钥文件未创建!";
break;
case 65:
dispstr="错误代码:65,传送长度错误!";
break;
case 66:
dispstr="错误代码:66,Le错误,即接收的数据长度指定过大!";
break;
case 67:
dispstr="错误代码:67,功能不支持或卡中无MF 或卡片已锁定!";
break;
case 68:
dispstr="错误代码:68,密码认证错误次数过多,该密码已被锁死!";
break;
case 70:
case 71:
case 72:
case 73:
case 74:
case 75:
case 76:
case 77:
case 78:
case 79:
case 80:
case 81:
case 82:
case 83:
case 84:
case 85:
dispstr="错误代码:"+Integer.toString(errcode)+",密码错误,剩余次数为"+Integer.toString(errcode-70)+",如果为0,该密码将锁死,无法再认证!";
break;
case 86:
dispstr="错误代码:"+Integer.toString(errcode)+",更改后的密码长度必须和创建时的长度一致!";
break;
case 87:
dispstr="错误代码:"+Integer.toString(errcode)+",应用目录不存在!";
break;
case 88:
dispstr="错误代码:"+Integer.toString(errcode)+",应用文件不存在!";
break;
case 89:
dispstr="错误代码:"+Integer.toString(errcode)+",文件号不能超过 5 ";
break;
case 90:
dispstr="错误代码:"+Integer.toString(errcode)+",读取文件时返回的长度不足,数据可能不正确!";
break;
case 91:
dispstr="错误代码:"+Integer.toString(errcode)+",一次读文件的长度不能超过 255";
break;
case 92:
dispstr="错误代码:"+Integer.toString(errcode)+",一次写文件的长度不能超过 247";
break;
default:
dispstr="未知错误,错误代码:"+Integer.toString(errcode);
break;
}
tv = findViewById(R.id.sample_text);
tv.setText(dispstr);
}
}