Webserver+3d print

Dependents:   Nucleo

Revision:
0:8918a71cdbe9
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/cyclone_tcp/drivers/aps3_eth.c	Sat Feb 04 18:15:49 2017 +0000
@@ -0,0 +1,652 @@
+/**
+ * @file aps3_eth.c
+ * @brief Cortus APS3 Ethernet MAC controller
+ *
+ * @section License
+ *
+ * Copyright (C) 2010-2017 Oryx Embedded SARL. All rights reserved.
+ *
+ * This file is part of CycloneTCP Open.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
+ *
+ * @author Oryx Embedded SARL (www.oryx-embedded.com)
+ * @version 1.7.6
+ **/
+
+//Switch to the appropriate trace level
+#define TRACE_LEVEL NIC_TRACE_LEVEL
+
+//Dependencies
+#include <machine/sfradr.h>
+#include <machine/sfradr_eth.h>
+#include <machine/ethernet.h>
+#include <machine/ic.h>
+#undef _ETHERNET_H
+#include "core/net.h"
+#include "drivers/aps3_eth.h"
+#include "debug.h"
+
+//Transmit buffer
+#define txBuffer ((uint8_t *) SFRADR_ETH_TX_MEM_BOTTOM_AD)
+//Receive buffer
+#define rxBuffer ((uint8_t *) SFRADR_ETH_RX_MEM_BOTTOM_AD)
+
+//Transmit DMA descriptors
+#define txDmaDesc ((Aps3TxDmaDesc *) (SFRADR_ETH_TX_MEM_BOTTOM_AD + \
+   APS3_ETH_TX_BUFFER_COUNT * APS3_ETH_TX_BUFFER_SIZE))
+
+//Receive DMA descriptors
+#define rxDmaDesc ((Aps3RxDmaDesc *) (SFRADR_ETH_RX_MEM_BOTTOM_AD + \
+   APS3_ETH_RX_BUFFER_COUNT * APS3_ETH_RX_BUFFER_SIZE))
+
+//Underlying network interface
+static NetInterface *nicDriverInterface;
+
+
+/**
+ * @brief Cortus APS3 Ethernet MAC driver
+ **/
+
+const NicDriver aps3EthDriver =
+{
+   NIC_TYPE_ETHERNET,
+   ETH_MTU,
+   aps3EthInit,
+   aps3EthTick,
+   aps3EthEnableIrq,
+   aps3EthDisableIrq,
+   aps3EthEventHandler,
+   aps3EthSendPacket,
+   aps3EthSetMulticastFilter,
+   aps3EthUpdateMacConfig,
+   aps3EthWritePhyReg,
+   aps3EthReadPhyReg,
+   TRUE,
+   TRUE,
+   TRUE,
+   FALSE
+};
+
+
+/**
+ * @brief Cortus APS3 Ethernet MAC initialization
+ * @param[in] interface Underlying network interface
+ * @return Error code
+ **/
+
+error_t aps3EthInit(NetInterface *interface)
+{
+   error_t error;
+
+   //Debug message
+   TRACE_INFO("Initializing Cortus APS3 Ethernet MAC...\r\n");
+
+   //Save underlying network interface
+   nicDriverInterface = interface;
+
+   //Adjust MDC clock range
+   eth_miim->miim_clock_divider = 32;
+
+   //PHY transceiver initialization
+   error = interface->phyDriver->init(interface);
+   //Failed to initialize PHY transceiver?
+   if(error)
+      return error;
+
+   //Reset Ethernet MAC peripheral
+   eth_mac->sw_reset = 1;
+
+   //Set the MAC address
+   eth_mac->addr_low = interface->macAddr.w[0] | (interface->macAddr.w[1] << 16);
+   eth_mac->addr_high = interface->macAddr.w[2];
+
+   //Initialize hash table
+   eth_mac->hash_filter_low = 0;
+   eth_mac->hash_filter_high = 0;
+
+   //Configure the receive filter
+   eth_mac->unicast = 1;
+   eth_mac->multicast = 0;
+   eth_mac->broadcast = 1;
+   eth_mac->hash = 1;
+   eth_mac->exact_addr = 1;
+
+   //Default duplex mode
+   eth_mac->full_duplex = 0;
+
+   //Automatic padding and CRC generation
+   eth_mac->no_padding = 0;
+   eth_mac->crc_disable = 0;
+
+   //Set the maximum frame length
+   eth_mac->max_frame_size = 1518;
+
+   //Set transmit and receive thresholds
+   eth_tx->tx_threshold = 0;
+   eth_rx->rx_threshold = 0;
+
+   //Disable indefinite deferral
+   eth_mac->indefinite_deferral = 0;
+   //Number of attempts to transmit a frame before aborting
+   eth_mac->max_deferral = 15;
+
+   //Use default collision window (112 half-octets)
+   eth_mac->collision_window = 111;
+   //Maximum Number of Collisions
+   eth_mac->max_collision = 15;
+
+   //Automatic backoff on collision
+   eth_mac->no_backoff = 0;
+
+   //Use the default interframe gap (24 half-octets or 96 bits)
+   eth_mac->interframe_gap = 23;
+
+   //Initialize DMA descriptor lists
+   aps3EthInitDmaDesc(interface);
+
+   //Configure TX interrupts
+   eth_tx->tx_irq_mask = TX_IRQ_MASK_MEMORY_AVAILABLE;
+   //Configure RX interrupts
+   eth_rx->rx_irq_mask = RX_IRQ_MASK_FRAME_READY;
+
+   //Configure TX interrupt priority
+   irq[IRQ_ETH_TX].ipl = APS3_ETH_IRQ_PRIORITY;
+   //Configure RX interrupt priority
+   irq[IRQ_ETH_RX].ipl = APS3_ETH_IRQ_PRIORITY;
+
+   //Enable transmission and reception
+   eth_tx->tx_enable = 1;
+   eth_rx->rx_enable = 1;
+
+   //Accept any packets from the upper layer
+   osSetEvent(&interface->nicTxEvent);
+
+   //Successful initialization
+   return NO_ERROR;
+}
+
+
+/**
+ * @brief Initialize DMA descriptor lists
+ * @param[in] interface Underlying network interface
+ **/
+
+void aps3EthInitDmaDesc(NetInterface *interface)
+{
+   uint_t i;
+
+   //Initialize TX DMA descriptor list
+   for(i = 0; i < APS3_ETH_TX_BUFFER_COUNT; i++)
+   {
+      //Transmit buffer address
+      txDmaDesc[i].addr = (uint32_t) txBuffer + (APS3_ETH_TX_BUFFER_SIZE * i);
+      //Transmit buffer size
+      txDmaDesc[i].size = 0;
+      //Transmit status
+      txDmaDesc[i].status = 0;
+   }
+
+   //Initialize RX DMA descriptor list
+   for(i = 0; i < APS3_ETH_RX_BUFFER_COUNT; i++)
+   {
+      //Receive buffer address
+      rxDmaDesc[i].addr = (uint32_t) rxBuffer + (APS3_ETH_RX_BUFFER_SIZE * i);
+      //Receive buffer size
+      rxDmaDesc[i].size = 0;
+      //Receive status
+      rxDmaDesc[i].status = 0;
+   }
+
+   //Start location of the TX descriptor list
+   eth_tx->tx_desc_base_addr = (uint32_t) txDmaDesc;
+   //Number of TX descriptors
+   eth_tx->tx_desc_number = APS3_ETH_TX_BUFFER_COUNT - 1;
+
+   //Start location of the RX descriptor list
+   eth_rx->rx_desc_base_addr = (uint32_t) rxDmaDesc;
+   //Number of RX descriptors
+   eth_rx->rx_desc_number = APS3_ETH_RX_BUFFER_COUNT - 1;
+}
+
+
+/**
+ * @brief Cortus APS3 Ethernet MAC timer handler
+ *
+ * This routine is periodically called by the TCP/IP stack to
+ * handle periodic operations such as polling the link state
+ *
+ * @param[in] interface Underlying network interface
+ **/
+
+void aps3EthTick(NetInterface *interface)
+{
+   //Handle periodic operations
+   interface->phyDriver->tick(interface);
+}
+
+
+/**
+ * @brief Enable interrupts
+ * @param[in] interface Underlying network interface
+ **/
+
+void aps3EthEnableIrq(NetInterface *interface)
+{
+   //Enable Ethernet MAC interrupts
+   irq[IRQ_ETH_TX].ien = 1;
+   irq[IRQ_ETH_RX].ien = 1;
+   //Enable Ethernet PHY interrupts
+   interface->phyDriver->enableIrq(interface);
+}
+
+
+/**
+ * @brief Disable interrupts
+ * @param[in] interface Underlying network interface
+ **/
+
+void aps3EthDisableIrq(NetInterface *interface)
+{
+   //Disable Ethernet MAC interrupts
+   irq[IRQ_ETH_TX].ien = 0;
+   irq[IRQ_ETH_RX].ien = 0;
+   //Disable Ethernet PHY interrupts
+   interface->phyDriver->disableIrq(interface);
+}
+
+
+/**
+ * @brief Ethernet MAC transmit interrupt service routine
+ **/
+
+void aps3EthTxIrqHandler(void)
+{
+   bool_t flag;
+
+   //Enter interrupt service routine
+   osEnterIsr();
+
+   //This flag will be set if a higher priority task must be woken
+   flag = FALSE;
+
+   //Check interrupt flag
+   if(eth_tx->tx_status & TX_IRQ_MASK_MEMORY_AVAILABLE)
+   {
+      //Disable TX interrupts
+      eth_tx->tx_irq_mask = 0;
+
+      //Check whether the TX buffer is available for writing
+      if(!(eth_tx->tx_desc_status))
+      {
+         //Notify the TCP/IP stack that the transmitter is ready to send
+         flag = osSetEventFromIsr(&nicDriverInterface->nicTxEvent);
+      }
+   }
+
+   //Leave interrupt service routine
+   osExitIsr(flag);
+}
+
+
+/**
+ * @brief Ethernet MAC receive interrupt service routine
+ **/
+
+void aps3EthRxIrqHandler(void)
+{
+   bool_t flag;
+
+   //Enter interrupt service routine
+   osEnterIsr();
+
+   //This flag will be set if a higher priority task must be woken
+   flag = FALSE;
+
+   //Disable RX interrupts
+   eth_rx->rx_irq_mask = 0;
+
+   //Set event flag
+   nicDriverInterface->nicEvent = TRUE;
+   //Notify the TCP/IP stack of the event
+   flag = osSetEventFromIsr(&netEvent);
+
+   //Leave interrupt service routine
+   osExitIsr(flag);
+}
+
+
+/**
+ * @brief Cortus APS3 Ethernet MAC event handler
+ * @param[in] interface Underlying network interface
+ **/
+
+void aps3EthEventHandler(NetInterface *interface)
+{
+   error_t error;
+
+   //A packet has been received?
+   if(eth_rx->rx_status & RX_IRQ_MASK_FRAME_READY)
+   {
+      //Process all pending packets
+      do
+      {
+         //Read incoming packet
+         error = aps3EthReceivePacket(interface);
+
+         //No more data in the receive buffer?
+      } while(error != ERROR_BUFFER_EMPTY);
+   }
+
+   //Re-enable RX interrupts
+   eth_rx->rx_irq_mask = RX_IRQ_MASK_FRAME_READY;
+}
+
+
+/**
+ * @brief Send a packet
+ * @param[in] interface Underlying network interface
+ * @param[in] buffer Multi-part buffer containing the data to send
+ * @param[in] offset Offset to the first data byte
+ * @return Error code
+ **/
+
+error_t aps3EthSendPacket(NetInterface *interface,
+   const NetBuffer *buffer, size_t offset)
+{
+   uint_t i;
+   size_t length;
+
+   //Retrieve the length of the packet
+   length = netBufferGetLength(buffer) - offset;
+
+   //Check the frame length
+   if(length > APS3_ETH_TX_BUFFER_SIZE)
+   {
+      //The transmitter can accept another packet
+      osSetEvent(&interface->nicTxEvent);
+      //Report an error
+      return ERROR_INVALID_LENGTH;
+   }
+
+   //Make sure the current buffer is available for writing
+   if(eth_tx->tx_desc_status)
+   {
+      //Re-enable TX interrupts
+      eth_tx->tx_irq_mask = TX_IRQ_MASK_MEMORY_AVAILABLE;
+      //Report an error
+      return ERROR_FAILURE;
+   }
+
+   //Get the index of the current descriptor
+   i = eth_tx->tx_desc_produce;
+
+   //Copy user data to the transmit buffer
+   netBufferRead((uint8_t *) txDmaDesc[i].addr, buffer, offset, length);
+   //Write the number of bytes to send
+   txDmaDesc[i].size = length;
+
+   //Start transmission
+   eth_tx->tx_sw_done = 1;
+
+   //Check whether the next buffer is available for writing
+   if(!eth_tx->tx_desc_status)
+   {
+      //The transmitter can accept another packet
+      osSetEvent(&interface->nicTxEvent);
+   }
+   else
+   {
+      //Re-enable TX interrupts
+      eth_tx->tx_irq_mask = TX_IRQ_MASK_MEMORY_AVAILABLE;
+   }
+
+   //Data successfully written
+   return NO_ERROR;
+}
+
+
+/**
+ * @brief Receive a packet
+ * @param[in] interface Underlying network interface
+ * @return Error code
+ **/
+
+error_t aps3EthReceivePacket(NetInterface *interface)
+{
+   error_t error;
+   uint_t i;
+   size_t n;
+
+   //The current buffer is available for reading?
+   if(!(eth_rx->rx_desc_status))
+   {
+      //Point to the current descriptor
+      i = eth_rx->rx_desc_consume;
+
+      //Make sure no error occurred
+      if(!(rxDmaDesc[i].status & RX_DESC_RECEIVE_ERROR))
+      {
+         //Retrieve the length of the frame
+         n = rxDmaDesc[i].size;
+         //Limit the number of data to read
+         n = MIN(n, APS3_ETH_RX_BUFFER_SIZE);
+
+         //Pass the packet to the upper layer
+         nicProcessPacket(interface, (uint8_t *) rxDmaDesc[i].addr, n);
+
+         //Valid packet received
+         error = NO_ERROR;
+      }
+      else
+      {
+         //The received packet contains an error
+         error = ERROR_INVALID_PACKET;
+      }
+
+      //The frame has been has been processed by the software
+      //and is no longer needed
+      eth_rx->rx_sw_done = 1;
+   }
+   else
+   {
+      //No more data in the receive buffer
+      error = ERROR_BUFFER_EMPTY;
+   }
+
+   //Return status code
+   return error;
+}
+
+
+/**
+ * @brief Configure multicast MAC address filtering
+ * @param[in] interface Underlying network interface
+ * @return Error code
+ **/
+
+error_t aps3EthSetMulticastFilter(NetInterface *interface)
+{
+   uint_t i;
+   uint_t k;
+   uint32_t crc;
+   uint32_t hashTable[2];
+   MacFilterEntry *entry;
+
+   //Debug message
+   TRACE_DEBUG("Updating Cortus APS3 hash table...\r\n");
+
+   //Clear hash table
+   hashTable[0] = 0;
+   hashTable[1] = 0;
+
+   //The MAC filter table contains the multicast MAC addresses
+   //to accept when receiving an Ethernet frame
+   for(i = 0; i < MAC_MULTICAST_FILTER_SIZE; i++)
+   {
+      //Point to the current entry
+      entry = &interface->macMulticastFilter[i];
+
+      //Valid entry?
+      if(entry->refCount > 0)
+      {
+         //Compute CRC over the current MAC address
+         crc = aps3EthCalcCrc(&entry->addr, sizeof(MacAddr));
+         //Calculate the corresponding index in the table
+         k = (crc >> 23) & 0x3F;
+         //Update hash table contents
+         hashTable[k / 32] |= (1 << (k % 32));
+      }
+   }
+
+   //Disable transmission and reception
+   eth_tx->tx_enable = 0;
+   eth_rx->rx_enable = 0;
+
+   //Write the hash table
+   eth_mac->hash_filter_low = hashTable[0];
+   eth_mac->hash_filter_high = hashTable[1];
+
+   //Debug message
+   TRACE_DEBUG("  hash_filter_low = %08" PRIX32 "\r\n", hashTable[0]);
+   TRACE_DEBUG("  hash_filter_high = %08" PRIX32 "\r\n", hashTable[1]);
+
+   //Re-enable transmission and reception
+   eth_tx->tx_enable = 1;
+   eth_rx->rx_enable = 1;
+
+   //Successful processing
+   return NO_ERROR;
+}
+
+
+/**
+ * @brief Adjust MAC configuration parameters for proper operation
+ * @param[in] interface Underlying network interface
+ * @return Error code
+ **/
+
+error_t aps3EthUpdateMacConfig(NetInterface *interface)
+{
+   //Disable transmission and reception
+   eth_tx->tx_enable = 0;
+   eth_rx->rx_enable = 0;
+
+   //Half-duplex or full-duplex mode?
+   if(interface->duplexMode == NIC_FULL_DUPLEX_MODE)
+      eth_mac->full_duplex = 1;
+   else
+      eth_mac->full_duplex = 0;
+
+   //Re-enable transmission and reception
+   eth_tx->tx_enable = 1;
+   eth_rx->rx_enable = 1;
+
+   //Successful processing
+   return NO_ERROR;
+}
+
+
+/**
+ * @brief Write PHY register
+ * @param[in] phyAddr PHY address
+ * @param[in] regAddr Register address
+ * @param[in] data Register value
+ **/
+
+void aps3EthWritePhyReg(uint8_t phyAddr, uint8_t regAddr, uint16_t data)
+{
+   //Wait for the MII management module to be ready
+   while(!eth_miim->miim_status);
+
+   //PHY address
+   eth_miim->miim_phy_addr = phyAddr;
+   //Register address
+   eth_miim->miim_phy_register_addr = regAddr;
+   //Data to be written in the PHY register
+   eth_miim->miim_data = data;
+
+   //Start a write operation
+   eth_miim->miim_read_write = 0;
+   //Wait for the write to complete
+   while(!eth_miim->miim_status);
+}
+
+
+/**
+ * @brief Read PHY register
+ * @param[in] phyAddr PHY address
+ * @param[in] regAddr Register address
+ * @return Register value
+ **/
+
+uint16_t aps3EthReadPhyReg(uint8_t phyAddr, uint8_t regAddr)
+{
+   //Wait for the MII management module to be ready
+   while(!eth_miim->miim_status);
+
+   //PHY address
+   eth_miim->miim_phy_addr = phyAddr;
+   //Register address
+   eth_miim->miim_phy_register_addr = regAddr;
+
+   //Start a read operation
+   eth_miim->miim_read_write = 1;
+   //Wait for the read to complete
+   while(!eth_miim->miim_status);
+
+   //Return PHY register contents
+   return eth_miim->miim_data;
+}
+
+
+/**
+ * @brief CRC calculation
+ * @param[in] data Pointer to the data over which to calculate the CRC
+ * @param[in] length Number of bytes to process
+ * @return Resulting CRC value
+ **/
+
+uint32_t aps3EthCalcCrc(const void *data, size_t length)
+{
+   uint_t i;
+   uint_t j;
+
+   //Point to the data over which to calculate the CRC
+   const uint8_t *p = (uint8_t *) data;
+   //CRC preset value
+   uint32_t crc = 0xFFFFFFFF;
+
+   //Loop through data
+   for(i = 0; i < length; i++)
+   {
+      //Update CRC value
+      crc ^= p[i];
+
+      //The message is processed bit by bit
+      for(j = 0; j < 8; j++)
+      {
+         //Update CRC value
+         if(crc & 0x00000001)
+            crc = (crc >> 1) ^ 0xEDB88320;
+         else
+            crc = crc >> 1;
+      }
+   }
+
+   //Return CRC value
+   return ~crc;
+}
+