enable at91sam9260 mmu, update SDIO and EMAC drivers

This commit is contained in:
weety
2013-04-02 20:24:51 +08:00
parent 92d4c1939b
commit fb9ea5eada
5 changed files with 341 additions and 254 deletions

11
bsp/at91sam9260/at91_mci.c Normal file → Executable file
View File

@@ -27,6 +27,12 @@
#define mci_dbg(fmt, ...) #define mci_dbg(fmt, ...)
#endif #endif
#define MMU_NOCACHE_ADDR(a) ((rt_uint32_t)a | (1UL<<31))
extern void mmu_clean_dcache(rt_uint32_t buffer, rt_uint32_t size);
extern void mmu_invalidate_dcache(rt_uint32_t buffer, rt_uint32_t size);
#define AT91_MCI_ERRORS (AT91_MCI_RINDE | AT91_MCI_RDIRE | AT91_MCI_RCRCE \ #define AT91_MCI_ERRORS (AT91_MCI_RINDE | AT91_MCI_RDIRE | AT91_MCI_RCRCE \
| AT91_MCI_RENDE | AT91_MCI_RTOE | AT91_MCI_DCRCE \ | AT91_MCI_RENDE | AT91_MCI_RTOE | AT91_MCI_DCRCE \
| AT91_MCI_DTOE | AT91_MCI_OVRE | AT91_MCI_UNRE) | AT91_MCI_DTOE | AT91_MCI_OVRE | AT91_MCI_UNRE)
@@ -334,6 +340,7 @@ static void at91_mci_send_command(struct at91_mci *mci, struct rt_mmcsd_cmd *cmd
* Handle a read * Handle a read
*/ */
mmu_invalidate_dcache(data->buf, data->blksize*data->blks);
at91_mci_init_dma_read(mci); at91_mci_init_dma_read(mci);
ier = AT91_MCI_ENDRX /* | AT91_MCI_RXBUFF */; ier = AT91_MCI_ENDRX /* | AT91_MCI_RXBUFF */;
} }
@@ -359,13 +366,15 @@ static void at91_mci_send_command(struct at91_mci *mci, struct rt_mmcsd_cmd *cmd
return; return;
} }
rt_memset(mci->buf, 0, 12); rt_memset(mci->buf, 0, 12);
rt_memcpy(mci->buf, data->buf, block_length * blocks); rt_memcpy(mci->buf, data->buf, length);
mmu_clean_dcache(mci->buf, length);
at91_mci_write(AT91_PDC_TPR, (rt_uint32_t)(mci->buf)); at91_mci_write(AT91_PDC_TPR, (rt_uint32_t)(mci->buf));
at91_mci_write(AT91_PDC_TCR, (data->blksize & 0x3) ? at91_mci_write(AT91_PDC_TCR, (data->blksize & 0x3) ?
length : length / 4); length : length / 4);
} }
else else
{ {
mmu_clean_dcache(data->buf, data->blksize*data->blks);
at91_mci_write(AT91_PDC_TPR, (rt_uint32_t)(data->buf)); at91_mci_write(AT91_PDC_TPR, (rt_uint32_t)(data->buf));
at91_mci_write(AT91_PDC_TCR, (data->blksize & 0x3) ? at91_mci_write(AT91_PDC_TCR, (data->blksize & 0x3) ?
length : length / 4); length : length / 4);

View File

@@ -341,7 +341,7 @@ void rt_hw_board_init()
rt_hw_uart_init(); rt_hw_uart_init();
/* initialize mmu */ /* initialize mmu */
//rt_hw_mmu_init(); rt_hw_mmu_init();
/* initialize timer0 */ /* initialize timer0 */
rt_hw_timer_init(); rt_hw_timer_init();

View File

@@ -18,6 +18,26 @@
#include <at91sam926x.h> #include <at91sam926x.h>
#include "macb.h" #include "macb.h"
#define MMU_NOCACHE_ADDR(a) ((rt_uint32_t)a | (1UL<<31))
extern void mmu_clean_dcache(rt_uint32_t buffer, rt_uint32_t size);
extern void mmu_invalidate_dcache(rt_uint32_t buffer, rt_uint32_t size);
/* Cache macros - Packet buffers would be from pbuf pool which is cached */
#define EMAC_VIRT_NOCACHE(addr) (addr)
#define EMAC_CACHE_INVALIDATE(addr, size) \
mmu_invalidate_dcache((rt_uint32_t)addr, size)
#define EMAC_CACHE_WRITEBACK(addr, size) \
mmu_clean_dcache((rt_uint32_t)addr, size)
#define EMAC_CACHE_WRITEBACK_INVALIDATE(addr, size) \
mmu_clean_invalidated_dcache((rt_uint32_t)addr, size)
/* EMAC has BD's in cached memory - so need cache functions */
#define BD_CACHE_INVALIDATE(addr, size)
#define BD_CACHE_WRITEBACK(addr, size)
#define BD_CACHE_WRITEBACK_INVALIDATE(addr, size)
#define CONFIG_RMII #define CONFIG_RMII
#define MACB_RX_BUFFER_SIZE 4096*4 #define MACB_RX_BUFFER_SIZE 4096*4
@@ -378,6 +398,8 @@ static rt_err_t rt_macb_init(rt_device_t dev)
} }
macb->rx_tail = macb->tx_head = macb->tx_tail = 0; macb->rx_tail = macb->tx_head = macb->tx_tail = 0;
BD_CACHE_WRITEBACK_INVALIDATE(macb->rx_ring, MACB_RX_RING_SIZE * sizeof(struct macb_dma_desc));
BD_CACHE_WRITEBACK_INVALIDATE(macb->tx_ring, MACB_TX_RING_SIZE * sizeof(struct macb_dma_desc));
macb_writel(macb, RBQP, macb->rx_ring_dma); macb_writel(macb, RBQP, macb->rx_ring_dma);
macb_writel(macb, TBQP, macb->tx_ring_dma); macb_writel(macb, TBQP, macb->tx_ring_dma);
@@ -470,28 +492,12 @@ static rt_err_t rt_macb_control(rt_device_t dev, rt_uint8_t cmd, void *args)
/* transmit packet. */ /* transmit packet. */
rt_err_t rt_macb_tx( rt_device_t dev, struct pbuf* p) rt_err_t rt_macb_tx( rt_device_t dev, struct pbuf* p)
{ {
struct pbuf* q;
rt_uint8_t* bufptr, *buf = RT_NULL;
unsigned long ctrl; unsigned long ctrl;
struct rt_macb_eth *macb = dev->user_data; struct rt_macb_eth *macb = dev->user_data;
unsigned int tx_head = macb->tx_head; unsigned int tx_head = macb->tx_head;
/* lock macb device */
rt_sem_take(&sem_lock, RT_WAITING_FOREVER); rt_sem_take(&sem_lock, RT_WAITING_FOREVER);
buf = rt_malloc(p->tot_len); EMAC_CACHE_WRITEBACK(p->payload, p->tot_len);
if (!buf) {
rt_kprintf("%s:alloc buf failed\n", __func__);
return -RT_ENOMEM;
}
bufptr = buf;
for (q = p; q != NULL; q = q->next)
{
memcpy(bufptr, q->payload, q->len);
bufptr += q->len;
}
ctrl = p->tot_len & TXBUF_FRMLEN_MASK; ctrl = p->tot_len & TXBUF_FRMLEN_MASK;
ctrl |= TXBUF_FRAME_END; ctrl |= TXBUF_FRAME_END;
if (tx_head == (MACB_TX_RING_SIZE - 1)) { if (tx_head == (MACB_TX_RING_SIZE - 1)) {
@@ -499,17 +505,14 @@ rt_err_t rt_macb_tx( rt_device_t dev, struct pbuf* p)
macb->tx_head = 0; macb->tx_head = 0;
} else } else
macb->tx_head++; macb->tx_head++;
macb->tx_ring[tx_head].ctrl = ctrl; macb->tx_ring[tx_head].ctrl = ctrl;
macb->tx_ring[tx_head].addr = (rt_uint32_t)buf; macb->tx_ring[tx_head].addr = (rt_uint32_t)p->payload;
BD_CACHE_WRITEBACK_INVALIDATE(macb->tx_ring[tx_head], sizeof(struct macb_dma_desc));
macb_writel(macb, NCR, MACB_BIT(TE) | MACB_BIT(RE) | MACB_BIT(TSTART)); macb_writel(macb, NCR, MACB_BIT(TE) | MACB_BIT(RE) | MACB_BIT(TSTART));
/* unlock macb device */
rt_sem_release(&sem_lock);
rt_sem_release(&sem_lock);
/* wait ack */ /* wait ack */
rt_sem_take(&sem_ack, RT_WAITING_FOREVER); rt_sem_take(&sem_ack, RT_WAITING_FOREVER);
rt_free(buf);
return RT_EOK; return RT_EOK;
} }
@@ -546,12 +549,7 @@ struct pbuf *rt_macb_rx(rt_device_t dev)
int wrapped = 0; int wrapped = 0;
rt_uint32_t status; rt_uint32_t status;
struct pbuf* q;
rt_uint8_t *buf = RT_NULL;
/* lock macb device */
rt_sem_take(&sem_lock, RT_WAITING_FOREVER); rt_sem_take(&sem_lock, RT_WAITING_FOREVER);
for (;;) { for (;;) {
if (!(macb->rx_ring[rx_tail].addr & RXADDR_USED)) if (!(macb->rx_ring[rx_tail].addr & RXADDR_USED))
break; break;
@@ -574,39 +572,18 @@ struct pbuf *rt_macb_rx(rt_device_t dev)
} }
if (wrapped) { if (wrapped) {
unsigned int headlen, taillen; unsigned int headlen, taillen;
buf = rt_malloc(len);
if (!buf)
{
rt_kprintf("%s:alloc memory failed\n", __func__);
pbuf_free(p);
p = RT_NULL;
break;
}
headlen = 128 * (MACB_RX_RING_SIZE headlen = 128 * (MACB_RX_RING_SIZE
- macb->rx_tail); - macb->rx_tail);
taillen = len - headlen; taillen = len - headlen;
memcpy((void *)buf, EMAC_CACHE_INVALIDATE(buffer, headlen);
buffer, headlen); EMAC_CACHE_INVALIDATE(macb->rx_buffer, taillen);
memcpy((void *)((unsigned int)buf + headlen), memcpy((void *)p->payload, buffer, headlen);
memcpy((void *)((unsigned int)p->payload + headlen),
macb->rx_buffer, taillen); macb->rx_buffer, taillen);
buffer = (void *)buf;
for (q = p; q != RT_NULL; q= q->next)
{
/* read data from device */
memcpy((void *)q->payload, buffer, q->len);
buffer = (void *)((unsigned int)buffer + q->len);
}
rt_free(buf);
buf = RT_NULL;
} else { } else {
for (q = p; q != RT_NULL; q= q->next) EMAC_CACHE_INVALIDATE(buffer, len);
{ memcpy((void *)p->payload, buffer, p->len);
/* read data from device */
memcpy((void *)q->payload, buffer, q->len);
buffer = (void *)((unsigned int)buffer + q->len);
}
} }
if (++rx_tail >= MACB_RX_RING_SIZE) if (++rx_tail >= MACB_RX_RING_SIZE)
@@ -620,7 +597,7 @@ struct pbuf *rt_macb_rx(rt_device_t dev)
} }
} }
} }
/* unlock macb device */
rt_sem_release(&sem_lock); rt_sem_release(&sem_lock);
return p; return p;
@@ -647,11 +624,17 @@ void macb_initialize()
macb->rx_buffer = rt_malloc(MACB_RX_BUFFER_SIZE); macb->rx_buffer = rt_malloc(MACB_RX_BUFFER_SIZE);
macb->rx_ring = rt_malloc(MACB_RX_RING_SIZE * sizeof(struct macb_dma_desc)); macb->rx_ring = rt_malloc(MACB_RX_RING_SIZE * sizeof(struct macb_dma_desc));
macb->tx_ring = rt_malloc(MACB_TX_RING_SIZE * sizeof(struct macb_dma_desc)); macb->tx_ring = rt_malloc(MACB_TX_RING_SIZE * sizeof(struct macb_dma_desc));
EMAC_CACHE_INVALIDATE(macb->rx_ring, MACB_RX_RING_SIZE * sizeof(struct macb_dma_desc));
EMAC_CACHE_INVALIDATE(macb->tx_ring, MACB_TX_RING_SIZE * sizeof(struct macb_dma_desc));
macb->rx_buffer_dma = (unsigned long)macb->rx_buffer; macb->rx_buffer_dma = (unsigned long)macb->rx_buffer;
macb->rx_ring_dma = (unsigned long)macb->rx_ring; macb->rx_ring_dma = (unsigned long)macb->rx_ring;
macb->tx_ring_dma = (unsigned long)macb->tx_ring; macb->tx_ring_dma = (unsigned long)macb->tx_ring;
macb->rx_ring = MMU_NOCACHE_ADDR(macb->rx_ring);
macb->tx_ring = MMU_NOCACHE_ADDR(macb->tx_ring);
macb->regs = AT91SAM9260_BASE_EMAC; macb->regs = AT91SAM9260_BASE_EMAC;
macb->phy_addr = 0x00; macb->phy_addr = 0x00;

View File

@@ -141,8 +141,6 @@
#define RT_USING_LWIP #define RT_USING_LWIP
#define RT_LWIP_DNS #define RT_LWIP_DNS
#define LWIP_NETIF_LINK_CALLBACK 1
/* Trace LwIP protocol */ /* Trace LwIP protocol */
// #define RT_LWIP_DEBUG // #define RT_LWIP_DEBUG

481
libcpu/arm/at91sam926x/mmu.c Executable file → Normal file

File diff suppressed because it is too large Load Diff