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.

Table 2-2. DMA Settings for Private I3C/I2C Read and IBI Transactions
DMA2 ParameterSetting
SSASource AddressBeginning of software transmit buffer
SSZSource SizeNumber of bytes to send
SMRSource Memory RegionGPR
SMODESource Address ModeSPTR is incremented
DSADestination Address&I3CxTXB
DSZDestination Size1
DMODEDestination Address ModeDPTR remains unchanged
SIRQENStart Trigger EnableYes (enable when ready to send data)
SIRQStart TriggerI3CxTX (I3CxTXB is empty)
SSTPSource Counter Reload StopYes – SIRQEN is cleared when SCNT reloads
DSTPDestination Counter Reload StopNo – SIRQEN is not cleared
AIRQENAbort Trigger EnableNo (stops when DCNT reloads) (1)
AIRQAbort TriggerN/A(1)
Note:
  1. A TCOMPIF-based abort trigger (through the system level I3Cx General Interrupt flag) can optionally be set to abort the DMA when the controller ends the transaction prematurely.

How It Works:

  1. 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.
  2. When the I3CxTXB register becomes empty again, the events in step 1 are repeated.
  3. 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.
  4. 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
}