Source to ./dev_c3600_eth.c


Enter a symbol's name here to quickly find it.

/*  
 * Cisco C3600 simulation platform.
 * Copyright (c) 2006 Christophe Fillot ([email protected])
 *
 * Ethernet Network Modules.
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include <unistd.h>
#include <time.h>
#include <errno.h>
#include <assert.h>

#include "utils.h"
#include "net.h"
#include "net_io.h"
#include "ptask.h"
#include "dev_am79c971.h"
#include "dev_nm_16esw.h"
#include "dev_c3600.h"
#include "dev_c3600_bay.h"

/* Multi-Ethernet NM with Am79c971 chips */
struct nm_eth_data {
   u_int nr_port;
   struct am79c971_data *port[8];
};

/*
 * dev_c3600_nm_eth_init()
 *
 * Add an Ethernet Network Module into specified slot.
 */
static int dev_c3600_nm_eth_init(vm_instance_t *vm,struct cisco_card *card,
                                 int nr_port,int interface_type,
                                 const struct cisco_eeprom *eeprom)
{
   struct nm_bay_info *bay_info;
   struct nm_eth_data *data;
   u_int slot = card->slot_id;
   u_int chassis_id;
   int i;

   /* Allocate the private data structure */
   if (!(data = malloc(sizeof(*data)))) {
      vm_error(vm,"%s: out of memory.\n",card->dev_name);
      return(-1);
   }

   memset(data,0,sizeof(*data));
   data->nr_port = nr_port;

   /* Set the PCI bus */
   card->pci_bus = vm->slots_pci_bus[slot];

   /* Set the EEPROM */
   cisco_card_set_eeprom(vm,card,eeprom);
   c3600_set_slot_eeprom(VM_C3600(vm),slot,&card->eeprom);

   /* Get PCI bus info about this bay */
   chassis_id = c3600_chassis_get_id(VM_C3600(vm));
   bay_info = c3600_nm_get_bay_info(chassis_id,slot);
      
   if (!bay_info) {
      vm_error(vm,"unable to get info for NM bay %u\n",slot);
      return(-1);
   }

   /* Create the AMD Am971c971 chip(s) */
   for(i=0;i<data->nr_port;i++) {
      data->port[i] = dev_am79c971_init(vm,card->dev_name,interface_type,
                                        card->pci_bus,bay_info->pci_device+i,
                                        c3600_net_irq_for_slot_port(slot,i));
   }

   /* Store device info into the router structure */
   card->drv_info = data;
   return(0);
}

/* Remove an Ethernet NM from the specified slot */
static int dev_c3600_nm_eth_shutdown(vm_instance_t *vm,struct cisco_card *card)
{
   struct nm_eth_data *data = card->drv_info;
   int i;

   /* Remove the NM EEPROM */
   cisco_card_unset_eeprom(card);
   c3600_set_slot_eeprom(VM_C3600(vm),card->slot_id,NULL);

   /* Remove the AMD Am79c971 chips */
   for(i=0;i<data->nr_port;i++)
      dev_am79c971_remove(data->port[i]);

   free(data);
   return(0);
}

/* Bind a Network IO descriptor */
static int dev_c3600_nm_eth_set_nio(vm_instance_t *vm,struct cisco_card *card,
                                    u_int port_id,netio_desc_t *nio)
{
   struct nm_eth_data *d = card->drv_info;

   if (!d || (port_id >= d->nr_port))
      return(-1);

   dev_am79c971_set_nio(d->port[port_id],nio);
   return(0);
}

/* Unbind a Network IO descriptor */
static int dev_c3600_nm_eth_unset_nio(vm_instance_t *vm,
                                      struct cisco_card *card,
                                      u_int port_id)
{
   struct nm_eth_data *d = card->drv_info;

   if (!d || (port_id >= d->nr_port))
      return(-1);

   dev_am79c971_unset_nio(d->port[port_id]);
   return(0);
}

/* ====================================================================== */
/* NM-1E                                                                  */
/* ====================================================================== */

/*
 * dev_c3600_nm_1e_init()
 *
 * Add a NM-1E Network Module into specified slot.
 */
static int dev_c3600_nm_1e_init(vm_instance_t *vm,struct cisco_card *card)
{
   return(dev_c3600_nm_eth_init(vm,card,1,AM79C971_TYPE_10BASE_T,
                                cisco_eeprom_find_nm("NM-1E")));
}

/* ====================================================================== */
/* NM-4E                                                                  */
/* ====================================================================== */

/*
 * dev_c3600_nm_4e_init()
 *
 * Add a NM-4E Network Module into specified slot.
 */
static int dev_c3600_nm_4e_init(vm_instance_t *vm,struct cisco_card *card)
{
   return(dev_c3600_nm_eth_init(vm,card,4,AM79C971_TYPE_10BASE_T,
                                cisco_eeprom_find_nm("NM-4E")));
}

/* ====================================================================== */
/* NM-1FE-TX                                                              */
/* ====================================================================== */

/*
 * dev_c3600_nm_1fe_tx_init()
 *
 * Add a NM-1FE-TX Network Module into specified slot.
 */
static int dev_c3600_nm_1fe_tx_init(vm_instance_t *vm,struct cisco_card *card)
{
   return(dev_c3600_nm_eth_init(vm,card,1,AM79C971_TYPE_100BASE_TX,
                                cisco_eeprom_find_nm("NM-1FE-TX")));
}

/* ====================================================================== */
/* ESW cards, common part                                                 */
/* ====================================================================== */

/* Remove a ESW NM from the specified slot */
static int dev_c3600_nm_esw_shutdown(vm_instance_t *vm,struct cisco_card *card)
{
   struct esw_data *data = card->drv_info;

   /* Remove the NM EEPROM */
   cisco_card_unset_eeprom(card);
   c3600_set_slot_eeprom(VM_C3600(vm),card->slot_id,NULL);

   /* Remove the BCM5600 chip */
   esw_remove(data);
   return(0);
}

/* Bind a Network IO descriptor */
static int 
dev_c3600_nm_esw_set_nio(vm_instance_t *vm,struct cisco_card *card,
                         u_int port_id,netio_desc_t *nio)
{
   struct esw_data *d = card->drv_info;
   esw_set_nio(d,port_id,nio);
   return(0);
}

/* Unbind a Network IO descriptor */
static int dev_c3600_nm_esw_unset_nio(vm_instance_t *vm,
                                      struct cisco_card *card,
                                      u_int port_id)
{
   struct esw_data *d = card->drv_info;
   esw_unset_nio(d,port_id);
   return(0);
}

/* Show debug info */
static int 
dev_c3600_nm_esw_show_info(vm_instance_t *vm,struct cisco_card *card)
{
   struct esw_data *d = card->drv_info;
   esw_show_info(d);
   return(0);
}

/* ====================================================================== */
/* NM-16ESW                                                               */
/* ====================================================================== */

/* Add a NM-16ESW */
static int dev_c3600_nm_16esw_init(vm_instance_t *vm,struct cisco_card *card)
{
   struct nm_bay_info *bay_info;
   struct esw_data *data;
   u_int slot = card->slot_id;
   u_int chassis_id;

   /* Set the PCI bus */
   card->pci_bus = vm->slots_pci_bus[slot];

   /* Set the EEPROM */
   cisco_card_set_eeprom(vm,card,cisco_eeprom_find_nm("NM-16ESW"));
   esw_burn_mac_addr(vm,slot,&card->eeprom);
   c3600_set_slot_eeprom(VM_C3600(vm),slot,&card->eeprom);

   /* Get PCI bus info about this bay */
   chassis_id = c3600_chassis_get_id(VM_C3600(vm));
   bay_info = c3600_nm_get_bay_info(chassis_id,slot);
      
   if (!bay_info) {
      vm_error(vm,"unable to get info for NM bay %u\n",slot);
      return(-1);
   }

   /* Create the device */
   data = dev_nm_16esw_init(vm,card->dev_name,slot,
                            card->pci_bus,bay_info->pci_device,
                            c3600_net_irq_for_slot_port(slot,0));

   /* Store device info into the router structure */
   card->drv_info = data;
   return(0);
}

/* ====================================================================== */
/* NMD-36ESW                                                              */
/* ====================================================================== */

/* Add a NMD-36ESW */
static int dev_c3600_nmd_36esw_init(vm_instance_t *vm,struct cisco_card *card)
{
   struct nm_bay_info *bay_info;
   struct esw_data *data;
   u_int slot = card->slot_id;
   u_int chassis_id;

   /* Set the PCI bus */
   card->pci_bus = vm->slots_pci_bus[slot];

   /* Set the EEPROM */
   cisco_card_set_eeprom(vm,card,cisco_eeprom_find_nm("NMD-36ESW"));
   esw_burn_mac_addr(vm,slot,&card->eeprom);
   c3600_set_slot_eeprom(VM_C3600(vm),slot,&card->eeprom);

   /* Get PCI bus info about this bay */
   chassis_id = c3600_chassis_get_id(VM_C3600(vm));
   bay_info = c3600_nm_get_bay_info(chassis_id,slot);
      
   if (!bay_info) {
      vm_error(vm,"unable to get info for NM bay %u\n",slot);
      return(-1);
   }

   /* Create the device */
   data = dev_nmd_36esw_init(vm,card->dev_name,slot,
                             card->pci_bus,bay_info->pci_device,
                             c3600_net_irq_for_slot_port(slot,0));

   /* Store device info into the router structure */
   card->drv_info = data;
   return(0);
}

/* ====================================================================== */
/* Leopard-2FE                                                            */
/* ====================================================================== */

/* 
 * Leopard-2FE: 2 FastEthernet ports on C3660 motherboard.
 *
 * Leopard-2FE is the FRU/Product Number displayed by "show diag".
 */
static m_uint16_t eeprom_c3600_leopard_2fe_data[] = {
   0x04FF, 0xC18B, 0x4A41, 0x4230, 0x3530, 0x3330, 0x3454, 0x3809,
   0x3440, 0x00B3, 0xC046, 0x0320, 0x0012, 0x8104, 0x4241, 0x3085,
   0x1C0C, 0xA202, 0x80FF, 0xFFFF, 0xFFC4, 0x08FF, 0xFFFF, 0xFFFF,
   0xFFFF, 0xFFA1, 0xFFFF, 0xFFFF, 0x03FF, 0x04FF, 0xC508, 0xFFFF,
   0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF,
   0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF,
   0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF,
   0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFF00,
};

static const struct cisco_eeprom eeprom_c3600_leopard_2fe = {
   "Leopard-2FE", (m_uint16_t *)eeprom_c3600_leopard_2fe_data, 
   sizeof(eeprom_c3600_leopard_2fe_data)/2,
};

/*
 * dev_c3600_leopard_2fe_init()
 *
 * Add Leopard-2FE (only Cisco 3660, in slot 0).
 */
static int 
dev_c3600_leopard_2fe_init(vm_instance_t *vm,struct cisco_card *card)
{
   if (card->slot_id != 0) {
      vm_error(vm,"dev_c3600_leopard_2fe_init: bad slot %u specified.\n",
               card->slot_id);
      return(-1);
   }

   return(dev_c3600_nm_eth_init(vm,card,2,AM79C971_TYPE_100BASE_TX,
                                &eeprom_c3600_leopard_2fe));
}

/* ====================================================================== */

/* NM-1FE-TX driver */
struct cisco_card_driver dev_c3600_nm_1fe_tx_driver = {
   "NM-1FE-TX", 1, 0,
   dev_c3600_nm_1fe_tx_init, 
   dev_c3600_nm_eth_shutdown,
   NULL,
   dev_c3600_nm_eth_set_nio,
   dev_c3600_nm_eth_unset_nio,
   NULL,
};

/* NM-1E driver */
struct cisco_card_driver dev_c3600_nm_1e_driver = {
   "NM-1E", 1, 0,
   dev_c3600_nm_1e_init, 
   dev_c3600_nm_eth_shutdown,
   NULL,
   dev_c3600_nm_eth_set_nio,
   dev_c3600_nm_eth_unset_nio,
   NULL,
};

/* NM-4E driver */
struct cisco_card_driver dev_c3600_nm_4e_driver = {
   "NM-4E", 1, 0,
   dev_c3600_nm_4e_init, 
   dev_c3600_nm_eth_shutdown,
   NULL,
   dev_c3600_nm_eth_set_nio,
   dev_c3600_nm_eth_unset_nio,
   NULL,
};

/* NM-16ESW driver */
struct cisco_card_driver dev_c3600_nm_16esw_driver = {
   "NM-16ESW", 1, 0,
   dev_c3600_nm_16esw_init, 
   dev_c3600_nm_esw_shutdown,
   NULL,
   dev_c3600_nm_esw_set_nio,
   dev_c3600_nm_esw_unset_nio,
   dev_c3600_nm_esw_show_info,
};

/* NMD-36ESW driver */
struct cisco_card_driver dev_c3600_nmd_36esw_driver = {
   "NMD-36ESW", 1, 0,
   dev_c3600_nmd_36esw_init, 
   dev_c3600_nm_esw_shutdown,
   NULL,
   dev_c3600_nm_esw_set_nio,
   dev_c3600_nm_esw_unset_nio,
   dev_c3600_nm_esw_show_info,
};

/* Leopard-2FE driver */
struct cisco_card_driver dev_c3600_leopard_2fe_driver = {
   "Leopard-2FE", 1, 0,
   dev_c3600_leopard_2fe_init, 
   dev_c3600_nm_eth_shutdown,
   NULL,
   dev_c3600_nm_eth_set_nio,
   dev_c3600_nm_eth_unset_nio,
   NULL,
};