2.2 Private I3C/I2C Read and In-Band Interrupt Transactions using DMA
For a Private I3C/I2C Read transaction, it is recommended for the user to configure the DMA such that the data is transferred from a software buffer in the GPR space to the I3CxTXB register. The DMA can be triggered every time the I3CxTXB register is empty, which is available as a system level I3CxTXIF Transmit Interrupt. It is not required to enable the system level I3CxTXIF interrupt for it to be used as a DMA trigger.
The DMA can be configured to transfer a set number of bytes to write to the I3CxTXB register. If it is possible that the controller might read less data bytes than that, the DMA can optionally be configured to be stopped when a TCOMPIF interrupt flag is observed. The TCOMPIF interrupt is available through the system level I3Cx General Interrupt. Even though it is not required to enable the system level I3Cx interrupt to be used as a DMA trigger, the module level TCOMPIF interrupt must be enabled for this function.
For an IBI transaction, while the mandatory data byte is transmitted from the I3CxIBIMDB register, the payload is transmitted from the Transmit FIFO just like a Private Read transaction. The DMA configuration for sending the IBI payload from a software buffer is the same as that of a Private Read transaction.
DMA2 Parameter | Setting | |
---|---|---|
SSA | Source Address | Beginning of software transmit buffer |
SSZ | Source Size | Number of bytes to send |
SMR | Source Memory Region | GPR |
SMODE | Source Address Mode | SPTR is incremented |
DSA | Destination Address | &I3CxTXB |
DSZ | Destination Size | 1 |
DMODE | Destination Address Mode | DPTR remains unchanged |
SIRQEN | Start Trigger Enable | Yes (enable when ready to send data) |
SIRQ | Start Trigger | I3CxTX (I3CxTXB is empty) |
SSTP | Source Counter Reload Stop | Yes – SIRQEN is cleared when SCNT reloads |
DSTP | Destination Counter Reload Stop | No – SIRQEN is not cleared |
AIRQEN | Abort Trigger Enable | No (stops when DCNT reloads) (1) |
AIRQ | Abort Trigger | N/A(1) |
Note:
|
How It Works:
- The DMA is triggered when SIRQEN and
the I3CxTXIF interrupt flag are set. The content of the first address of the
software transmit buffer is transferred to the I3CxTXB register. SPTR is incremented
and now points to the following address in the software transmit buffer. Because DSZ
=
1
, the DMA stops after the first transfer. Since DSTP =0
, SIRQEN remains set, and DMA waits for the next trigger. - When the I3CxTXB register becomes empty again, the events in step 1 are repeated.
- After the desired number of bytes
have been transferred by the DMA (SSZ value), the SPTR reaches the end of the
allocated software buffer, and SCNT reloads. Since SSTP =
1
, SIRQEN is cleared, and no further DMA transfers happen. - Optionally, if an abort trigger is configured using TCOMPIF, the DMA is aborted when the transaction is completed on the bus, regardless of the number of bytes transferred, which clears the SIRQEN bit.
Example Code for Private I3C/I2C/IBI Read Using DMA
// Example code for PIC18-Q20 device family uint8_t txData[SWBUF_SIZE]; // Private Read/IBI DMA2 void DMA2_Initialize(void) { // Source and Destination settings DMASELECT = 1; // DMA Instance DMAnSSA = &txData; // Source Address: Variable in GPR space DMAnDSA = &I3C1TXB; // Destination Address: Register in SFR space DMAnSSZ = SWBUF_SIZE; // Source Size: Size of variable in GPR space DMAnDSZ = 1; // Destination Size: Only one register DMAnSIRQ = 0x41; // Start Trigger: I3C1TX DMAnAIRQ = 0x00; // Abort Trigger: None DMAnCON1bits.SMODE = 0b01; // Source Mode: Pointer incremented DMAnCON1bits.SSTP = 1; // Source Reload: Yes; SIRQEN is cleared DMAnCON1bits.SMR = 0b00; // Source Region: SFR/GPR space DMAnCON1bits.DMODE = 0b00; // Destination Mode: Pointer unchanged DMAnCON1bits.DSTP = 0; // Destination Reload: No; SIRQEN not cleared DMAnCON0bits.SIRQEN = 0; // Start Trigger: Disabled (enable later when ready) DMAnCON0bits.AIRQEN = 0; // Abort Trigger: Disabled (stops when DCNT reloads) DMAnCON0bits.DGO = 0; // DMA Transaction: Do not start transaction yet // Clear interrupt flags PIR1bits.DMA2DCNTIF = 0; // Destination Count Interrupt PIR1bits.DMA2SCNTIF = 0; // Source Count Interrupt PIR1bits.DMA2AIF = 0; // Abort Interrupt PIR1bits.DMA2ORIF = 0; // Overrun Interrupt // Enable appropriate interrupts as needed PIE1bits.DMA2DCNTIE = 0; // Destination Count Interrupt PIE1bits.DMA2SCNTIE = 1; // Source Count Interrupt (enabled) PIE1bits.DMA2AIE = 0; // Abort Interrupt PIE1bits.DMA2ORIE = 0; // Overrun Interrupt //Enable DMA DMAnCON0bits.EN = 1; } void I3C1_SendDataWithDMA(void) { // Initialize txData[] with data to be sent for Private Read (or payload to be sent for IBI) initializeTxData(); // Clear TXB+TXFIFO if appropriate I3C1CON0bits.CLRTXB = 1; // Enable One-shot ACK (if ACKP = NACK by default) I3C1CON1bits.ACKPOS = 1; // Start DMA transfer with trigger DMASELECT = 1; DMAnCON0bits.SIRQEN = 1; } void I3C1_RequestIBI(void) { // Configure IBI mandatory data byte I3C1IBIMDB = 0xAA; // Configure DMA and send payload I3C1_SendDataWithDMA(); // Request IBI I3C1CON0bits.IBIREQ = 1; } void __interrupt(irq(IRQ_DMA2SCNT)) DMA2_SCNT_ISR(void) { PIR1bits.DMA2SCNTIF = 0; // Code execution reaches here when DMA has completed transfer }