| /** |
| * @file IxEthDBDBMem.c |
| * |
| * @brief Memory handling routines for the MAC address database |
| * |
| * @par |
| * IXP400 SW Release version 2.0 |
| * |
| * -- Copyright Notice -- |
| * |
| * @par |
| * Copyright 2001-2005, Intel Corporation. |
| * All rights reserved. |
| * |
| * @par |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of the Intel Corporation nor the names of its contributors |
| * may be used to endorse or promote products derived from this software |
| * without specific prior written permission. |
| * |
| * @par |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE |
| * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS |
| * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) |
| * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
| * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY |
| * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF |
| * SUCH DAMAGE. |
| * |
| * @par |
| * -- End of Copyright Notice -- |
| */ |
| |
| |
| #include "IxEthDB_p.h" |
| |
| IX_ETH_DB_PRIVATE HashNode *nodePool = NULL; |
| IX_ETH_DB_PRIVATE MacDescriptor *macPool = NULL; |
| IX_ETH_DB_PRIVATE MacTreeNode *treePool = NULL; |
| |
| IX_ETH_DB_PRIVATE HashNode nodePoolArea[NODE_POOL_SIZE]; |
| IX_ETH_DB_PRIVATE MacDescriptor macPoolArea[MAC_POOL_SIZE]; |
| IX_ETH_DB_PRIVATE MacTreeNode treePoolArea[TREE_POOL_SIZE]; |
| |
| IX_ETH_DB_PRIVATE IxOsalMutex nodePoolLock; |
| IX_ETH_DB_PRIVATE IxOsalMutex macPoolLock; |
| IX_ETH_DB_PRIVATE IxOsalMutex treePoolLock; |
| |
| #define LOCK_NODE_POOL { ixOsalMutexLock(&nodePoolLock, IX_OSAL_WAIT_FOREVER); } |
| #define UNLOCK_NODE_POOL { ixOsalMutexUnlock(&nodePoolLock); } |
| |
| #define LOCK_MAC_POOL { ixOsalMutexLock(&macPoolLock, IX_OSAL_WAIT_FOREVER); } |
| #define UNLOCK_MAC_POOL { ixOsalMutexUnlock(&macPoolLock); } |
| |
| #define LOCK_TREE_POOL { ixOsalMutexLock(&treePoolLock, IX_OSAL_WAIT_FOREVER); } |
| #define UNLOCK_TREE_POOL { ixOsalMutexUnlock(&treePoolLock); } |
| |
| /* private function prototypes */ |
| IX_ETH_DB_PRIVATE MacDescriptor* ixEthDBPoolAllocMacDescriptor(void); |
| IX_ETH_DB_PRIVATE void ixEthDBPoolFreeMacDescriptor(MacDescriptor *macDescriptor); |
| |
| /** |
| * @addtogroup EthMemoryManagement |
| * |
| * @{ |
| */ |
| |
| /** |
| * @brief initializes the memory pools used by the ethernet database component |
| * |
| * Initializes the hash table node, mac descriptor and mac tree node pools. |
| * Called at initialization time by @ref ixEthDBInit(). |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| void ixEthDBInitMemoryPools(void) |
| { |
| int local_index; |
| |
| /* HashNode pool */ |
| ixOsalMutexInit(&nodePoolLock); |
| |
| for (local_index = 0 ; local_index < NODE_POOL_SIZE ; local_index++) |
| { |
| HashNode *freeNode = &nodePoolArea[local_index]; |
| |
| freeNode->nextFree = nodePool; |
| nodePool = freeNode; |
| } |
| |
| /* MacDescriptor pool */ |
| ixOsalMutexInit(&macPoolLock); |
| |
| for (local_index = 0 ; local_index < MAC_POOL_SIZE ; local_index++) |
| { |
| MacDescriptor *freeDescriptor = &macPoolArea[local_index]; |
| |
| freeDescriptor->nextFree = macPool; |
| macPool = freeDescriptor; |
| } |
| |
| /* MacTreeNode pool */ |
| ixOsalMutexInit(&treePoolLock); |
| |
| for (local_index = 0 ; local_index < TREE_POOL_SIZE ; local_index++) |
| { |
| MacTreeNode *freeNode = &treePoolArea[local_index]; |
| |
| freeNode->nextFree = treePool; |
| treePool = freeNode; |
| } |
| } |
| |
| /** |
| * @brief allocates a hash node from the pool |
| * |
| * Allocates a hash node and resets its value. |
| * |
| * @return the allocated hash node or NULL if the pool is empty |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| HashNode* ixEthDBAllocHashNode(void) |
| { |
| HashNode *allocatedNode = NULL; |
| |
| if (nodePool != NULL) |
| { |
| LOCK_NODE_POOL; |
| |
| allocatedNode = nodePool; |
| nodePool = nodePool->nextFree; |
| |
| UNLOCK_NODE_POOL; |
| |
| memset(allocatedNode, 0, sizeof(HashNode)); |
| } |
| |
| return allocatedNode; |
| } |
| |
| /** |
| * @brief frees a hash node into the pool |
| * |
| * @param hashNode node to be freed |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| void ixEthDBFreeHashNode(HashNode *hashNode) |
| { |
| if (hashNode != NULL) |
| { |
| LOCK_NODE_POOL; |
| |
| hashNode->nextFree = nodePool; |
| nodePool = hashNode; |
| |
| UNLOCK_NODE_POOL; |
| } |
| } |
| |
| /** |
| * @brief allocates a mac descriptor from the pool |
| * |
| * Allocates a mac descriptor and resets its value. |
| * This function is not used directly, instead @ref ixEthDBAllocMacDescriptor() |
| * is used, which keeps track of the pointer reference count. |
| * |
| * @see ixEthDBAllocMacDescriptor() |
| * |
| * @warning this function is not used directly by any other function |
| * apart from ixEthDBAllocMacDescriptor() |
| * |
| * @return the allocated mac descriptor or NULL if the pool is empty |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PRIVATE |
| MacDescriptor* ixEthDBPoolAllocMacDescriptor(void) |
| { |
| MacDescriptor *allocatedDescriptor = NULL; |
| |
| if (macPool != NULL) |
| { |
| LOCK_MAC_POOL; |
| |
| allocatedDescriptor = macPool; |
| macPool = macPool->nextFree; |
| |
| UNLOCK_MAC_POOL; |
| |
| memset(allocatedDescriptor, 0, sizeof(MacDescriptor)); |
| } |
| |
| return allocatedDescriptor; |
| } |
| |
| /** |
| * @brief allocates and initializes a mac descriptor smart pointer |
| * |
| * Uses @ref ixEthDBPoolAllocMacDescriptor() to allocate a mac descriptor |
| * from the pool and initializes its reference count. |
| * |
| * @see ixEthDBPoolAllocMacDescriptor() |
| * |
| * @return the allocated mac descriptor or NULL if the pool is empty |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| MacDescriptor* ixEthDBAllocMacDescriptor(void) |
| { |
| MacDescriptor *allocatedDescriptor = ixEthDBPoolAllocMacDescriptor(); |
| |
| if (allocatedDescriptor != NULL) |
| { |
| LOCK_MAC_POOL; |
| |
| allocatedDescriptor->refCount++; |
| |
| UNLOCK_MAC_POOL; |
| } |
| |
| return allocatedDescriptor; |
| } |
| |
| /** |
| * @brief frees a mac descriptor back into the pool |
| * |
| * @param macDescriptor mac descriptor to be freed |
| * |
| * @warning this function is not to be called by anyone but |
| * ixEthDBFreeMacDescriptor() |
| * |
| * @see ixEthDBFreeMacDescriptor() |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PRIVATE |
| void ixEthDBPoolFreeMacDescriptor(MacDescriptor *macDescriptor) |
| { |
| LOCK_MAC_POOL; |
| |
| macDescriptor->nextFree = macPool; |
| macPool = macDescriptor; |
| |
| UNLOCK_MAC_POOL; |
| } |
| |
| /** |
| * @brief frees or reduces the usage count of a mac descriptor smart pointer |
| * |
| * If the reference count reaches 0 (structure is no longer used anywhere) |
| * then the descriptor is freed back into the pool using ixEthDBPoolFreeMacDescriptor(). |
| * |
| * @see ixEthDBPoolFreeMacDescriptor() |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| void ixEthDBFreeMacDescriptor(MacDescriptor *macDescriptor) |
| { |
| if (macDescriptor != NULL) |
| { |
| LOCK_MAC_POOL; |
| |
| if (macDescriptor->refCount > 0) |
| { |
| macDescriptor->refCount--; |
| |
| if (macDescriptor->refCount == 0) |
| { |
| UNLOCK_MAC_POOL; |
| |
| ixEthDBPoolFreeMacDescriptor(macDescriptor); |
| } |
| else |
| { |
| UNLOCK_MAC_POOL; |
| } |
| } |
| else |
| { |
| UNLOCK_MAC_POOL; |
| } |
| } |
| } |
| |
| /** |
| * @brief clones a mac descriptor smart pointer |
| * |
| * @param macDescriptor mac descriptor to clone |
| * |
| * Increments the usage count of the smart pointer |
| * |
| * @returns the cloned smart pointer |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| MacDescriptor* ixEthDBCloneMacDescriptor(MacDescriptor *macDescriptor) |
| { |
| LOCK_MAC_POOL; |
| |
| if (macDescriptor->refCount == 0) |
| { |
| UNLOCK_MAC_POOL; |
| |
| return NULL; |
| } |
| |
| macDescriptor->refCount++; |
| |
| UNLOCK_MAC_POOL; |
| |
| return macDescriptor; |
| } |
| |
| /** |
| * @brief allocates a mac tree node from the pool |
| * |
| * Allocates and initializes a mac tree node from the pool. |
| * |
| * @return the allocated mac tree node or NULL if the pool is empty |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| MacTreeNode* ixEthDBAllocMacTreeNode(void) |
| { |
| MacTreeNode *allocatedNode = NULL; |
| |
| if (treePool != NULL) |
| { |
| LOCK_TREE_POOL; |
| |
| allocatedNode = treePool; |
| treePool = treePool->nextFree; |
| |
| UNLOCK_TREE_POOL; |
| |
| memset(allocatedNode, 0, sizeof(MacTreeNode)); |
| } |
| |
| return allocatedNode; |
| } |
| |
| /** |
| * @brief frees a mac tree node back into the pool |
| * |
| * @param macNode mac tree node to be freed |
| * |
| * @warning not to be used except from ixEthDBFreeMacTreeNode(). |
| * |
| * @see ixEthDBFreeMacTreeNode() |
| * |
| * @internal |
| */ |
| void ixEthDBPoolFreeMacTreeNode(MacTreeNode *macNode) |
| { |
| if (macNode != NULL) |
| { |
| LOCK_TREE_POOL; |
| |
| macNode->nextFree = treePool; |
| treePool = macNode; |
| |
| UNLOCK_TREE_POOL; |
| } |
| } |
| |
| /** |
| * @brief frees or reduces the usage count of a mac tree node smart pointer |
| * |
| * @param macNode mac tree node to free |
| * |
| * Reduces the usage count of the given mac node. If the usage count |
| * reaches 0 the node is freed back into the pool using ixEthDBPoolFreeMacTreeNode() |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| void ixEthDBFreeMacTreeNode(MacTreeNode *macNode) |
| { |
| if (macNode->descriptor != NULL) |
| { |
| ixEthDBFreeMacDescriptor(macNode->descriptor); |
| } |
| |
| if (macNode->left != NULL) |
| { |
| ixEthDBFreeMacTreeNode(macNode->left); |
| } |
| |
| if (macNode->right != NULL) |
| { |
| ixEthDBFreeMacTreeNode(macNode->right); |
| } |
| |
| ixEthDBPoolFreeMacTreeNode(macNode); |
| } |
| |
| /** |
| * @brief clones a mac tree node |
| * |
| * @param macNode mac tree node to be cloned |
| * |
| * Increments the usage count of the node, <i>its associated descriptor |
| * and <b>recursively</b> of all its child nodes</i>. |
| * |
| * @warning this function is recursive and clones whole trees/subtrees, use only for |
| * root nodes |
| * |
| * @internal |
| */ |
| IX_ETH_DB_PUBLIC |
| MacTreeNode* ixEthDBCloneMacTreeNode(MacTreeNode *macNode) |
| { |
| if (macNode != NULL) |
| { |
| MacTreeNode *clonedMacNode = ixEthDBAllocMacTreeNode(); |
| |
| if (clonedMacNode != NULL) |
| { |
| if (macNode->right != NULL) |
| { |
| clonedMacNode->right = ixEthDBCloneMacTreeNode(macNode->right); |
| } |
| |
| if (macNode->left != NULL) |
| { |
| clonedMacNode->left = ixEthDBCloneMacTreeNode(macNode->left); |
| } |
| |
| if (macNode->descriptor != NULL) |
| { |
| clonedMacNode->descriptor = ixEthDBCloneMacDescriptor(macNode->descriptor); |
| } |
| } |
| |
| return clonedMacNode; |
| } |
| else |
| { |
| return NULL; |
| } |
| } |
| |
| #ifndef NDEBUG |
| /* Debug statistical functions for memory usage */ |
| |
| extern HashTable dbHashtable; |
| int ixEthDBNumHashElements(void); |
| |
| int ixEthDBNumHashElements(void) |
| { |
| UINT32 bucketIndex; |
| int numElements = 0; |
| HashTable *hashTable = &dbHashtable; |
| |
| for (bucketIndex = 0 ; bucketIndex < hashTable->numBuckets ; bucketIndex++) |
| { |
| if (hashTable->hashBuckets[bucketIndex] != NULL) |
| { |
| HashNode *node = hashTable->hashBuckets[bucketIndex]; |
| |
| while (node != NULL) |
| { |
| numElements++; |
| |
| node = node->next; |
| } |
| } |
| } |
| |
| return numElements; |
| } |
| |
| UINT32 ixEthDBSearchTreeUsageGet(MacTreeNode *tree) |
| { |
| if (tree == NULL) |
| { |
| return 0; |
| } |
| else |
| { |
| return 1 /* this node */ + ixEthDBSearchTreeUsageGet(tree->left) + ixEthDBSearchTreeUsageGet(tree->right); |
| } |
| } |
| |
| int ixEthDBShowMemoryStatus(void) |
| { |
| MacDescriptor *mac; |
| MacTreeNode *tree; |
| HashNode *node; |
| |
| int macCounter = 0; |
| int treeCounter = 0; |
| int nodeCounter = 0; |
| |
| int totalTreeUsage = 0; |
| int totalDescriptorUsage = 0; |
| int totalCloneDescriptorUsage = 0; |
| int totalNodeUsage = 0; |
| |
| UINT32 portIndex; |
| |
| LOCK_NODE_POOL; |
| LOCK_MAC_POOL; |
| LOCK_TREE_POOL; |
| |
| mac = macPool; |
| tree = treePool; |
| node = nodePool; |
| |
| while (mac != NULL) |
| { |
| macCounter++; |
| |
| mac = mac->nextFree; |
| |
| if (macCounter > MAC_POOL_SIZE) |
| { |
| break; |
| } |
| } |
| |
| while (tree != NULL) |
| { |
| treeCounter++; |
| |
| tree = tree->nextFree; |
| |
| if (treeCounter > TREE_POOL_SIZE) |
| { |
| break; |
| } |
| } |
| |
| while (node != NULL) |
| { |
| nodeCounter++; |
| |
| node = node->nextFree; |
| |
| if (nodeCounter > NODE_POOL_SIZE) |
| { |
| break; |
| } |
| } |
| |
| for (portIndex = 0 ; portIndex < IX_ETH_DB_NUMBER_OF_PORTS ; portIndex++) |
| { |
| int treeUsage = ixEthDBSearchTreeUsageGet(ixEthDBPortInfo[portIndex].updateMethod.searchTree); |
| |
| totalTreeUsage += treeUsage; |
| totalCloneDescriptorUsage += treeUsage; /* each tree node contains a descriptor */ |
| } |
| |
| totalNodeUsage = ixEthDBNumHashElements(); |
| totalDescriptorUsage += totalNodeUsage; /* each hash table entry contains a descriptor */ |
| |
| UNLOCK_NODE_POOL; |
| UNLOCK_MAC_POOL; |
| UNLOCK_TREE_POOL; |
| |
| printf("Ethernet database memory usage stats:\n\n"); |
| |
| if (macCounter <= MAC_POOL_SIZE) |
| { |
| printf("\tMAC descriptor pool : %d free out of %d entries (%d%%)\n", macCounter, MAC_POOL_SIZE, macCounter * 100 / MAC_POOL_SIZE); |
| } |
| else |
| { |
| printf("\tMAC descriptor pool : invalid state (ring within the pool), normally %d entries\n", MAC_POOL_SIZE); |
| } |
| |
| if (treeCounter <= TREE_POOL_SIZE) |
| { |
| printf("\tTree node pool : %d free out of %d entries (%d%%)\n", treeCounter, TREE_POOL_SIZE, treeCounter * 100 / TREE_POOL_SIZE); |
| } |
| else |
| { |
| printf("\tTREE descriptor pool : invalid state (ring within the pool), normally %d entries\n", TREE_POOL_SIZE); |
| } |
| |
| if (nodeCounter <= NODE_POOL_SIZE) |
| { |
| printf("\tHash node pool : %d free out of %d entries (%d%%)\n", nodeCounter, NODE_POOL_SIZE, nodeCounter * 100 / NODE_POOL_SIZE); |
| } |
| else |
| { |
| printf("\tNODE descriptor pool : invalid state (ring within the pool), normally %d entries\n", NODE_POOL_SIZE); |
| } |
| |
| printf("\n"); |
| printf("\tMAC descriptor usage : %d entries, %d cloned\n", totalDescriptorUsage, totalCloneDescriptorUsage); |
| printf("\tTree node usage : %d entries\n", totalTreeUsage); |
| printf("\tHash node usage : %d entries\n", totalNodeUsage); |
| printf("\n"); |
| |
| /* search for duplicate nodes in the mac pool */ |
| { |
| MacDescriptor *reference = macPool; |
| |
| while (reference != NULL) |
| { |
| MacDescriptor *comparison = reference->nextFree; |
| |
| while (comparison != NULL) |
| { |
| if (reference == comparison) |
| { |
| printf("Warning: reached a duplicate (%p), invalid MAC pool state\n", reference); |
| |
| return 1; |
| } |
| |
| comparison = comparison->nextFree; |
| } |
| |
| reference = reference->nextFree; |
| } |
| } |
| |
| printf("No duplicates found in the MAC pool (sanity check ok)\n"); |
| |
| return 0; |
| } |
| |
| #endif /* NDEBUG */ |
| |
| /** |
| * @} EthMemoryManagement |
| */ |