mirror of
https://github.com/RfidResearchGroup/proxmark3.git
synced 2025-01-11 02:31:23 +08:00
207 lines
6 KiB
C
207 lines
6 KiB
C
//-----------------------------------------------------------------------------
|
|
// Iceman, July 2018
|
|
// edits by - Anticat, August 2018
|
|
//
|
|
// This code is licensed to you under the terms of the GNU GPL, version 2 or,
|
|
// at your option, any later version. See the LICENSE.txt file for the text of
|
|
// the license.
|
|
//-----------------------------------------------------------------------------
|
|
// The main USART code, for serial communications over FPC connector
|
|
//-----------------------------------------------------------------------------
|
|
#include "usart.h"
|
|
#include "string.h"
|
|
#include "../armsrc/ticks.h" // startcountus
|
|
|
|
volatile AT91PS_USART pUS1 = AT91C_BASE_US1;
|
|
volatile AT91PS_PIO pPIO = AT91C_BASE_PIOA;
|
|
volatile AT91PS_PDC pPDC = AT91C_BASE_PDC_US1;
|
|
|
|
/*
|
|
void usart_close(void) {
|
|
// Reset the USART mode
|
|
pUS1->US_MR = 0;
|
|
|
|
// Reset the baud rate divisor register
|
|
pUS1->US_BRGR = 0;
|
|
|
|
// Reset the Timeguard Register
|
|
pUS1->US_TTGR = 0;
|
|
|
|
// Disable all interrupts
|
|
pUS1->US_IDR = 0xFFFFFFFF;
|
|
|
|
// Abort the Peripheral Data Transfers
|
|
pUS1->US_PTCR = AT91C_PDC_RXTDIS | AT91C_PDC_TXTDIS;
|
|
|
|
// Disable receiver and transmitter and stop any activity immediately
|
|
pUS1->US_CR = AT91C_US_TXDIS | AT91C_US_RXDIS | AT91C_US_RSTTX | AT91C_US_RSTRX;
|
|
}
|
|
*/
|
|
|
|
static uint8_t us_inbuf[sizeof(UsbCommand)];
|
|
static uint8_t us_outbuf[sizeof(UsbCommand)];
|
|
/*
|
|
// transfer from client to device
|
|
inline int16_t usart_readbuffer(uint8_t *data) {
|
|
uint32_t rcr = pUS1->US_RCR;
|
|
if (rcr < sizeof(us_inbuf)) {
|
|
pUS1->US_PTCR = AT91C_PDC_RXTDIS;
|
|
memcpy(data, us_inbuf, sizeof(us_inbuf) - rcr);
|
|
// Reset DMA buffer
|
|
pUS1->US_RPR = (uint32_t)us_inbuf;
|
|
pUS1->US_RCR = sizeof(us_inbuf);
|
|
pUS1->US_PTCR = AT91C_PDC_RXTEN;
|
|
return sizeof(us_inbuf) - rcr;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
*/
|
|
|
|
uint8_t check = 0;
|
|
inline int16_t usart_readbuffer(uint8_t *data) {
|
|
// Check if the first PDC bank is free
|
|
if (pUS1->US_RCR == 0) {
|
|
pUS1->US_RPR = (uint32_t)data;
|
|
pUS1->US_RCR = sizeof(UsbCommand);
|
|
pUS1->US_PTCR = AT91C_PDC_RXTEN | AT91C_PDC_TXTEN;
|
|
check = 0;
|
|
return 2;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void usart_readcheck(uint8_t *data, size_t len) {
|
|
if (pUS1->US_RCR < len) {
|
|
if (check == 0) {
|
|
StartCountUS();
|
|
check = 1;
|
|
}
|
|
//300ms
|
|
if (GetCountUS() > 300000) {
|
|
pUS1->US_RPR = (uint32_t)data;
|
|
pUS1->US_RCR = len;
|
|
check = 0;
|
|
}
|
|
} else {
|
|
check = 0;
|
|
}
|
|
}
|
|
|
|
inline bool usart_dataavailable(void) {
|
|
return pUS1->US_RCR < sizeof(us_inbuf);
|
|
}
|
|
|
|
inline int16_t usart_readcommand(uint8_t *data) {
|
|
if (pUS1->US_RCR == 0)
|
|
return usart_readbuffer(data);
|
|
|
|
return 0;
|
|
}
|
|
|
|
inline bool usart_commandavailable(void) {
|
|
return (pUS1->US_RCR == 0);
|
|
}
|
|
/*
|
|
// transfer from device to client
|
|
inline int16_t usart_writebuffer(uint8_t *data, size_t len) {
|
|
|
|
|
|
if (pUS1->US_CSR & AT91C_US_ENDTX) {
|
|
memcpy(us_outbuf, data, len);
|
|
pUS1->US_TPR = (uint32_t)us_outbuf;
|
|
pUS1->US_TCR = len;
|
|
pUS1->US_PTCR = AT91C_PDC_TXTEN;
|
|
while (!(pUS1->US_CSR & AT91C_US_ENDTX)) {};
|
|
pUS1->US_PTCR = AT91C_PDC_TXTDIS;
|
|
return len;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
*/
|
|
|
|
// transfer from device to client
|
|
inline int16_t usart_writebuffer(uint8_t *data, size_t len) {
|
|
|
|
while (pUS1->US_TCR && pUS1->US_TNCR) {};
|
|
|
|
// Check if the first PDC bank is free
|
|
if (pUS1->US_TCR == 0) {
|
|
memcpy(us_outbuf, data, len);
|
|
pUS1->US_TPR = (uint32_t)us_outbuf;
|
|
pUS1->US_TCR = sizeof(us_outbuf);
|
|
pUS1->US_PTCR = AT91C_PDC_TXTEN | AT91C_PDC_RXTEN;
|
|
}
|
|
// Check if the second PDC bank is free
|
|
else if (pUS1->US_TNCR == 0) {
|
|
memcpy(us_outbuf, data, len);
|
|
pUS1->US_TNPR = (uint32_t)us_outbuf;
|
|
pUS1->US_TNCR = sizeof(us_outbuf);
|
|
pUS1->US_PTCR = AT91C_PDC_TXTEN | AT91C_PDC_RXTEN;
|
|
}
|
|
//wait until finishing transfer
|
|
while (pUS1->US_TCR && pUS1->US_TNCR) {};
|
|
return 0;
|
|
}
|
|
|
|
|
|
void usart_init(void) {
|
|
|
|
// For a nice detailed sample, interrupt driven but still relevant.
|
|
// See https://www.sparkfun.com/datasheets/DevTools/SAM7/at91sam7%20serial%20communications.pdf
|
|
|
|
// disable & reset receiver / transmitter for configuration
|
|
pUS1->US_CR = (AT91C_US_RSTRX | AT91C_US_RSTTX | AT91C_US_RXDIS | AT91C_US_TXDIS);
|
|
|
|
//enable the USART1 Peripheral clock
|
|
AT91C_BASE_PMC->PMC_PCER = (1 << AT91C_ID_US1);
|
|
|
|
// disable PIO control of receive / transmit pins
|
|
pPIO->PIO_PDR |= (AT91C_PA21_RXD1 | AT91C_PA22_TXD1);
|
|
|
|
// enable peripheral mode A on receive / transmit pins
|
|
pPIO->PIO_ASR |= (AT91C_PA21_RXD1 | AT91C_PA22_TXD1);
|
|
pPIO->PIO_BSR = 0;
|
|
|
|
// enable pull-up on receive / transmit pins (see 31.5.1 I/O Lines)
|
|
pPIO->PIO_PPUER |= (AT91C_PA21_RXD1 | AT91C_PA22_TXD1);
|
|
|
|
// set mode
|
|
pUS1->US_MR = AT91C_US_USMODE_NORMAL | // normal mode
|
|
AT91C_US_CLKS_CLOCK | // MCK (48MHz)
|
|
AT91C_US_OVER | // oversampling
|
|
AT91C_US_CHRL_8_BITS | // 8 bits
|
|
AT91C_US_PAR_NONE | // parity: none
|
|
AT91C_US_NBSTOP_1_BIT | // 1 stop bit
|
|
AT91C_US_CHMODE_NORMAL; // channel mode: normal
|
|
|
|
// all interrupts disabled
|
|
pUS1->US_IDR = 0xFFFF;
|
|
|
|
pUS1->US_BRGR = 48054841 / (AT91_BAUD_RATE << 3);
|
|
// Need speed?
|
|
//pUS1->US_BRGR = 48054841 / (460800 << 3);
|
|
|
|
// Write the Timeguard Register
|
|
pUS1->US_TTGR = 0;
|
|
pUS1->US_RTOR = 0;
|
|
pUS1->US_FIDI = 0;
|
|
pUS1->US_IF = 0;
|
|
|
|
// Disable double buffers for now
|
|
pUS1->US_TNPR = (uint32_t)0;
|
|
pUS1->US_TNCR = 0;
|
|
pUS1->US_RNPR = (uint32_t)0;
|
|
pUS1->US_RNCR = 0;
|
|
|
|
// re-enable receiver / transmitter
|
|
pUS1->US_CR = (AT91C_US_RXEN | AT91C_US_TXEN);
|
|
|
|
// ready to receive
|
|
pUS1->US_RPR = (uint32_t)us_inbuf;
|
|
pUS1->US_RCR = 0;
|
|
pUS1->US_RNCR = 0;
|
|
pUS1->US_PTCR = AT91C_PDC_RXTEN;
|
|
}
|