久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费
標題:
FM1702SL 51單片機測試程序
[打印本頁]
作者:
c煌煌煌
時間:
2018-12-8 16:57
標題:
FM1702SL 51單片機測試程序
FM1702SL源碼:
/****************************************************************************
* File : main.c *
* COPYRIGHT BY HUOYAN LTD.COMPANY *
* Version: V1.3 *
* Author: NIL *
* *
* Compiler: KEIL C51 V7.10 *
* *
* Description: AT89S52-Firmware for FM1702 Demo Serial Reader *
* *
****************************************************************************/
#include <reg52.h>
#define __SRC
#include "main.h"
#undef __SRC
#include <intrins.h>
//#include <string.h>
//#include <stdio.h>
//#include <absacc.h>
#include "FM1702.h"
#define uchar unsigned char
#define uint unsigned int
//pin define mcu控制管腳定義
sbit RF_MISO = P1^0;
sbit RF_MOSI = P1^1;
sbit RF_SCK = P1^2;
sbit RF_NSS = P1^3;
sbit RF_RST = P1^4; //由高變低時啟動內部復位程序
sbit CARD_LED = P2^0;
sbit SPEAKER = P2^0;
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~//
//名稱: spi_byte_transceive //
//功能: 該函數實現SPI通訊的數據收發 //
// //
//輸入: //
// 發送數據 //
// //
//輸出: //
// 接收數據 //
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~//
char spi_byte_transceive(char sendbyte)
{
char i,temp;
for(i=0;i<8;i++)
{
RF_SCK=0;
if(sendbyte & 0x80) //位運算,判斷最高位是否為1
{
RF_MOSI=1;
}
else
{
RF_MOSI=0;
}
sendbyte <<= 1;
RF_SCK=1;
temp <<= 1;
if(RF_MISO)
temp |= 0x01;
}
RF_SCK=0;
_nop_();
_nop_();
RF_MOSI=0;
return (temp);
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~//
//名稱: rc531_register_write //
//功能: 該函數實現通過SPI接口對RC531中一個寄存器寫入值 //
// //
//輸入: //
// RC531目標寄存器地址和寫入值 //
// //
//輸出: //
// N/A //
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~//
void SPIWrite(char reg_ad,char reg_data)
{
RF_SCK=0;
reg_ad <<= 1;
RF_NSS=0;
reg_ad &= 0x7F;
spi_byte_transceive(reg_ad);
spi_byte_transceive(reg_data);
RF_NSS=1;
return;
}
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~//
//名稱: rc531_register_read //
//功能: 該函數實現通過SPI接口讀取RC531中一個寄存器的值 //
// //
//輸入: //
// RC531目標寄存器地址 //
// //
//輸出: //
// 目標寄存器的值 //
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~//
unsigned char SPIRead(char reg_ad)
{ char temp;
RF_SCK=0;
_nop_();
_nop_();
RF_NSS=0;
reg_ad <<= 1;
reg_ad |= 0x80;
spi_byte_transceive(reg_ad);
temp=spi_byte_transceive(0x00);
RF_NSS=1;
return (temp);
}
///////////////////////////////////////////////////////////////////////
// 主函數
///////////////////////////////////////////////////////////////////////
void main(void)
{ //設置變量
uchar baud;
InitSystem(); //初始化系統
while (1)
{
//檢查命令標志
if (CmdValid) //if LEVEL 1
{
CmdValid = FALSE;
if(RevBuffer[0]==11) //if LEVEL 2
{
RevBuffer[2]=RevBuffer[1];
RevBuffer[0]=1;
RevBuffer[1]=0;
CALL_isr_UART(); //equal to 'SETB TI', defined in main.h
SPEAKER=0; //開蜂鳴器和指示燈
delay_10ms(RevBuffer[2]);
SPEAKER=1;
}
else if(RevBuffer[0]==13) //設置通訊波特率 //if LEVEL 2
{
switch(RevBuffer[1])
{
case 0:
baud=BAUD_9600;
break;
case 1:
baud=BAUD_14400;
break;
case 2:
baud=BAUD_19200;
break;
case 3:
baud=BAUD_28800;
break;
case 4:
baud=BAUD_38400;
break;
case 5:
baud=BAUD_57600;
break;
case 6:
baud=BAUD_115200;
break;
default:
baud=BAUD_19200;
break;
} //switch body
RevBuffer[0]=1; //contact
RevBuffer[1]=0;
CALL_isr_UART();
delay_10ms(5);
TR1 = 0;
TH1 = baud;
TL1 = TH1;
delay_10ms(2);
TR1 = TRUE;
}//if LEVEL 2
else
{
cmd_process(); // 進入IC卡處理程序
CALL_isr_UART();
}
}
}//while循環體
}//main函數
///////////////////////////////////////////////////////////////////////
// 系統初始化
///////////////////////////////////////////////////////////////////////
void InitSystem(void)
{
RF_NSS=1;
RF_RST=0;
ET2 = 0; //Timer 2 disabled
T2CON = 0x04; //start Timer 2(internal timer, auto reload)
PCON = 0x80; //baud rate double
SCON = 0x70; //UART mode 1, enable receive, if No valid stop bit, RI not activated.
//TMOD = 0x22;
TMOD = 0x21; //Timer 1 8bit auto reload TR1 control
//Timer 0 16bit TR0 control
TH1 = BAUD_19200; //默認波特率
TL1 = TH1;
TR1 = TRUE; // 波特率發生器
TH0 = 0x60;
TL0 = 0x60;
TR0 = 0; //Timer 0 doesn't run
ET0=0;
ET1=0;
EA=1;
EX0=1;
IT0 = 1;
TR2=0;
ES = TRUE; //enable UART interrupt
CmdValid=0; //flag initiation
//喇叭和指示燈測試
SPEAKER=0;
delay_10ms(10);
delay_10ms(10);
SPEAKER=1;
Init_FM1702(0);
}
///////////////////////////////////////////////////////////////////////
// 串口接收和發送中斷
///////////////////////////////////////////////////////////////////////
void isr_UART(void) interrupt 4 using 1
{
uchar len, i;
unsigned int j=0;
if(RI)
{
len=SBUF;
RI=0;
for(i=0;i<len;i++)
{
while(!RI)
{
j++;
if(j>1000)
{
break;
}
}
if(j<1000)
{
RevBuffer[i]=SBUF;
RI=0;
j=0;
}
else
{
break;
}
}
if(i==len)
{
REN=0;
CmdValid=1;
}
}
else if(!RI && TI)
{
TI=0;
len=RevBuffer[0];
for(i=0;i<len+1;i++)
{
SBUF=RevBuffer[i];
while(!TI);
TI=0;
}
REN=1;
}
}
///////////////////////////////////////////////////////////////////////
// IC卡處理函數
///////////////////////////////////////////////////////////////////////
void cmd_process(void)
{
uchar cmd;
uchar status;
cmd = RevBuffer[0];
switch(cmd)
{
case 1: // Halt the card //終止卡的操作
status=MIF_Halt();
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
case 2:
status = Request(RF_CMD_REQUEST_ALL); //RF_CMD_REQUEST_STD=0x26, request Idle
if(status != FM1702_OK)
{
status = Request(RF_CMD_REQUEST_ALL);
if(status != FM1702_OK)
{
RevBuffer[0] = 1;
RevBuffer[1] = FM1702_REQERR;
break;
}
}
if(tagtype[0]==2)
cardtype=mifarepro; // Mifare Pro 卡
else if(tagtype[0]==4)
cardtype=mifare1; // Mifare One 卡
else if(tagtype[0]==16)
cardtype=mifarelight; // Mifare Light 卡
else
cardtype=unknowncard;
RevBuffer[0]=3;
RevBuffer[1]=status;
RevBuffer[2]=tagtype[0];
RevBuffer[3]=tagtype[1];
break;
case 3: // 防沖突 讀卡的系列號 MLastSelectedSnr
status = AntiColl();;
if(status!=FM1702_OK)
{
RevBuffer[0]=1;
RevBuffer[1]=FM1702_ANTICOLLERR;
break;
}
//memcpy(MLastSelectedSnr,&RevBuffer[2],4);
RevBuffer[0]=5;
RevBuffer[1]=status;
RevBuffer[2]=UID[0];
RevBuffer[3]=UID[1];
RevBuffer[4]=UID[2];
RevBuffer[5]=UID[3];
break;
case 4: // 選擇卡 Select Card
status=Select_Card();
if(status!=FM1702_OK)
{
RevBuffer[0]=1;
RevBuffer[1]=FM1702_SELERR;
break;
}
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
case 5: //下載密鑰
status = Load_keyE2(RevBuffer[2],RevBuffer[1]); //%40
status = Authentication(UID, RevBuffer[2], RevBuffer[1]);
if(status != FM1702_OK)
{
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
}
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
case 6: // Key loading into the MF RC500's EEPROM
// 校驗卡密碼(E2) Load_keyE2_CPY(uchar Secnr, uchar Mode)
status = Load_keyE2_CPY(RevBuffer[2],RevBuffer[1]); //%40
if(status != FM1702_OK)
{
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
}
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
case 8: //讀卡
status=MIF_READ(&RevBuffer[2],RevBuffer[1]);
if(status != FM1702_OK)
{
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
}
else
{
if(cardtype==mifare1||cardtype==mifarepro)
RevBuffer[0]=17;
else if(cardtype==1)
RevBuffer[0]=9;
else
RevBuffer[0]=16;
}
RevBuffer[1]=status;
break;
case 9: //寫卡
status=MIF_Write(&RevBuffer[2],RevBuffer[1]);
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
case 10: //加值減值
if(RevBuffer[1] == RF_CMD_INC)
{
status = MIF_Increment(&RevBuffer[3],RevBuffer[2]);
MIF_Transfer(RevBuffer[2]);
}
else if(RevBuffer[1] == RF_CMD_DEC)
{
status = MIF_Decrement(&RevBuffer[3],RevBuffer[2]);
MIF_Transfer(RevBuffer[2]);
}
else
{
status = 1;
}
RevBuffer[0]=1;
RevBuffer[1]=status;
break;
case 12:
RevBuffer[0]=1;
RevBuffer[1]=0;
break;
default:
RevBuffer[0] = 1;
RevBuffer[1] = 1;
break;
}
}
/****************************************************************/
/*名稱: Clear_FIFO */
/*功能: 該函數實現清空FM1702中FIFO的數據*/
/*輸入: N/A */
/*輸出: TRUE, FIFO被清空*/
/* FALSE, FIFO未被清空*/
/****************************************************************/
uchar Clear_FIFO(void)
{
uchar temp;
uint i;
temp = SPIRead(Control); /* 清空FIFO */
temp = (temp | 0x01);
SPIWrite(Control,temp);
for(i = 0; i < RF_TimeOut; i++) /* 檢查FIFO是否被清空 */
{
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return TRUE;
}
}
return FALSE;
}
/****************************************************************/
/*名稱: Write_FIFO */
/*功能: 該函數實現向FM1702的FIFO中寫入x bytes數據*/
/*輸入: count, 待寫入字節的長度*/
/* buff, 指向待寫入數據的指針*/
/*輸出: N/A */
/****************************************************************/
void Write_FIFO(uchar count, uchar idata *buff)
{
uchar i;
for(i = 0; i < count; i++)
{
SPIWrite(FIFO,*(buff + i));
}
}
/****************************************************************/
/*名稱: Read_FIFO */
/*功能: 該函數實現從FM1702的FIFO中讀出x bytes數據*/
/*輸入: buff, 指向讀出數據的指針*/
/*輸出: N/A */
/****************************************************************/
uchar Read_FIFO(uchar idata *buff)
{
uchar temp;
uchar i;
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return 0;
}
if(temp >= 24)
{
temp = 24;
}
for(i = 0; i < temp; i++)
{
*(buff + i) = SPIRead(FIFO);
}
return temp;
}
/****************************************************************/
/*名稱: Judge_Req */
/*功能: 該函數實現對卡片復位應答信號的判斷*/
/*輸入: *buff, 指向應答數據的指針*/
/*輸出: TRUE, 卡片應答信號正確*/
/* FALSE, 卡片應答信號錯誤*/
/****************************************************************/
uchar Judge_Req(uchar idata *buff)
{
uchar temp1, temp2;
temp1 = *buff;
temp2 = *(buff + 1);
if((temp1 == 0x02) || (temp1 == 0x04) || (temp1 == 0x05) || (temp1 == 0x53) || (temp1 == 0x03))
{
if (temp2 == 0x00)
{
return TRUE;
}
}
return FALSE;
}
/****************************************************************/
/*名稱: Check_UID */
/*功能: 該函數實現對收到的卡片的序列號的判斷*/
/*輸入: N/A */
/*輸出: TRUE: 序列號正確*/
/* FALSE: 序列號錯誤*/
/****************************************************************/
uchar Check_UID(void)
{
uchar temp;
uchar i;
temp = 0x00;
for(i = 0; i < 5; i++)
{
temp = temp ^ UID[i];
}
if(temp == 0)
{
return TRUE;
}
return FALSE;
}
/****************************************************************/
/*名稱: Save_UID */
/*功能: 該函數實現保存卡片收到的序列號*/
/*輸入: row: 產生沖突的行*/
/* col: 產生沖突的列*/
/* length: 接収到的UID數據長度*/
/*輸出: N/A */
/****************************************************************/
void Save_UID(uchar row, uchar col, uchar length)
{
uchar i;
uchar temp;
uchar temp1;
if((row == 0x00) && (col == 0x00))
{
for(i = 0; i < length; i++)
{
UID[i] = RevBuffer[i];
}
}
else
{
temp = RevBuffer[0];
temp1 = UID[row - 1];
switch(col)
{
case 0: temp1 = 0x00; row = row + 1; break;
case 1: temp = temp & 0xFE; temp1 = temp1 & 0x01; break;
case 2: temp = temp & 0xFC; temp1 = temp1 & 0x03; break;
case 3: temp = temp & 0xF8; temp1 = temp1 & 0x07; break;
case 4: temp = temp & 0xF0; temp1 = temp1 & 0x0F; break;
case 5: temp = temp & 0xE0; temp1 = temp1 & 0x1F; break;
case 6: temp = temp & 0xC0; temp1 = temp1 & 0x3F; break;
case 7: temp = temp & 0x80; temp1 = temp1 & 0x7F; break;
default: break;
}
RevBuffer[0] = temp;
UID[row - 1] = temp1 | temp;
for(i = 1; i < length; i++)
{
UID[row - 1 + i] = RevBuffer[i];
}
}
}
/****************************************************************/
/*名稱: Set_BitFraming */
/*功能: 該函數設置待發送數據的字節數*/
/*輸入: row: 產生沖突的行*/
/* col: 產生沖突的列*/
/*輸出: N/A */
/****************************************************************/
void Set_BitFraming(uchar row, uchar col)
{
switch(row)
{
case 0: RevBuffer[1] = 0x20; break;
case 1: RevBuffer[1] = 0x30; break;
case 2: RevBuffer[1] = 0x40; break;
case 3: RevBuffer[1] = 0x50; break;
case 4: RevBuffer[1] = 0x60; break;
default: break;
}
switch(col)
{
case 0: SPIWrite(Bit_Frame,0x00); break;
case 1: SPIWrite(Bit_Frame,0x11); RevBuffer[1] = (RevBuffer[1] | 0x01); break;
case 2: SPIWrite(Bit_Frame,0x22); RevBuffer[1] = (RevBuffer[1] | 0x02); break;
case 3: SPIWrite(Bit_Frame,0x33); RevBuffer[1] = (RevBuffer[1] | 0x03); break;
case 4: SPIWrite(Bit_Frame,0x44); RevBuffer[1] = (RevBuffer[1] | 0x04); break;
case 5: SPIWrite(Bit_Frame,0x55); RevBuffer[1] = (RevBuffer[1] | 0x05); break;
case 6: SPIWrite(Bit_Frame,0x66); RevBuffer[1] = (RevBuffer[1] | 0x06); break;
case 7: SPIWrite(Bit_Frame,0x77); RevBuffer[1] = (RevBuffer[1] | 0x07); break;
default: break;
}
}
/****************************************************************/
/*名稱: Init_FM1702 */
/*功能: 該函數實現對FM1702初始化操作*/
/*輸入: mode:工作模式, 0:TYPEA模式*/
/* 1:TYPEB模式*/
/* 2:上海模式*/
/*輸出: N/A */
/****************************************************************/
void Init_FM1702(uchar mode)
{
uchar temp;
uint i;
// unsigned int idata timecnt=0;
RF_SCK = 1;
RF_MISO = 1;
RF_MOSI = 1;
RF_RST = 1; /* FM1702復位 */
for(i = 0; i < 0x1fff; i++)
{
_nop_(); /*等待約140ms, 11.0592*/
}
RF_RST = 0;
for(i = 0; i < 0x1fff; i++)
{
_nop_();
}
////////////////////////////////////////////////////////////////////////////
while(temp = SPIRead(Command) != 0) /* 等待Command = 0,FM1702復位成功 */
{
_nop_();
}
////////////////////////////////////////////////////////////////////////////
// delay_10ms(1);
SPIWrite(Page_Sel,0x80);
for(i = 0; i < 0x1fff; i++) /* 延時 */
{
if(temp=SPIRead(Command) == 0x00)
{
SPIWrite(Page_Sel,0x00);
}
}
// P2 = SPIRead(TimerClock); //for debug
SPIWrite(TimerClock,0x0b); //address 2AH /* 定時器周期設置寄存器 */
SPIWrite(TimerControl,0x02); //address 2BH /* 定時器控制寄存器 */
SPIWrite(TimerReload,0x42); //address 2CH /* 定時器初值寄存器 */
SPIWrite(InterruptEn,0x7f); //address 06H /* 中斷使能/禁止寄存器 */
// temp = SPIRead(InterruptEn);
SPIWrite(Int_Req,0x7f); //address 07H /* 中斷請求標識寄存器 */
SPIWrite(MFOUTSelect,0x02); //address 26H /* mf OUT 選擇配置寄存器 */
//設置調制器的輸入源為內部編碼器, 并且設置TX1和TX2
SPIWrite(TxControl,0x5b); //address 11H /* 發送控制寄存器 */
SPIWrite(RxControl2,0x01);
// SPIWrite(RxControl1,0x73);
SPIWrite(RxWait,0x07);
if(mode ==2)
{
SPIWrite(TypeSH,0x01);
}
else
{
SPIWrite(TypeSH,0x00);
}
}
/****************************************************************/
/*名稱: Command_Send */
/*功能: 該函數實現向FM1702發送命令集的功能*/
/*輸入: count, 待發送命令集的長度*/
/* buff, 指向待發送數據的指針*/
/* Comm_Set, 命令碼*/
/*輸出: TRUE, 命令被正確執行*/
/* FALSE, 命令執行錯誤*/
/****************************************************************/
uchar Command_Send(uchar count, uchar idata *buff, uchar Comm_Set)
{
uint j;
uchar idata temp, temp1;
SPIWrite(Command,0x00);
Clear_FIFO();
Write_FIFO(count, buff);
SPIWrite(Command,Comm_Set); /* 命令執行 */
for(j = 0; j < RF_TimeOut; j++) /* 檢查命令執行否 */
{
temp = SPIRead(Command);
temp1 = SPIRead(Int_Req) & 0x80;
if(temp == 0x00)
{
return TRUE;
}
}
return FALSE;
}
/****************************************************************/
/*名稱: Read_E2 */
/*功能: 該函數實現從FM1702的EE中讀出數據*/
/*輸入: lsb, EE地址(低字節) */
/* msb, EE地址(高字節) */
/* count, 待讀出數據EE的字節個數*/
/* buff, 指向待讀出數據的指針*/
/*輸出: TRUE, EE數據正確讀出*/
/* FALSE, EE數據讀出有誤*/
/****************************************************************/
uchar Read_E2(uchar lsb, uchar msb, uchar count, uchar idata *buff)
{
uchar temp;
*buff = lsb;
*(buff + 1) = msb;
*(buff + 2) = count;
temp = Command_Send(3, buff, ReadE2);
Read_FIFO(buff);
if(temp == FALSE) return(TRUE);
return(FALSE);
}
/****************************************************************/
/*名稱: Write_E2 */
/*功能: 該函數實現向FM1702的EE中寫入數據*/
/*輸入: lsb, EE地址(低字節) */
/* msb, EE地址(高字節) */
/* count, 待寫入數據EE的字節個數*/
/* buff, 指向待寫入數據的指針*/
/*輸出: TRUE, EE數據正確寫入*/
/* FALSE, EE數據寫入有誤*/
/****************************************************************/
uchar Write_E2(uchar lsb, uchar msb, uchar count, uchar idata *buff)
{
uchar idata temp, i;
for(i = 0; i < count; i++)
{
*(buff + count - i + 2) = *(buff - i + count);
}
*buff = lsb;
*(buff + 1) = msb;
temp = Command_Send(count + 2, buff, WriteE2);
temp = SPIRead(SecondaryStatus);
temp = temp & 0x40;
if(temp == 0x40)
{
SPIWrite(Command,0x00); //added 2006/03/15
return TRUE;
}
return FALSE;
}
/****************************************************************/
/*名稱: MIF_Halt */
/*功能: 該函數實現暫停MIFARE卡*/
/*輸入: N/A */
/*輸出: FM1702_OK: 應答正確*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_NOTAGERR: 無卡*/
/****************************************************************/
uchar MIF_Halt(void)
{
uchar temp;
uint i;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
SPIWrite(ChannelRedundancy,0x03);
*RevBuffer = RF_CMD_HALT;
*(RevBuffer + 1) = 0x00;
temp = Command_Send(2, RevBuffer, Transmit);
if(temp == TRUE)
{
for(i = 0; i < 0x50; i++)
{
_nop_();
}
return FM1702_OK;
}
else
{
temp = SPIRead(ErrorFlag);
if((temp & 0x02) == 0x02)
{
return(FM1702_PARITYERR);
}
if((temp & 0x04) == 0x04)
{
return(FM1702_FRAMINGERR);
}
return(FM1702_NOTAGERR);
}
}
///////////////////////////////////////////////////////////////////////
// 轉換密鑰格式
///////////////////////////////////////////////////////////////////////
char M500HostCodeKey( uchar *uncoded, uchar *coded)
{
char idata status = FM1702_OK;
uchar idata cnt = 0;
uchar idata ln = 0;
uchar idata hn = 0;
for (cnt = 0; cnt < 6; cnt++)
{
ln = uncoded[cnt] & 0x0F;
hn = uncoded[cnt] >> 4;
coded[cnt * 2 + 1] = (~ln << 4) | ln;
coded[cnt * 2 ] = (~hn << 4) | hn;
}
return FM1702_OK;
}
/****************************************************************/
/*名稱: Load_keyE2 */
/*功能: 該函數實現把E2中密碼存入FM1702的keyRevBuffer中*/
/*輸入: Secnr: EE起始地址*/
/*輸出: True: 密鑰裝載成功*/
/* False: 密鑰裝載失敗*/
/****************************************************************/
uchar Load_keyE2_CPY(uchar Secnr, uchar Mode)
{
char idata status;
uchar temp;
uchar msb = 0;
uchar lsb = 0;
uchar coded_keys[12];
uchar temp1;
if(Secnr >= 0x20)
{
temp1 = Secnr - 0x20;
Secnr = 0x20 + temp1 * 4;
}
temp = Secnr * 12;
if(Mode == RF_CMD_AUTH_LA)
{
if(temp >= 0x80)
{
lsb = temp - 0x80;
msb = 0x01;
}
else
{
lsb = temp + 0x80;
msb = 0x00;
}
}
else
{
msb = 0x01;
lsb = temp + 0x40;
}
status = M500HostCodeKey(&RevBuffer[3],coded_keys);////////////////
status = Write_E2(lsb, msb, 12, coded_keys); ////////////////
if(status != FM1702_OK)
{
return FALSE;
}
return TRUE;
}
/****************************************************************/
/*名稱: Load_keyE2 */
/*功能: 該函數實現把E2中密碼存入FM1702的keyRevBuffer中*/
/*輸入: Secnr: EE起始地址*/
/*輸出: True: 密鑰裝載成功*/
/* False: 密鑰裝載失敗*/
/****************************************************************/
uchar Load_keyE2(uchar Secnr, uchar Mode)
{
uchar temp;
uchar msb = 0;
uchar lsb = 0;
uchar temp1;
if(Secnr >= 0x20)
{
temp1 = Secnr - 0x20;
Secnr = 0x20 + temp1 * 4;
}
temp = Secnr * 12;
if(Mode == RF_CMD_AUTH_LA)
{
if(temp >= 0x80)
{
lsb = temp - 0x80;
msb = 0x01;
}
else
{
lsb = temp + 0x80;
msb = 0x00;
}
}
else
{
msb = 0x01;
lsb = temp + 0x40;
}
RevBuffer[0] = lsb;
RevBuffer[1] = msb;
temp = Command_Send(2, RevBuffer, LoadKeyE2);
temp = SPIRead(ErrorFlag) & 0x40;
if(temp == 0x40)
{
return FALSE;
}
return TRUE;
}
/****************************************************************/
/*名稱: Request */
/*功能: 該函數實現對放入FM1702操作范圍之內的卡片的Request操作*/
/*輸入: mode: ALL(監測所以FM1702操作范圍之內的卡片) */
/* STD(監測在FM1702操作范圍之內處于HALT狀態的卡片) */
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_OK: 應答正確*/
/* FM1702_REQERR: 應答錯誤*/
/****************************************************************/
uchar Request(uchar mode)
{
uchar idata temp;
// SPIWrite(CRCPresetLSB,0x63); //CRCPresetLSB address is 0x23
// SPIWrite(CWConductance,0x3f);
RevBuffer[0] = mode;
SPIWrite(Bit_Frame,0x07);
SPIWrite(ChannelRedundancy,0x03);
temp = SPIRead(Control);
temp = temp & (0xf7);
SPIWrite(Control,temp); //Control reset value is 00
temp = Command_Send(1, RevBuffer, Transceive); //Transceive=0x1E
if(temp == FALSE)
{
return FM1702_NOTAGERR;
}
Read_FIFO(RevBuffer);
temp = Judge_Req(RevBuffer);
if(temp == TRUE)
{
tagtype[0] = RevBuffer[0];
tagtype[1] = RevBuffer[1];
return FM1702_OK;
}
return FM1702_REQERR;
}
/****************************************************************/
/*名稱: AntiColl */
/*功能: 該函數實現對放入FM1702操作范圍之內的卡片的防沖突檢測*/
/*輸入: N/A */
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_SERNRERR: 卡片序列號應答錯誤*/
/* FM1702_OK: 卡片應答正確*/
/****************************************************************/
uchar AntiColl(void)
{
uchar temp;
uchar i;
uchar row, col;
uchar pre_row;
row = 0;
col = 0;
pre_row = 0;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
// SPIWrite(ModConductance,0x3f);
RevBuffer[0] = RF_CMD_ANTICOL;
RevBuffer[1] = 0x20;
SPIWrite(ChannelRedundancy,0x03);
temp = Command_Send(2, RevBuffer, Transceive);
while(1)
{
if(temp == FALSE)
{
return(FM1702_NOTAGERR);
}
//temp = ErrorFlag;
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return FM1702_BYTECOUNTERR;
}
Read_FIFO(RevBuffer);
Save_UID(row, col, temp); /* 將收到的UID放入UID數組中 */
temp = SPIRead(ErrorFlag); /* 判斷接収數據是否出錯 */
temp = temp & 0x01;
if(temp == 0x00)
{
temp = Check_UID(); /* 校驗收到的UID */
if(temp == FALSE)
{
return(FM1702_SERNRERR);
}
return(FM1702_OK);
}
else
{
temp = SPIRead(CollPos); /* 讀取沖突檢測寄存器 */
row = temp / 8;
col = temp % 8;
RevBuffer[0] = RF_CMD_ANTICOL;
Set_BitFraming(row + pre_row, col); /* 設置待發送數據的字節數 */
pre_row = pre_row + row;
for(i = 0; i < pre_row + 1; i++)
{
RevBuffer[i + 2] = UID[i];
}
if(col != 0x00)
{
row = pre_row + 1;
}
else
{
row = pre_row;
}
temp = Command_Send(row + 2, RevBuffer, Transceive);
}
}
}
/****************************************************************/
/*名稱: Select_Card */
/*功能: 該函數實現對放入FM1702操作范圍之內的某張卡片進行選擇*/
/*輸入: N/A */
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_OK: 應答正確*/
/* FM1702_SELERR: 選卡出錯*/
/****************************************************************/
uchar Select_Card(void)
{
uchar temp, i;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
RevBuffer[0] = RF_CMD_SELECT;
RevBuffer[1] = 0x70;
for(i = 0; i < 5; i++)
{
RevBuffer[i + 2] = UID[i];
}
SPIWrite(ChannelRedundancy,0x0f);
temp = Command_Send(7, RevBuffer, Transceive);
if(temp == FALSE)
{
return(FM1702_NOTAGERR);
}
else
{
temp = SPIRead(ErrorFlag);
if((temp & 0x02) == 0x02) return(FM1702_PARITYERR);
if((temp & 0x04) == 0x04) return(FM1702_FRAMINGERR);
if((temp & 0x08) == 0x08) return(FM1702_CRCERR);
temp = SPIRead(FIFO_Length);
if(temp != 1) return(FM1702_BYTECOUNTERR);
Read_FIFO(RevBuffer); /* 從FIFO中讀取應答信息 */
temp = *RevBuffer;
if((temp == 0x18) || (temp == 0x08) || (temp == 0x88) || (temp == 0x53)) /* 判斷應答信號是否正確 */
return(FM1702_OK);
else
return(FM1702_SELERR);
}
}
/****************************************************************/
/*名稱: Authentication */
/*功能: 該函數實現密碼認證的過程*/
/*輸入: UID: 卡片序列號地址*/
/* SecNR: 扇區號*/
/* mode: 模式*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_OK: 應答正確*/
/* FM1702_AUTHERR: 權威認證有錯*/
/****************************************************************/
uchar Authentication(uchar idata *UID, uchar SecNR, uchar mode)
{
uchar idata i;
uchar idata temp, temp1;
uchar temp0;
if(SecNR >= 0x20)
{
temp0 = SecNR -0x20;
SecNR = 0x20 + temp0 * 4;
}
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
// SPIWrite(ModConductance,0X3f);
// temp1 = SPIRead(Control);
// temp1 = temp1 & 0xf7;
// SPIWrite(Control,temp1);
if(mode == RF_CMD_AUTH_LB)
RevBuffer[0] = RF_CMD_AUTH_LB;
else
RevBuffer[0] = RF_CMD_AUTH_LA;
RevBuffer[1] = SecNR * 4 + 3;
for(i = 0; i < 4; i++)
{
RevBuffer[2 + i] = UID[i];
}
SPIWrite(ChannelRedundancy,0x0f);
temp = Command_Send(6, RevBuffer, Authent1);
if(temp == FALSE)
{
return 0x99;
}
temp = SPIRead(ErrorFlag); //ErrorFlag address is 0x0A.
if((temp & 0x02) == 0x02) return FM1702_PARITYERR;
if((temp & 0x04) == 0x04) return FM1702_FRAMINGERR;
if((temp & 0x08) == 0x08) return FM1702_CRCERR;
temp = Command_Send(0, RevBuffer, Authent2);
if(temp == FALSE)
{
return 0x88;
}
temp = SPIRead(ErrorFlag);
if((temp & 0x02) == 0x02) return FM1702_PARITYERR;
if((temp & 0x04) == 0x04) return FM1702_FRAMINGERR;
if((temp & 0x08) == 0x08) return FM1702_CRCERR;
temp1 = SPIRead(Control);
temp1 = temp1 & 0x08;
if(temp1 == 0x08)
{
return FM1702_OK;
}
return FM1702_AUTHERR;
}
/****************************************************************/
/*名稱: MIF_Read */
/*功能: 該函數實現讀MIFARE卡塊的數值*/
/*輸入: buff: 緩沖區首地址*/
/* Block_Adr: 塊地址*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_OK: 應答正確*/
/****************************************************************/
uchar MIF_READ(uchar idata *buff, uchar Block_Adr)
{
uchar idata temp;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
SPIWrite(ModConductance,0x3f);
SPIWrite(ChannelRedundancy,0x0f);
buff[0] = RF_CMD_READ;
buff[1] = Block_Adr;
temp = Command_Send(2, buff, Transceive);
if(temp == 0)
{
return FM1702_NOTAGERR;
}
temp = SPIRead(ErrorFlag);
if((temp & 0x02) == 0x02) return FM1702_PARITYERR;
if((temp & 0x04) == 0x04) return FM1702_FRAMINGERR;
if((temp & 0x08) == 0x08) return FM1702_CRCERR;
temp = SPIRead(FIFO_Length);
if(temp == 0x10)
{
Read_FIFO(buff);
return FM1702_OK;
}
else if(temp == 0x04)
{
Read_FIFO(buff);
return FM1702_OK;
}
else
{
return FM1702_BYTECOUNTERR;
}
}
/****************************************************************/
/*名稱: MIF_Write */
/*功能: 該函數實現寫MIFARE卡塊的數值*/
/*輸入: buff: 緩沖區首地址*/
/* Block_Adr: 塊地址*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_NOTAUTHERR: 未經權威認證*/
/* FM1702_EMPTY: 數據溢出錯誤*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_WRITEERR: 寫卡塊數據出錯*/
/* FM1702_OK: 應答正確*/
/****************************************************************/
uchar MIF_Write(uchar idata *buff, uchar Block_Adr)
{
uchar idata temp;
uchar idata *F_buff;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
F_buff = buff + 0x10;
SPIWrite(ChannelRedundancy,0x07); /* Note: this line is for 1702, different from RC500*/
*F_buff = RF_CMD_WRITE;
*(F_buff + 1) = Block_Adr;
temp = Command_Send(2, F_buff, Transceive);
if(temp == FALSE)
{
return(FM1702_NOTAGERR);
}
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return(FM1702_BYTECOUNTERR);
}
Read_FIFO(F_buff);
temp = *F_buff;
switch(temp)
{
case 0x00: return(FM1702_NOTAUTHERR);
case 0x04: return(FM1702_EMPTY);
case 0x0a: break;
case 0x01: return(FM1702_CRCERR);
case 0x05: return(FM1702_PARITYERR);
default: return(FM1702_WRITEERR);
}
temp = Command_Send(16, buff, Transceive);
if(temp == TRUE)
{
return(FM1702_OK);
}
else
{
temp = SPIRead(ErrorFlag);
if((temp & 0x02) == 0x02)
return(FM1702_PARITYERR);
else if((temp & 0x04) == 0x04)
return(FM1702_FRAMINGERR);
else if((temp & 0x08) == 0x08)
return(FM1702_CRCERR);
else
return(FM1702_WRITEERR);
}
}
/****************************************************************/
/*名稱: MIF_Increment */
/*功能: 該函數實現MIFARE卡自動增值操作*/
/*輸入: buff: 四個字節數值起始地址*/
/* Block_Adr: 塊地址*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_NOTAUTHERR: 未經權威認證*/
/* FM1702_EMPTY: 數據溢出錯誤*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_INCRERR: 卡片增款操作失敗*/
/* FM1702_OK: 應答正確*/
/****************************************************************/
uchar MIF_Increment(uchar idata *buff, uchar Block_Adr)
{
uchar temp;
uchar idata *F_buff;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
F_buff = buff + 4;
*F_buff = RF_CMD_INC;
*(F_buff + 1) = Block_Adr;
SPIWrite(ChannelRedundancy,0x07);
temp = Command_Send(2, F_buff, Transceive);
if(temp == FALSE)
{
return FM1702_NOTAGERR;
}
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return FM1702_BYTECOUNTERR;
}
Read_FIFO(F_buff);
temp = *F_buff;
switch(temp)
{
case 0x00: /* break; */return(FM1702_NOTAUTHERR);
case 0x04: return(FM1702_EMPTY);
case 0x0a: break;
case 0x01: return(FM1702_CRCERR);
case 0x05: return(FM1702_PARITYERR);
default: return(FM1702_INCRERR);
}
temp = Command_Send(4, buff, Transmit);
if(temp == FALSE)
{
return FM1702_INCRERR;
}
return FM1702_OK;
}
/****************************************************************/
/*名稱: MIF_Decrement */
/*功能: 該函數實現MIFARE卡自動減值操作*/
/*輸入: buff: 四個字節數值起始地址*/
/* Block_Adr: 塊地址*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_NOTAUTHERR: 未經權威認證*/
/* FM1702_EMPTY: 數據溢出錯誤*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_DECRERR: 卡片扣款操作失敗*/
/* FM1702_OK: 應答正確*/
/****************************************************************/
uchar MIF_Decrement(uchar idata *buff, uchar Block_Adr)
{
uchar temp;
uchar idata *F_buff;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
F_buff = buff + 4;
*F_buff = RF_CMD_DEC;
*(F_buff + 1) = Block_Adr;
SPIWrite(ChannelRedundancy,0x07);
temp = Command_Send(2, F_buff, Transceive);
if(temp == FALSE)
{
return FM1702_NOTAGERR;
}
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return FM1702_BYTECOUNTERR;
}
Read_FIFO(F_buff);
temp = *F_buff;
switch(temp)
{
case 0x00: /* break; */return(FM1702_NOTAUTHERR);
case 0x04: return(FM1702_EMPTY);
case 0x0a: break;
case 0x01: return(FM1702_CRCERR);
case 0x05: return(FM1702_PARITYERR);
default: return(FM1702_DECRERR);
}
temp = Command_Send(4, buff, Transmit);
if(temp == FALSE)
{
return(FM1702_DECRERR);
}
return FM1702_OK;
}
/****************************************************************/
/*名稱: MIF_Restore */
/*功能: 該函數實現MIFARE卡自動恢復,備份操作*/
/*輸入: Block_Adr: 塊地址*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_NOTAUTHERR: 未經權威認證*/
/* FM1702_EMPTY: 數據溢出錯誤*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_RESTERR: 卡片恢復,備份操作失敗*/
/* FM1702_OK: 應答正確*/
/****************************************************************/
uchar MIF_Restore(uchar Block_Adr)
{
uchar temp, i;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
SPIWrite(ChannelRedundancy,0x07);
*RevBuffer = RF_CMD_RESTORE;
*(RevBuffer + 1) = Block_Adr;
temp = Command_Send(2, RevBuffer, Transceive);
if(temp == FALSE)
{
return FM1702_NOTAGERR;
}
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return FM1702_BYTECOUNTERR;
}
Read_FIFO(RevBuffer);
temp = *RevBuffer;
switch(temp)
{
case 0x00: /* break; */return(FM1702_NOTAUTHERR);
case 0x04: return(FM1702_EMPTY);
case 0x0a: break;
case 0x01: return(FM1702_CRCERR);
case 0x05: return(FM1702_PARITYERR);
default: return(FM1702_RESTERR);
}
for(i = 0; i < 4; i++) RevBuffer[i] = 0x00;
temp = Command_Send(4, RevBuffer, Transmit);
if(temp == FALSE)
{
return FM1702_RESTERR;
}
return FM1702_OK;
}
/****************************************************************/
/*名稱: MIF_Transfer */
/*功能: 該函數實現MIFARE卡電子錢包保存操作*/
/*輸入: Block_Adr: 塊地址*/
/*輸出: FM1702_NOTAGERR: 無卡*/
/* FM1702_BYTECOUNTERR: 接收字節錯誤*/
/* FM1702_NOTAUTHERR: 未經權威認證*/
/* FM1702_EMPTY: 數據溢出錯誤*/
/* FM1702_CRCERR: CRC校驗錯*/
/* FM1702_PARITYERR: 奇偶校驗錯*/
/* FM1702_TRANSERR: 卡片恢復,備份操作失敗*/
/* FM1702_OK: 應答正確*/
/****************************************************************/
uchar MIF_Transfer(uchar Block_Adr)
{
uchar temp;
SPIWrite(CRCPresetLSB,0x63);
SPIWrite(CWConductance,0x3f);
SPIWrite(ChannelRedundancy,0x07);
RevBuffer[0] = RF_CMD_TRANSFER;
RevBuffer[1] = Block_Adr;
temp = Command_Send(2, RevBuffer, Transceive);
if(temp == FALSE)
{
return FM1702_NOTAGERR;
}
temp = SPIRead(FIFO_Length);
if(temp == 0)
{
return FM1702_BYTECOUNTERR;
}
Read_FIFO(RevBuffer);
temp = *RevBuffer;
switch(temp)
{
case 0x00: /* break; */return(FM1702_NOTAUTHERR);
case 0x04: return(FM1702_EMPTY);
case 0x0a: return(FM1702_OK);
case 0x01: return(FM1702_CRCERR);
case 0x05: return(FM1702_PARITYERR);
default: return(FM1702_TRANSERR);
}
}
#if 0
/****************************************************************/
/*名稱: HL_Active */
/*功能: 該函數實現高級MIFARE卡激活命令*/
/*輸入: Secnr: 扇區號*/
/* Block_Adr: 塊地址*/
/*輸出: 操作狀態碼*/
/* 讀出數據存于RevBuffer中*/
/****************************************************************/
uchar HL_Active(uchar Block_Adr, uchar Mode)
{
uchar temp;
Secnr = Block_Adr / 4;
MIF_Halt(); /* Halt */
temp = Request(RF_CMD_REQUEST_STD);
if(temp != FM1702_OK)
{
return(FM1702_REQERR);
}
temp = AntiColl();
if(temp != FM1702_OK)
{
return(FM1702_ANTICOLLERR);
}
temp = Select_Card();
if(temp != FM1702_OK)
{
return(FM1702_SELERR);
}
Load_keyE2_CPY(Secnr, Mode); //%40
……………………
…………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼
所有資料51hei提供下載:
1702SL開發板源碼.rar
(42.75 KB, 下載次數: 37)
2018-12-9 00:49 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
作者:
admin
時間:
2018-12-9 00:51
補全原理圖或者詳細說明一下電路連接即可獲得100+黑幣
作者:
zilchwei
時間:
2019-1-4 13:14
不錯!~~~~~
作者:
wzswp
時間:
2022-8-19 14:14
謝謝分享
作者:
aaronjob2012
時間:
2023-10-20 09:01
可否提供原理圖學習呀,謝謝!
作者:
qq250061277
時間:
2023-11-21 17:35
請問,是什么型號的單片機?
歡迎光臨 (http://m.zg4o1577.cn/bbs/)
Powered by Discuz! X3.1
主站蜘蛛池模板:
一区二区三区四区五区在线视频
|
欧美11一13sex性hd
|
国产一在线
|
我想看国产一级毛片
|
亚洲品质自拍视频
|
久久成人免费
|
午夜爽爽爽男女免费观看
|
在线亚洲免费
|
一区二区三区四区在线
|
色网站视频
|
亚洲精品乱码久久久久久蜜桃91
|
www.国产视频
|
精品一二三
|
亚洲高清在线视频
|
一级亚洲
|
成人国产免费观看
|
美女视频h
|
亚洲精品乱码久久久久久9色
|
日韩高清不卡
|
国产精品美女久久久久久免费
|
国产视频精品区
|
免费观看av网站
|
久久躁日日躁aaaaxxxx
|
欧美精品一区二区三区四区 在线
|
中文字幕亚洲国产
|
国产日韩精品在线
|
免费观看一区二区三区毛片
|
乳色吐息在线观看
|
欧美一区二区视频
|
97伊人
|
国产精品毛片在线
|
日韩一区二区三区四区五区
|
国产精品影视在线观看
|
玖玖视频网
|
日韩在线欧美
|
免费视频一区二区
|
caoporn视频在线
|
狠狠干网站
|
国产精品免费视频一区
|
色眯眯视频在线观看
|
日韩精品在线网站
|