2009-09-08 20:07:35 +00:00
|
|
|
/**
|
|
|
|
* \file
|
2012-03-05 16:28:06 +00:00
|
|
|
* Driver for the cc2430 DMA controller. Can be assigned to any bank
|
|
|
|
*
|
2009-09-08 20:07:35 +00:00
|
|
|
* \author
|
|
|
|
* Original: Martti Huttunen <martti@sensinode.com>
|
|
|
|
* Port: Zach Shelby <zach@sensinode.com>
|
2012-03-05 16:28:06 +00:00
|
|
|
* Further Modifications:
|
|
|
|
* George Oikonomou <oikonomou@users.sourceforge.net>
|
2010-01-25 23:12:09 +00:00
|
|
|
*
|
2009-09-08 20:07:35 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "contiki.h"
|
|
|
|
#include "dev/dma.h"
|
|
|
|
#include "cc2430_sfr.h"
|
|
|
|
|
2012-03-05 16:28:06 +00:00
|
|
|
#if DMA_ON
|
|
|
|
struct dma_config dma_conf[DMA_CHANNEL_COUNT]; /* DMA Descriptors */
|
|
|
|
struct process * dma_callback[DMA_CHANNEL_COUNT];
|
2009-09-08 20:07:35 +00:00
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void
|
2012-03-05 16:28:06 +00:00
|
|
|
dma_init(void)
|
2009-09-08 20:07:35 +00:00
|
|
|
{
|
|
|
|
uint16_t tmp_ptr;
|
2012-03-05 16:28:06 +00:00
|
|
|
|
|
|
|
memset(dma_conf, 0, 4 * sizeof(dma_config_t));
|
|
|
|
|
|
|
|
for(tmp_ptr = 0; tmp_ptr < DMA_CHANNEL_COUNT; tmp_ptr++) {
|
2009-09-08 20:07:35 +00:00
|
|
|
dma_callback[tmp_ptr] = 0;
|
|
|
|
}
|
2012-03-05 16:28:06 +00:00
|
|
|
|
|
|
|
/* The address of the descriptor for Channel 0 is configured separately */
|
2009-09-08 20:07:35 +00:00
|
|
|
tmp_ptr = (uint16_t) &(dma_conf[0]);
|
2012-03-05 16:28:06 +00:00
|
|
|
DMA0CFGH = tmp_ptr >> 8;
|
|
|
|
DMA0CFGL = tmp_ptr;
|
2009-09-08 20:07:35 +00:00
|
|
|
|
2012-03-05 16:28:06 +00:00
|
|
|
/*
|
|
|
|
* Descriptors for Channels 1-4 must be consecutive in RAM.
|
|
|
|
* We write the address of the 1st one to the register and the rest are
|
|
|
|
* derived by the SoC
|
|
|
|
*/
|
|
|
|
#if (DMA_CHANNEL_COUNT > 1)
|
|
|
|
tmp_ptr = (uint16_t) &(dma_conf[1]);
|
2009-09-08 20:07:35 +00:00
|
|
|
DMA1CFGH = tmp_ptr >> 8;
|
|
|
|
DMA1CFGL = tmp_ptr;
|
2012-03-05 16:28:06 +00:00
|
|
|
#endif
|
2009-09-08 20:07:35 +00:00
|
|
|
|
2012-03-05 16:28:06 +00:00
|
|
|
IEN1_DMAIE = 1; /* Enable DMA interrupts */
|
2009-09-08 20:07:35 +00:00
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
2012-03-05 16:28:06 +00:00
|
|
|
/*
|
|
|
|
* Associate process p with DMA channel c. When a transfer on that channel
|
|
|
|
* completes, the ISR will poll this process.
|
2009-09-08 20:07:35 +00:00
|
|
|
*/
|
|
|
|
void
|
2012-03-05 16:28:06 +00:00
|
|
|
dma_associate_process(struct process * p, uint8_t c)
|
2009-09-08 20:07:35 +00:00
|
|
|
{
|
2012-03-05 16:28:06 +00:00
|
|
|
if((!c) || (c >= DMA_CHANNEL_COUNT)) {
|
2009-09-08 20:07:35 +00:00
|
|
|
return;
|
|
|
|
}
|
2012-03-05 16:28:06 +00:00
|
|
|
|
|
|
|
if(p) {
|
|
|
|
dma_conf[c].inc_prio |= 8; /* Enable interrupt generation */
|
|
|
|
IEN1_DMAIE = 1; /* Make sure DMA interrupts are acknowledged */
|
2009-09-08 20:07:35 +00:00
|
|
|
}
|
2012-03-05 16:28:06 +00:00
|
|
|
dma_callback[c] = p;
|
2009-09-08 20:07:35 +00:00
|
|
|
}
|
2012-03-05 16:28:06 +00:00
|
|
|
/*---------------------------------------------------------------------------*/
|
2009-09-08 20:07:35 +00:00
|
|
|
#endif
|