162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Driver for the Solos PCI ADSL2+ card, designed to support Linux by 462306a36Sopenharmony_ci * Traverse Technologies -- https://www.traverse.com.au/ 562306a36Sopenharmony_ci * Xrio Limited -- http://www.xrio.com/ 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Copyright © 2008 Traverse Technologies 862306a36Sopenharmony_ci * Copyright © 2008 Intel Corporation 962306a36Sopenharmony_ci * 1062306a36Sopenharmony_ci * Authors: Nathan Williams <nathan@traverse.com.au> 1162306a36Sopenharmony_ci * David Woodhouse <dwmw2@infradead.org> 1262306a36Sopenharmony_ci * Treker Chen <treker@xrio.com> 1362306a36Sopenharmony_ci */ 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#define DEBUG 1662306a36Sopenharmony_ci#define VERBOSE_DEBUG 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#include <linux/interrupt.h> 1962306a36Sopenharmony_ci#include <linux/module.h> 2062306a36Sopenharmony_ci#include <linux/kernel.h> 2162306a36Sopenharmony_ci#include <linux/errno.h> 2262306a36Sopenharmony_ci#include <linux/ioport.h> 2362306a36Sopenharmony_ci#include <linux/types.h> 2462306a36Sopenharmony_ci#include <linux/pci.h> 2562306a36Sopenharmony_ci#include <linux/atm.h> 2662306a36Sopenharmony_ci#include <linux/atmdev.h> 2762306a36Sopenharmony_ci#include <linux/skbuff.h> 2862306a36Sopenharmony_ci#include <linux/sysfs.h> 2962306a36Sopenharmony_ci#include <linux/device.h> 3062306a36Sopenharmony_ci#include <linux/kobject.h> 3162306a36Sopenharmony_ci#include <linux/firmware.h> 3262306a36Sopenharmony_ci#include <linux/ctype.h> 3362306a36Sopenharmony_ci#include <linux/swab.h> 3462306a36Sopenharmony_ci#include <linux/slab.h> 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci#define VERSION "1.04" 3762306a36Sopenharmony_ci#define DRIVER_VERSION 0x01 3862306a36Sopenharmony_ci#define PTAG "solos-pci" 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci#define CONFIG_RAM_SIZE 128 4162306a36Sopenharmony_ci#define FLAGS_ADDR 0x7C 4262306a36Sopenharmony_ci#define IRQ_EN_ADDR 0x78 4362306a36Sopenharmony_ci#define FPGA_VER 0x74 4462306a36Sopenharmony_ci#define IRQ_CLEAR 0x70 4562306a36Sopenharmony_ci#define WRITE_FLASH 0x6C 4662306a36Sopenharmony_ci#define PORTS 0x68 4762306a36Sopenharmony_ci#define FLASH_BLOCK 0x64 4862306a36Sopenharmony_ci#define FLASH_BUSY 0x60 4962306a36Sopenharmony_ci#define FPGA_MODE 0x5C 5062306a36Sopenharmony_ci#define FLASH_MODE 0x58 5162306a36Sopenharmony_ci#define GPIO_STATUS 0x54 5262306a36Sopenharmony_ci#define DRIVER_VER 0x50 5362306a36Sopenharmony_ci#define TX_DMA_ADDR(port) (0x40 + (4 * (port))) 5462306a36Sopenharmony_ci#define RX_DMA_ADDR(port) (0x30 + (4 * (port))) 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci#define DATA_RAM_SIZE 32768 5762306a36Sopenharmony_ci#define BUF_SIZE 2048 5862306a36Sopenharmony_ci#define OLD_BUF_SIZE 4096 /* For FPGA versions <= 2*/ 5962306a36Sopenharmony_ci/* Old boards use ATMEL AD45DB161D flash */ 6062306a36Sopenharmony_ci#define ATMEL_FPGA_PAGE 528 /* FPGA flash page size*/ 6162306a36Sopenharmony_ci#define ATMEL_SOLOS_PAGE 512 /* Solos flash page size*/ 6262306a36Sopenharmony_ci#define ATMEL_FPGA_BLOCK (ATMEL_FPGA_PAGE * 8) /* FPGA block size*/ 6362306a36Sopenharmony_ci#define ATMEL_SOLOS_BLOCK (ATMEL_SOLOS_PAGE * 8) /* Solos block size*/ 6462306a36Sopenharmony_ci/* Current boards use M25P/M25PE SPI flash */ 6562306a36Sopenharmony_ci#define SPI_FLASH_BLOCK (256 * 64) 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci#define RX_BUF(card, nr) ((card->buffers) + (nr)*(card->buffer_size)*2) 6862306a36Sopenharmony_ci#define TX_BUF(card, nr) ((card->buffers) + (nr)*(card->buffer_size)*2 + (card->buffer_size)) 6962306a36Sopenharmony_ci#define FLASH_BUF ((card->buffers) + 4*(card->buffer_size)*2) 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci#define RX_DMA_SIZE 2048 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci#define FPGA_VERSION(a,b) (((a) << 8) + (b)) 7462306a36Sopenharmony_ci#define LEGACY_BUFFERS 2 7562306a36Sopenharmony_ci#define DMA_SUPPORTED 4 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_cistatic int reset = 0; 7862306a36Sopenharmony_cistatic int atmdebug = 0; 7962306a36Sopenharmony_cistatic int firmware_upgrade = 0; 8062306a36Sopenharmony_cistatic int fpga_upgrade = 0; 8162306a36Sopenharmony_cistatic int db_firmware_upgrade = 0; 8262306a36Sopenharmony_cistatic int db_fpga_upgrade = 0; 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_cistruct pkt_hdr { 8562306a36Sopenharmony_ci __le16 size; 8662306a36Sopenharmony_ci __le16 vpi; 8762306a36Sopenharmony_ci __le16 vci; 8862306a36Sopenharmony_ci __le16 type; 8962306a36Sopenharmony_ci}; 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistruct solos_skb_cb { 9262306a36Sopenharmony_ci struct atm_vcc *vcc; 9362306a36Sopenharmony_ci uint32_t dma_addr; 9462306a36Sopenharmony_ci}; 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci#define SKB_CB(skb) ((struct solos_skb_cb *)skb->cb) 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci#define PKT_DATA 0 10062306a36Sopenharmony_ci#define PKT_COMMAND 1 10162306a36Sopenharmony_ci#define PKT_POPEN 3 10262306a36Sopenharmony_ci#define PKT_PCLOSE 4 10362306a36Sopenharmony_ci#define PKT_STATUS 5 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_cistruct solos_card { 10662306a36Sopenharmony_ci void __iomem *config_regs; 10762306a36Sopenharmony_ci void __iomem *buffers; 10862306a36Sopenharmony_ci int nr_ports; 10962306a36Sopenharmony_ci int tx_mask; 11062306a36Sopenharmony_ci struct pci_dev *dev; 11162306a36Sopenharmony_ci struct atm_dev *atmdev[4]; 11262306a36Sopenharmony_ci struct tasklet_struct tlet; 11362306a36Sopenharmony_ci spinlock_t tx_lock; 11462306a36Sopenharmony_ci spinlock_t tx_queue_lock; 11562306a36Sopenharmony_ci spinlock_t cli_queue_lock; 11662306a36Sopenharmony_ci spinlock_t param_queue_lock; 11762306a36Sopenharmony_ci struct list_head param_queue; 11862306a36Sopenharmony_ci struct sk_buff_head tx_queue[4]; 11962306a36Sopenharmony_ci struct sk_buff_head cli_queue[4]; 12062306a36Sopenharmony_ci struct sk_buff *tx_skb[4]; 12162306a36Sopenharmony_ci struct sk_buff *rx_skb[4]; 12262306a36Sopenharmony_ci unsigned char *dma_bounce; 12362306a36Sopenharmony_ci wait_queue_head_t param_wq; 12462306a36Sopenharmony_ci wait_queue_head_t fw_wq; 12562306a36Sopenharmony_ci int using_dma; 12662306a36Sopenharmony_ci int dma_alignment; 12762306a36Sopenharmony_ci int fpga_version; 12862306a36Sopenharmony_ci int buffer_size; 12962306a36Sopenharmony_ci int atmel_flash; 13062306a36Sopenharmony_ci}; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_cistruct solos_param { 13462306a36Sopenharmony_ci struct list_head list; 13562306a36Sopenharmony_ci pid_t pid; 13662306a36Sopenharmony_ci int port; 13762306a36Sopenharmony_ci struct sk_buff *response; 13862306a36Sopenharmony_ci}; 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_ci#define SOLOS_CHAN(atmdev) ((int)(unsigned long)(atmdev)->phy_data) 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ciMODULE_AUTHOR("Traverse Technologies <support@traverse.com.au>"); 14362306a36Sopenharmony_ciMODULE_DESCRIPTION("Solos PCI driver"); 14462306a36Sopenharmony_ciMODULE_VERSION(VERSION); 14562306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 14662306a36Sopenharmony_ciMODULE_FIRMWARE("solos-FPGA.bin"); 14762306a36Sopenharmony_ciMODULE_FIRMWARE("solos-Firmware.bin"); 14862306a36Sopenharmony_ciMODULE_FIRMWARE("solos-db-FPGA.bin"); 14962306a36Sopenharmony_ciMODULE_PARM_DESC(reset, "Reset Solos chips on startup"); 15062306a36Sopenharmony_ciMODULE_PARM_DESC(atmdebug, "Print ATM data"); 15162306a36Sopenharmony_ciMODULE_PARM_DESC(firmware_upgrade, "Initiate Solos firmware upgrade"); 15262306a36Sopenharmony_ciMODULE_PARM_DESC(fpga_upgrade, "Initiate FPGA upgrade"); 15362306a36Sopenharmony_ciMODULE_PARM_DESC(db_firmware_upgrade, "Initiate daughter board Solos firmware upgrade"); 15462306a36Sopenharmony_ciMODULE_PARM_DESC(db_fpga_upgrade, "Initiate daughter board FPGA upgrade"); 15562306a36Sopenharmony_cimodule_param(reset, int, 0444); 15662306a36Sopenharmony_cimodule_param(atmdebug, int, 0644); 15762306a36Sopenharmony_cimodule_param(firmware_upgrade, int, 0444); 15862306a36Sopenharmony_cimodule_param(fpga_upgrade, int, 0444); 15962306a36Sopenharmony_cimodule_param(db_firmware_upgrade, int, 0444); 16062306a36Sopenharmony_cimodule_param(db_fpga_upgrade, int, 0444); 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_cistatic void fpga_queue(struct solos_card *card, int port, struct sk_buff *skb, 16362306a36Sopenharmony_ci struct atm_vcc *vcc); 16462306a36Sopenharmony_cistatic uint32_t fpga_tx(struct solos_card *); 16562306a36Sopenharmony_cistatic irqreturn_t solos_irq(int irq, void *dev_id); 16662306a36Sopenharmony_cistatic struct atm_vcc* find_vcc(struct atm_dev *dev, short vpi, int vci); 16762306a36Sopenharmony_cistatic int atm_init(struct solos_card *, struct device *); 16862306a36Sopenharmony_cistatic void atm_remove(struct solos_card *); 16962306a36Sopenharmony_cistatic int send_command(struct solos_card *card, int dev, const char *buf, size_t size); 17062306a36Sopenharmony_cistatic void solos_bh(unsigned long); 17162306a36Sopenharmony_cistatic int print_buffer(struct sk_buff *buf); 17262306a36Sopenharmony_ci 17362306a36Sopenharmony_cistatic inline void solos_pop(struct atm_vcc *vcc, struct sk_buff *skb) 17462306a36Sopenharmony_ci{ 17562306a36Sopenharmony_ci if (vcc->pop) 17662306a36Sopenharmony_ci vcc->pop(vcc, skb); 17762306a36Sopenharmony_ci else 17862306a36Sopenharmony_ci dev_kfree_skb_any(skb); 17962306a36Sopenharmony_ci} 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_cistatic ssize_t solos_param_show(struct device *dev, struct device_attribute *attr, 18262306a36Sopenharmony_ci char *buf) 18362306a36Sopenharmony_ci{ 18462306a36Sopenharmony_ci struct atm_dev *atmdev = container_of(dev, struct atm_dev, class_dev); 18562306a36Sopenharmony_ci struct solos_card *card = atmdev->dev_data; 18662306a36Sopenharmony_ci struct solos_param prm; 18762306a36Sopenharmony_ci struct sk_buff *skb; 18862306a36Sopenharmony_ci struct pkt_hdr *header; 18962306a36Sopenharmony_ci int buflen; 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_ci buflen = strlen(attr->attr.name) + 10; 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_ci skb = alloc_skb(sizeof(*header) + buflen, GFP_KERNEL); 19462306a36Sopenharmony_ci if (!skb) { 19562306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff in solos_param_show()\n"); 19662306a36Sopenharmony_ci return -ENOMEM; 19762306a36Sopenharmony_ci } 19862306a36Sopenharmony_ci 19962306a36Sopenharmony_ci header = skb_put(skb, sizeof(*header)); 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_ci buflen = snprintf((void *)&header[1], buflen - 1, 20262306a36Sopenharmony_ci "L%05d\n%s\n", current->pid, attr->attr.name); 20362306a36Sopenharmony_ci skb_put(skb, buflen); 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_ci header->size = cpu_to_le16(buflen); 20662306a36Sopenharmony_ci header->vpi = cpu_to_le16(0); 20762306a36Sopenharmony_ci header->vci = cpu_to_le16(0); 20862306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_COMMAND); 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_ci prm.pid = current->pid; 21162306a36Sopenharmony_ci prm.response = NULL; 21262306a36Sopenharmony_ci prm.port = SOLOS_CHAN(atmdev); 21362306a36Sopenharmony_ci 21462306a36Sopenharmony_ci spin_lock_irq(&card->param_queue_lock); 21562306a36Sopenharmony_ci list_add(&prm.list, &card->param_queue); 21662306a36Sopenharmony_ci spin_unlock_irq(&card->param_queue_lock); 21762306a36Sopenharmony_ci 21862306a36Sopenharmony_ci fpga_queue(card, prm.port, skb, NULL); 21962306a36Sopenharmony_ci 22062306a36Sopenharmony_ci wait_event_timeout(card->param_wq, prm.response, 5 * HZ); 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_ci spin_lock_irq(&card->param_queue_lock); 22362306a36Sopenharmony_ci list_del(&prm.list); 22462306a36Sopenharmony_ci spin_unlock_irq(&card->param_queue_lock); 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci if (!prm.response) 22762306a36Sopenharmony_ci return -EIO; 22862306a36Sopenharmony_ci 22962306a36Sopenharmony_ci buflen = prm.response->len; 23062306a36Sopenharmony_ci memcpy(buf, prm.response->data, buflen); 23162306a36Sopenharmony_ci kfree_skb(prm.response); 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_ci return buflen; 23462306a36Sopenharmony_ci} 23562306a36Sopenharmony_ci 23662306a36Sopenharmony_cistatic ssize_t solos_param_store(struct device *dev, struct device_attribute *attr, 23762306a36Sopenharmony_ci const char *buf, size_t count) 23862306a36Sopenharmony_ci{ 23962306a36Sopenharmony_ci struct atm_dev *atmdev = container_of(dev, struct atm_dev, class_dev); 24062306a36Sopenharmony_ci struct solos_card *card = atmdev->dev_data; 24162306a36Sopenharmony_ci struct solos_param prm; 24262306a36Sopenharmony_ci struct sk_buff *skb; 24362306a36Sopenharmony_ci struct pkt_hdr *header; 24462306a36Sopenharmony_ci int buflen; 24562306a36Sopenharmony_ci ssize_t ret; 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_ci buflen = strlen(attr->attr.name) + 11 + count; 24862306a36Sopenharmony_ci 24962306a36Sopenharmony_ci skb = alloc_skb(sizeof(*header) + buflen, GFP_KERNEL); 25062306a36Sopenharmony_ci if (!skb) { 25162306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff in solos_param_store()\n"); 25262306a36Sopenharmony_ci return -ENOMEM; 25362306a36Sopenharmony_ci } 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_ci header = skb_put(skb, sizeof(*header)); 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_ci buflen = snprintf((void *)&header[1], buflen - 1, 25862306a36Sopenharmony_ci "L%05d\n%s\n%s\n", current->pid, attr->attr.name, buf); 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_ci skb_put(skb, buflen); 26162306a36Sopenharmony_ci header->size = cpu_to_le16(buflen); 26262306a36Sopenharmony_ci header->vpi = cpu_to_le16(0); 26362306a36Sopenharmony_ci header->vci = cpu_to_le16(0); 26462306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_COMMAND); 26562306a36Sopenharmony_ci 26662306a36Sopenharmony_ci prm.pid = current->pid; 26762306a36Sopenharmony_ci prm.response = NULL; 26862306a36Sopenharmony_ci prm.port = SOLOS_CHAN(atmdev); 26962306a36Sopenharmony_ci 27062306a36Sopenharmony_ci spin_lock_irq(&card->param_queue_lock); 27162306a36Sopenharmony_ci list_add(&prm.list, &card->param_queue); 27262306a36Sopenharmony_ci spin_unlock_irq(&card->param_queue_lock); 27362306a36Sopenharmony_ci 27462306a36Sopenharmony_ci fpga_queue(card, prm.port, skb, NULL); 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci wait_event_timeout(card->param_wq, prm.response, 5 * HZ); 27762306a36Sopenharmony_ci 27862306a36Sopenharmony_ci spin_lock_irq(&card->param_queue_lock); 27962306a36Sopenharmony_ci list_del(&prm.list); 28062306a36Sopenharmony_ci spin_unlock_irq(&card->param_queue_lock); 28162306a36Sopenharmony_ci 28262306a36Sopenharmony_ci skb = prm.response; 28362306a36Sopenharmony_ci 28462306a36Sopenharmony_ci if (!skb) 28562306a36Sopenharmony_ci return -EIO; 28662306a36Sopenharmony_ci 28762306a36Sopenharmony_ci buflen = skb->len; 28862306a36Sopenharmony_ci 28962306a36Sopenharmony_ci /* Sometimes it has a newline, sometimes it doesn't. */ 29062306a36Sopenharmony_ci if (skb->data[buflen - 1] == '\n') 29162306a36Sopenharmony_ci buflen--; 29262306a36Sopenharmony_ci 29362306a36Sopenharmony_ci if (buflen == 2 && !strncmp(skb->data, "OK", 2)) 29462306a36Sopenharmony_ci ret = count; 29562306a36Sopenharmony_ci else if (buflen == 5 && !strncmp(skb->data, "ERROR", 5)) 29662306a36Sopenharmony_ci ret = -EIO; 29762306a36Sopenharmony_ci else { 29862306a36Sopenharmony_ci /* We know we have enough space allocated for this; we allocated 29962306a36Sopenharmony_ci it ourselves */ 30062306a36Sopenharmony_ci skb->data[buflen] = 0; 30162306a36Sopenharmony_ci 30262306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Unexpected parameter response: '%s'\n", 30362306a36Sopenharmony_ci skb->data); 30462306a36Sopenharmony_ci ret = -EIO; 30562306a36Sopenharmony_ci } 30662306a36Sopenharmony_ci kfree_skb(skb); 30762306a36Sopenharmony_ci 30862306a36Sopenharmony_ci return ret; 30962306a36Sopenharmony_ci} 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_cistatic char *next_string(struct sk_buff *skb) 31262306a36Sopenharmony_ci{ 31362306a36Sopenharmony_ci int i = 0; 31462306a36Sopenharmony_ci char *this = skb->data; 31562306a36Sopenharmony_ci 31662306a36Sopenharmony_ci for (i = 0; i < skb->len; i++) { 31762306a36Sopenharmony_ci if (this[i] == '\n') { 31862306a36Sopenharmony_ci this[i] = 0; 31962306a36Sopenharmony_ci skb_pull(skb, i + 1); 32062306a36Sopenharmony_ci return this; 32162306a36Sopenharmony_ci } 32262306a36Sopenharmony_ci if (!isprint(this[i])) 32362306a36Sopenharmony_ci return NULL; 32462306a36Sopenharmony_ci } 32562306a36Sopenharmony_ci return NULL; 32662306a36Sopenharmony_ci} 32762306a36Sopenharmony_ci 32862306a36Sopenharmony_ci/* 32962306a36Sopenharmony_ci * Status packet has fields separated by \n, starting with a version number 33062306a36Sopenharmony_ci * for the information therein. Fields are.... 33162306a36Sopenharmony_ci * 33262306a36Sopenharmony_ci * packet version 33362306a36Sopenharmony_ci * RxBitRate (version >= 1) 33462306a36Sopenharmony_ci * TxBitRate (version >= 1) 33562306a36Sopenharmony_ci * State (version >= 1) 33662306a36Sopenharmony_ci * LocalSNRMargin (version >= 1) 33762306a36Sopenharmony_ci * LocalLineAttn (version >= 1) 33862306a36Sopenharmony_ci */ 33962306a36Sopenharmony_cistatic int process_status(struct solos_card *card, int port, struct sk_buff *skb) 34062306a36Sopenharmony_ci{ 34162306a36Sopenharmony_ci char *str, *state_str, *snr, *attn; 34262306a36Sopenharmony_ci int ver, rate_up, rate_down, err; 34362306a36Sopenharmony_ci 34462306a36Sopenharmony_ci if (!card->atmdev[port]) 34562306a36Sopenharmony_ci return -ENODEV; 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_ci str = next_string(skb); 34862306a36Sopenharmony_ci if (!str) 34962306a36Sopenharmony_ci return -EIO; 35062306a36Sopenharmony_ci 35162306a36Sopenharmony_ci err = kstrtoint(str, 10, &ver); 35262306a36Sopenharmony_ci if (err) { 35362306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Unexpected status interrupt version\n"); 35462306a36Sopenharmony_ci return err; 35562306a36Sopenharmony_ci } 35662306a36Sopenharmony_ci if (ver < 1) { 35762306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Unexpected status interrupt version %d\n", 35862306a36Sopenharmony_ci ver); 35962306a36Sopenharmony_ci return -EIO; 36062306a36Sopenharmony_ci } 36162306a36Sopenharmony_ci 36262306a36Sopenharmony_ci str = next_string(skb); 36362306a36Sopenharmony_ci if (!str) 36462306a36Sopenharmony_ci return -EIO; 36562306a36Sopenharmony_ci if (!strcmp(str, "ERROR")) { 36662306a36Sopenharmony_ci dev_dbg(&card->dev->dev, "Status packet indicated Solos error on port %d (starting up?)\n", 36762306a36Sopenharmony_ci port); 36862306a36Sopenharmony_ci return 0; 36962306a36Sopenharmony_ci } 37062306a36Sopenharmony_ci 37162306a36Sopenharmony_ci err = kstrtoint(str, 10, &rate_down); 37262306a36Sopenharmony_ci if (err) 37362306a36Sopenharmony_ci return err; 37462306a36Sopenharmony_ci 37562306a36Sopenharmony_ci str = next_string(skb); 37662306a36Sopenharmony_ci if (!str) 37762306a36Sopenharmony_ci return -EIO; 37862306a36Sopenharmony_ci err = kstrtoint(str, 10, &rate_up); 37962306a36Sopenharmony_ci if (err) 38062306a36Sopenharmony_ci return err; 38162306a36Sopenharmony_ci 38262306a36Sopenharmony_ci state_str = next_string(skb); 38362306a36Sopenharmony_ci if (!state_str) 38462306a36Sopenharmony_ci return -EIO; 38562306a36Sopenharmony_ci 38662306a36Sopenharmony_ci /* Anything but 'Showtime' is down */ 38762306a36Sopenharmony_ci if (strcmp(state_str, "Showtime")) { 38862306a36Sopenharmony_ci atm_dev_signal_change(card->atmdev[port], ATM_PHY_SIG_LOST); 38962306a36Sopenharmony_ci dev_info(&card->dev->dev, "Port %d: %s\n", port, state_str); 39062306a36Sopenharmony_ci return 0; 39162306a36Sopenharmony_ci } 39262306a36Sopenharmony_ci 39362306a36Sopenharmony_ci snr = next_string(skb); 39462306a36Sopenharmony_ci if (!snr) 39562306a36Sopenharmony_ci return -EIO; 39662306a36Sopenharmony_ci attn = next_string(skb); 39762306a36Sopenharmony_ci if (!attn) 39862306a36Sopenharmony_ci return -EIO; 39962306a36Sopenharmony_ci 40062306a36Sopenharmony_ci dev_info(&card->dev->dev, "Port %d: %s @%d/%d kb/s%s%s%s%s\n", 40162306a36Sopenharmony_ci port, state_str, rate_down/1000, rate_up/1000, 40262306a36Sopenharmony_ci snr[0]?", SNR ":"", snr, attn[0]?", Attn ":"", attn); 40362306a36Sopenharmony_ci 40462306a36Sopenharmony_ci card->atmdev[port]->link_rate = rate_down / 424; 40562306a36Sopenharmony_ci atm_dev_signal_change(card->atmdev[port], ATM_PHY_SIG_FOUND); 40662306a36Sopenharmony_ci 40762306a36Sopenharmony_ci return 0; 40862306a36Sopenharmony_ci} 40962306a36Sopenharmony_ci 41062306a36Sopenharmony_cistatic int process_command(struct solos_card *card, int port, struct sk_buff *skb) 41162306a36Sopenharmony_ci{ 41262306a36Sopenharmony_ci struct solos_param *prm; 41362306a36Sopenharmony_ci unsigned long flags; 41462306a36Sopenharmony_ci int cmdpid; 41562306a36Sopenharmony_ci int found = 0, err; 41662306a36Sopenharmony_ci 41762306a36Sopenharmony_ci if (skb->len < 7) 41862306a36Sopenharmony_ci return 0; 41962306a36Sopenharmony_ci 42062306a36Sopenharmony_ci if (skb->data[0] != 'L' || !isdigit(skb->data[1]) || 42162306a36Sopenharmony_ci !isdigit(skb->data[2]) || !isdigit(skb->data[3]) || 42262306a36Sopenharmony_ci !isdigit(skb->data[4]) || !isdigit(skb->data[5]) || 42362306a36Sopenharmony_ci skb->data[6] != '\n') 42462306a36Sopenharmony_ci return 0; 42562306a36Sopenharmony_ci 42662306a36Sopenharmony_ci err = kstrtoint(&skb->data[1], 10, &cmdpid); 42762306a36Sopenharmony_ci if (err) 42862306a36Sopenharmony_ci return err; 42962306a36Sopenharmony_ci 43062306a36Sopenharmony_ci spin_lock_irqsave(&card->param_queue_lock, flags); 43162306a36Sopenharmony_ci list_for_each_entry(prm, &card->param_queue, list) { 43262306a36Sopenharmony_ci if (prm->port == port && prm->pid == cmdpid) { 43362306a36Sopenharmony_ci prm->response = skb; 43462306a36Sopenharmony_ci skb_pull(skb, 7); 43562306a36Sopenharmony_ci wake_up(&card->param_wq); 43662306a36Sopenharmony_ci found = 1; 43762306a36Sopenharmony_ci break; 43862306a36Sopenharmony_ci } 43962306a36Sopenharmony_ci } 44062306a36Sopenharmony_ci spin_unlock_irqrestore(&card->param_queue_lock, flags); 44162306a36Sopenharmony_ci return found; 44262306a36Sopenharmony_ci} 44362306a36Sopenharmony_ci 44462306a36Sopenharmony_cistatic ssize_t console_show(struct device *dev, struct device_attribute *attr, 44562306a36Sopenharmony_ci char *buf) 44662306a36Sopenharmony_ci{ 44762306a36Sopenharmony_ci struct atm_dev *atmdev = container_of(dev, struct atm_dev, class_dev); 44862306a36Sopenharmony_ci struct solos_card *card = atmdev->dev_data; 44962306a36Sopenharmony_ci struct sk_buff *skb; 45062306a36Sopenharmony_ci unsigned int len; 45162306a36Sopenharmony_ci 45262306a36Sopenharmony_ci spin_lock_bh(&card->cli_queue_lock); 45362306a36Sopenharmony_ci skb = skb_dequeue(&card->cli_queue[SOLOS_CHAN(atmdev)]); 45462306a36Sopenharmony_ci spin_unlock_bh(&card->cli_queue_lock); 45562306a36Sopenharmony_ci if(skb == NULL) 45662306a36Sopenharmony_ci return sprintf(buf, "No data.\n"); 45762306a36Sopenharmony_ci 45862306a36Sopenharmony_ci len = skb->len; 45962306a36Sopenharmony_ci memcpy(buf, skb->data, len); 46062306a36Sopenharmony_ci 46162306a36Sopenharmony_ci kfree_skb(skb); 46262306a36Sopenharmony_ci return len; 46362306a36Sopenharmony_ci} 46462306a36Sopenharmony_ci 46562306a36Sopenharmony_cistatic int send_command(struct solos_card *card, int dev, const char *buf, size_t size) 46662306a36Sopenharmony_ci{ 46762306a36Sopenharmony_ci struct sk_buff *skb; 46862306a36Sopenharmony_ci struct pkt_hdr *header; 46962306a36Sopenharmony_ci 47062306a36Sopenharmony_ci if (size > (BUF_SIZE - sizeof(*header))) { 47162306a36Sopenharmony_ci dev_dbg(&card->dev->dev, "Command is too big. Dropping request\n"); 47262306a36Sopenharmony_ci return 0; 47362306a36Sopenharmony_ci } 47462306a36Sopenharmony_ci skb = alloc_skb(size + sizeof(*header), GFP_ATOMIC); 47562306a36Sopenharmony_ci if (!skb) { 47662306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff in send_command()\n"); 47762306a36Sopenharmony_ci return 0; 47862306a36Sopenharmony_ci } 47962306a36Sopenharmony_ci 48062306a36Sopenharmony_ci header = skb_put(skb, sizeof(*header)); 48162306a36Sopenharmony_ci 48262306a36Sopenharmony_ci header->size = cpu_to_le16(size); 48362306a36Sopenharmony_ci header->vpi = cpu_to_le16(0); 48462306a36Sopenharmony_ci header->vci = cpu_to_le16(0); 48562306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_COMMAND); 48662306a36Sopenharmony_ci 48762306a36Sopenharmony_ci skb_put_data(skb, buf, size); 48862306a36Sopenharmony_ci 48962306a36Sopenharmony_ci fpga_queue(card, dev, skb, NULL); 49062306a36Sopenharmony_ci 49162306a36Sopenharmony_ci return 0; 49262306a36Sopenharmony_ci} 49362306a36Sopenharmony_ci 49462306a36Sopenharmony_cistatic ssize_t console_store(struct device *dev, struct device_attribute *attr, 49562306a36Sopenharmony_ci const char *buf, size_t count) 49662306a36Sopenharmony_ci{ 49762306a36Sopenharmony_ci struct atm_dev *atmdev = container_of(dev, struct atm_dev, class_dev); 49862306a36Sopenharmony_ci struct solos_card *card = atmdev->dev_data; 49962306a36Sopenharmony_ci int err; 50062306a36Sopenharmony_ci 50162306a36Sopenharmony_ci err = send_command(card, SOLOS_CHAN(atmdev), buf, count); 50262306a36Sopenharmony_ci 50362306a36Sopenharmony_ci return err?:count; 50462306a36Sopenharmony_ci} 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_cistruct geos_gpio_attr { 50762306a36Sopenharmony_ci struct device_attribute attr; 50862306a36Sopenharmony_ci int offset; 50962306a36Sopenharmony_ci}; 51062306a36Sopenharmony_ci 51162306a36Sopenharmony_ci#define SOLOS_GPIO_ATTR(_name, _mode, _show, _store, _offset) \ 51262306a36Sopenharmony_ci struct geos_gpio_attr gpio_attr_##_name = { \ 51362306a36Sopenharmony_ci .attr = __ATTR(_name, _mode, _show, _store), \ 51462306a36Sopenharmony_ci .offset = _offset } 51562306a36Sopenharmony_ci 51662306a36Sopenharmony_cistatic ssize_t geos_gpio_store(struct device *dev, struct device_attribute *attr, 51762306a36Sopenharmony_ci const char *buf, size_t count) 51862306a36Sopenharmony_ci{ 51962306a36Sopenharmony_ci struct geos_gpio_attr *gattr = container_of(attr, struct geos_gpio_attr, attr); 52062306a36Sopenharmony_ci struct solos_card *card = dev_get_drvdata(dev); 52162306a36Sopenharmony_ci uint32_t data32; 52262306a36Sopenharmony_ci 52362306a36Sopenharmony_ci if (count != 1 && (count != 2 || buf[1] != '\n')) 52462306a36Sopenharmony_ci return -EINVAL; 52562306a36Sopenharmony_ci 52662306a36Sopenharmony_ci spin_lock_irq(&card->param_queue_lock); 52762306a36Sopenharmony_ci data32 = ioread32(card->config_regs + GPIO_STATUS); 52862306a36Sopenharmony_ci if (buf[0] == '1') { 52962306a36Sopenharmony_ci data32 |= 1 << gattr->offset; 53062306a36Sopenharmony_ci iowrite32(data32, card->config_regs + GPIO_STATUS); 53162306a36Sopenharmony_ci } else if (buf[0] == '0') { 53262306a36Sopenharmony_ci data32 &= ~(1 << gattr->offset); 53362306a36Sopenharmony_ci iowrite32(data32, card->config_regs + GPIO_STATUS); 53462306a36Sopenharmony_ci } else { 53562306a36Sopenharmony_ci count = -EINVAL; 53662306a36Sopenharmony_ci } 53762306a36Sopenharmony_ci spin_unlock_irq(&card->param_queue_lock); 53862306a36Sopenharmony_ci return count; 53962306a36Sopenharmony_ci} 54062306a36Sopenharmony_ci 54162306a36Sopenharmony_cistatic ssize_t geos_gpio_show(struct device *dev, struct device_attribute *attr, 54262306a36Sopenharmony_ci char *buf) 54362306a36Sopenharmony_ci{ 54462306a36Sopenharmony_ci struct geos_gpio_attr *gattr = container_of(attr, struct geos_gpio_attr, attr); 54562306a36Sopenharmony_ci struct solos_card *card = dev_get_drvdata(dev); 54662306a36Sopenharmony_ci uint32_t data32; 54762306a36Sopenharmony_ci 54862306a36Sopenharmony_ci data32 = ioread32(card->config_regs + GPIO_STATUS); 54962306a36Sopenharmony_ci data32 = (data32 >> gattr->offset) & 1; 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_ci return sprintf(buf, "%d\n", data32); 55262306a36Sopenharmony_ci} 55362306a36Sopenharmony_ci 55462306a36Sopenharmony_cistatic ssize_t hardware_show(struct device *dev, struct device_attribute *attr, 55562306a36Sopenharmony_ci char *buf) 55662306a36Sopenharmony_ci{ 55762306a36Sopenharmony_ci struct geos_gpio_attr *gattr = container_of(attr, struct geos_gpio_attr, attr); 55862306a36Sopenharmony_ci struct solos_card *card = dev_get_drvdata(dev); 55962306a36Sopenharmony_ci uint32_t data32; 56062306a36Sopenharmony_ci 56162306a36Sopenharmony_ci data32 = ioread32(card->config_regs + GPIO_STATUS); 56262306a36Sopenharmony_ci switch (gattr->offset) { 56362306a36Sopenharmony_ci case 0: 56462306a36Sopenharmony_ci /* HardwareVersion */ 56562306a36Sopenharmony_ci data32 = data32 & 0x1F; 56662306a36Sopenharmony_ci break; 56762306a36Sopenharmony_ci case 1: 56862306a36Sopenharmony_ci /* HardwareVariant */ 56962306a36Sopenharmony_ci data32 = (data32 >> 5) & 0x0F; 57062306a36Sopenharmony_ci break; 57162306a36Sopenharmony_ci } 57262306a36Sopenharmony_ci return sprintf(buf, "%d\n", data32); 57362306a36Sopenharmony_ci} 57462306a36Sopenharmony_ci 57562306a36Sopenharmony_cistatic DEVICE_ATTR_RW(console); 57662306a36Sopenharmony_ci 57762306a36Sopenharmony_ci 57862306a36Sopenharmony_ci#define SOLOS_ATTR_RO(x) static DEVICE_ATTR(x, 0444, solos_param_show, NULL); 57962306a36Sopenharmony_ci#define SOLOS_ATTR_RW(x) static DEVICE_ATTR(x, 0644, solos_param_show, solos_param_store); 58062306a36Sopenharmony_ci 58162306a36Sopenharmony_ci#include "solos-attrlist.c" 58262306a36Sopenharmony_ci 58362306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(GPIO1, 0644, geos_gpio_show, geos_gpio_store, 9); 58462306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(GPIO2, 0644, geos_gpio_show, geos_gpio_store, 10); 58562306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(GPIO3, 0644, geos_gpio_show, geos_gpio_store, 11); 58662306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(GPIO4, 0644, geos_gpio_show, geos_gpio_store, 12); 58762306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(GPIO5, 0644, geos_gpio_show, geos_gpio_store, 13); 58862306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(PushButton, 0444, geos_gpio_show, NULL, 14); 58962306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(HardwareVersion, 0444, hardware_show, NULL, 0); 59062306a36Sopenharmony_cistatic SOLOS_GPIO_ATTR(HardwareVariant, 0444, hardware_show, NULL, 1); 59162306a36Sopenharmony_ci#undef SOLOS_ATTR_RO 59262306a36Sopenharmony_ci#undef SOLOS_ATTR_RW 59362306a36Sopenharmony_ci 59462306a36Sopenharmony_ci#define SOLOS_ATTR_RO(x) &dev_attr_##x.attr, 59562306a36Sopenharmony_ci#define SOLOS_ATTR_RW(x) &dev_attr_##x.attr, 59662306a36Sopenharmony_ci 59762306a36Sopenharmony_cistatic struct attribute *solos_attrs[] = { 59862306a36Sopenharmony_ci#include "solos-attrlist.c" 59962306a36Sopenharmony_ci NULL 60062306a36Sopenharmony_ci}; 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_cistatic const struct attribute_group solos_attr_group = { 60362306a36Sopenharmony_ci .attrs = solos_attrs, 60462306a36Sopenharmony_ci .name = "parameters", 60562306a36Sopenharmony_ci}; 60662306a36Sopenharmony_ci 60762306a36Sopenharmony_cistatic struct attribute *gpio_attrs[] = { 60862306a36Sopenharmony_ci &gpio_attr_GPIO1.attr.attr, 60962306a36Sopenharmony_ci &gpio_attr_GPIO2.attr.attr, 61062306a36Sopenharmony_ci &gpio_attr_GPIO3.attr.attr, 61162306a36Sopenharmony_ci &gpio_attr_GPIO4.attr.attr, 61262306a36Sopenharmony_ci &gpio_attr_GPIO5.attr.attr, 61362306a36Sopenharmony_ci &gpio_attr_PushButton.attr.attr, 61462306a36Sopenharmony_ci &gpio_attr_HardwareVersion.attr.attr, 61562306a36Sopenharmony_ci &gpio_attr_HardwareVariant.attr.attr, 61662306a36Sopenharmony_ci NULL 61762306a36Sopenharmony_ci}; 61862306a36Sopenharmony_ci 61962306a36Sopenharmony_cistatic const struct attribute_group gpio_attr_group = { 62062306a36Sopenharmony_ci .attrs = gpio_attrs, 62162306a36Sopenharmony_ci .name = "gpio", 62262306a36Sopenharmony_ci}; 62362306a36Sopenharmony_ci 62462306a36Sopenharmony_cistatic int flash_upgrade(struct solos_card *card, int chip) 62562306a36Sopenharmony_ci{ 62662306a36Sopenharmony_ci const struct firmware *fw; 62762306a36Sopenharmony_ci const char *fw_name; 62862306a36Sopenharmony_ci int blocksize = 0; 62962306a36Sopenharmony_ci int numblocks = 0; 63062306a36Sopenharmony_ci int offset; 63162306a36Sopenharmony_ci 63262306a36Sopenharmony_ci switch (chip) { 63362306a36Sopenharmony_ci case 0: 63462306a36Sopenharmony_ci fw_name = "solos-FPGA.bin"; 63562306a36Sopenharmony_ci if (card->atmel_flash) 63662306a36Sopenharmony_ci blocksize = ATMEL_FPGA_BLOCK; 63762306a36Sopenharmony_ci else 63862306a36Sopenharmony_ci blocksize = SPI_FLASH_BLOCK; 63962306a36Sopenharmony_ci break; 64062306a36Sopenharmony_ci case 1: 64162306a36Sopenharmony_ci fw_name = "solos-Firmware.bin"; 64262306a36Sopenharmony_ci if (card->atmel_flash) 64362306a36Sopenharmony_ci blocksize = ATMEL_SOLOS_BLOCK; 64462306a36Sopenharmony_ci else 64562306a36Sopenharmony_ci blocksize = SPI_FLASH_BLOCK; 64662306a36Sopenharmony_ci break; 64762306a36Sopenharmony_ci case 2: 64862306a36Sopenharmony_ci if (card->fpga_version > LEGACY_BUFFERS){ 64962306a36Sopenharmony_ci fw_name = "solos-db-FPGA.bin"; 65062306a36Sopenharmony_ci if (card->atmel_flash) 65162306a36Sopenharmony_ci blocksize = ATMEL_FPGA_BLOCK; 65262306a36Sopenharmony_ci else 65362306a36Sopenharmony_ci blocksize = SPI_FLASH_BLOCK; 65462306a36Sopenharmony_ci } else { 65562306a36Sopenharmony_ci dev_info(&card->dev->dev, "FPGA version doesn't support" 65662306a36Sopenharmony_ci " daughter board upgrades\n"); 65762306a36Sopenharmony_ci return -EPERM; 65862306a36Sopenharmony_ci } 65962306a36Sopenharmony_ci break; 66062306a36Sopenharmony_ci case 3: 66162306a36Sopenharmony_ci if (card->fpga_version > LEGACY_BUFFERS){ 66262306a36Sopenharmony_ci fw_name = "solos-Firmware.bin"; 66362306a36Sopenharmony_ci if (card->atmel_flash) 66462306a36Sopenharmony_ci blocksize = ATMEL_SOLOS_BLOCK; 66562306a36Sopenharmony_ci else 66662306a36Sopenharmony_ci blocksize = SPI_FLASH_BLOCK; 66762306a36Sopenharmony_ci } else { 66862306a36Sopenharmony_ci dev_info(&card->dev->dev, "FPGA version doesn't support" 66962306a36Sopenharmony_ci " daughter board upgrades\n"); 67062306a36Sopenharmony_ci return -EPERM; 67162306a36Sopenharmony_ci } 67262306a36Sopenharmony_ci break; 67362306a36Sopenharmony_ci default: 67462306a36Sopenharmony_ci return -ENODEV; 67562306a36Sopenharmony_ci } 67662306a36Sopenharmony_ci 67762306a36Sopenharmony_ci if (request_firmware(&fw, fw_name, &card->dev->dev)) 67862306a36Sopenharmony_ci return -ENOENT; 67962306a36Sopenharmony_ci 68062306a36Sopenharmony_ci dev_info(&card->dev->dev, "Flash upgrade starting\n"); 68162306a36Sopenharmony_ci 68262306a36Sopenharmony_ci /* New FPGAs require driver version before permitting flash upgrades */ 68362306a36Sopenharmony_ci iowrite32(DRIVER_VERSION, card->config_regs + DRIVER_VER); 68462306a36Sopenharmony_ci 68562306a36Sopenharmony_ci numblocks = fw->size / blocksize; 68662306a36Sopenharmony_ci dev_info(&card->dev->dev, "Firmware size: %zd\n", fw->size); 68762306a36Sopenharmony_ci dev_info(&card->dev->dev, "Number of blocks: %d\n", numblocks); 68862306a36Sopenharmony_ci 68962306a36Sopenharmony_ci dev_info(&card->dev->dev, "Changing FPGA to Update mode\n"); 69062306a36Sopenharmony_ci iowrite32(1, card->config_regs + FPGA_MODE); 69162306a36Sopenharmony_ci (void) ioread32(card->config_regs + FPGA_MODE); 69262306a36Sopenharmony_ci 69362306a36Sopenharmony_ci /* Set mode to Chip Erase */ 69462306a36Sopenharmony_ci if(chip == 0 || chip == 2) 69562306a36Sopenharmony_ci dev_info(&card->dev->dev, "Set FPGA Flash mode to FPGA Chip Erase\n"); 69662306a36Sopenharmony_ci if(chip == 1 || chip == 3) 69762306a36Sopenharmony_ci dev_info(&card->dev->dev, "Set FPGA Flash mode to Solos Chip Erase\n"); 69862306a36Sopenharmony_ci iowrite32((chip * 2), card->config_regs + FLASH_MODE); 69962306a36Sopenharmony_ci 70062306a36Sopenharmony_ci 70162306a36Sopenharmony_ci iowrite32(1, card->config_regs + WRITE_FLASH); 70262306a36Sopenharmony_ci wait_event(card->fw_wq, !ioread32(card->config_regs + FLASH_BUSY)); 70362306a36Sopenharmony_ci 70462306a36Sopenharmony_ci for (offset = 0; offset < fw->size; offset += blocksize) { 70562306a36Sopenharmony_ci int i; 70662306a36Sopenharmony_ci 70762306a36Sopenharmony_ci /* Clear write flag */ 70862306a36Sopenharmony_ci iowrite32(0, card->config_regs + WRITE_FLASH); 70962306a36Sopenharmony_ci 71062306a36Sopenharmony_ci /* Set mode to Block Write */ 71162306a36Sopenharmony_ci /* dev_info(&card->dev->dev, "Set FPGA Flash mode to Block Write\n"); */ 71262306a36Sopenharmony_ci iowrite32(((chip * 2) + 1), card->config_regs + FLASH_MODE); 71362306a36Sopenharmony_ci 71462306a36Sopenharmony_ci /* Copy block to buffer, swapping each 16 bits for Atmel flash */ 71562306a36Sopenharmony_ci for(i = 0; i < blocksize; i += 4) { 71662306a36Sopenharmony_ci uint32_t word; 71762306a36Sopenharmony_ci if (card->atmel_flash) 71862306a36Sopenharmony_ci word = swahb32p((uint32_t *)(fw->data + offset + i)); 71962306a36Sopenharmony_ci else 72062306a36Sopenharmony_ci word = *(uint32_t *)(fw->data + offset + i); 72162306a36Sopenharmony_ci if(card->fpga_version > LEGACY_BUFFERS) 72262306a36Sopenharmony_ci iowrite32(word, FLASH_BUF + i); 72362306a36Sopenharmony_ci else 72462306a36Sopenharmony_ci iowrite32(word, RX_BUF(card, 3) + i); 72562306a36Sopenharmony_ci } 72662306a36Sopenharmony_ci 72762306a36Sopenharmony_ci /* Specify block number and then trigger flash write */ 72862306a36Sopenharmony_ci iowrite32(offset / blocksize, card->config_regs + FLASH_BLOCK); 72962306a36Sopenharmony_ci iowrite32(1, card->config_regs + WRITE_FLASH); 73062306a36Sopenharmony_ci wait_event(card->fw_wq, !ioread32(card->config_regs + FLASH_BUSY)); 73162306a36Sopenharmony_ci } 73262306a36Sopenharmony_ci 73362306a36Sopenharmony_ci release_firmware(fw); 73462306a36Sopenharmony_ci iowrite32(0, card->config_regs + WRITE_FLASH); 73562306a36Sopenharmony_ci iowrite32(0, card->config_regs + FPGA_MODE); 73662306a36Sopenharmony_ci iowrite32(0, card->config_regs + FLASH_MODE); 73762306a36Sopenharmony_ci dev_info(&card->dev->dev, "Returning FPGA to Data mode\n"); 73862306a36Sopenharmony_ci return 0; 73962306a36Sopenharmony_ci} 74062306a36Sopenharmony_ci 74162306a36Sopenharmony_cistatic irqreturn_t solos_irq(int irq, void *dev_id) 74262306a36Sopenharmony_ci{ 74362306a36Sopenharmony_ci struct solos_card *card = dev_id; 74462306a36Sopenharmony_ci int handled = 1; 74562306a36Sopenharmony_ci 74662306a36Sopenharmony_ci iowrite32(0, card->config_regs + IRQ_CLEAR); 74762306a36Sopenharmony_ci 74862306a36Sopenharmony_ci /* If we're up and running, just kick the tasklet to process TX/RX */ 74962306a36Sopenharmony_ci if (card->atmdev[0]) 75062306a36Sopenharmony_ci tasklet_schedule(&card->tlet); 75162306a36Sopenharmony_ci else 75262306a36Sopenharmony_ci wake_up(&card->fw_wq); 75362306a36Sopenharmony_ci 75462306a36Sopenharmony_ci return IRQ_RETVAL(handled); 75562306a36Sopenharmony_ci} 75662306a36Sopenharmony_ci 75762306a36Sopenharmony_cistatic void solos_bh(unsigned long card_arg) 75862306a36Sopenharmony_ci{ 75962306a36Sopenharmony_ci struct solos_card *card = (void *)card_arg; 76062306a36Sopenharmony_ci uint32_t card_flags; 76162306a36Sopenharmony_ci uint32_t rx_done = 0; 76262306a36Sopenharmony_ci int port; 76362306a36Sopenharmony_ci 76462306a36Sopenharmony_ci /* 76562306a36Sopenharmony_ci * Since fpga_tx() is going to need to read the flags under its lock, 76662306a36Sopenharmony_ci * it can return them to us so that we don't have to hit PCI MMIO 76762306a36Sopenharmony_ci * again for the same information 76862306a36Sopenharmony_ci */ 76962306a36Sopenharmony_ci card_flags = fpga_tx(card); 77062306a36Sopenharmony_ci 77162306a36Sopenharmony_ci for (port = 0; port < card->nr_ports; port++) { 77262306a36Sopenharmony_ci if (card_flags & (0x10 << port)) { 77362306a36Sopenharmony_ci struct pkt_hdr _hdr, *header; 77462306a36Sopenharmony_ci struct sk_buff *skb; 77562306a36Sopenharmony_ci struct atm_vcc *vcc; 77662306a36Sopenharmony_ci int size; 77762306a36Sopenharmony_ci 77862306a36Sopenharmony_ci if (card->using_dma) { 77962306a36Sopenharmony_ci skb = card->rx_skb[port]; 78062306a36Sopenharmony_ci card->rx_skb[port] = NULL; 78162306a36Sopenharmony_ci 78262306a36Sopenharmony_ci dma_unmap_single(&card->dev->dev, SKB_CB(skb)->dma_addr, 78362306a36Sopenharmony_ci RX_DMA_SIZE, DMA_FROM_DEVICE); 78462306a36Sopenharmony_ci 78562306a36Sopenharmony_ci header = (void *)skb->data; 78662306a36Sopenharmony_ci size = le16_to_cpu(header->size); 78762306a36Sopenharmony_ci skb_put(skb, size + sizeof(*header)); 78862306a36Sopenharmony_ci skb_pull(skb, sizeof(*header)); 78962306a36Sopenharmony_ci } else { 79062306a36Sopenharmony_ci header = &_hdr; 79162306a36Sopenharmony_ci 79262306a36Sopenharmony_ci rx_done |= 0x10 << port; 79362306a36Sopenharmony_ci 79462306a36Sopenharmony_ci memcpy_fromio(header, RX_BUF(card, port), sizeof(*header)); 79562306a36Sopenharmony_ci 79662306a36Sopenharmony_ci size = le16_to_cpu(header->size); 79762306a36Sopenharmony_ci if (size > (card->buffer_size - sizeof(*header))){ 79862306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Invalid buffer size\n"); 79962306a36Sopenharmony_ci continue; 80062306a36Sopenharmony_ci } 80162306a36Sopenharmony_ci 80262306a36Sopenharmony_ci /* Use netdev_alloc_skb() because it adds NET_SKB_PAD of 80362306a36Sopenharmony_ci * headroom, and ensures we can route packets back out an 80462306a36Sopenharmony_ci * Ethernet interface (for example) without having to 80562306a36Sopenharmony_ci * reallocate. Adding NET_IP_ALIGN also ensures that both 80662306a36Sopenharmony_ci * PPPoATM and PPPoEoBR2684 packets end up aligned. */ 80762306a36Sopenharmony_ci skb = netdev_alloc_skb_ip_align(NULL, size + 1); 80862306a36Sopenharmony_ci if (!skb) { 80962306a36Sopenharmony_ci if (net_ratelimit()) 81062306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff for RX\n"); 81162306a36Sopenharmony_ci continue; 81262306a36Sopenharmony_ci } 81362306a36Sopenharmony_ci 81462306a36Sopenharmony_ci memcpy_fromio(skb_put(skb, size), 81562306a36Sopenharmony_ci RX_BUF(card, port) + sizeof(*header), 81662306a36Sopenharmony_ci size); 81762306a36Sopenharmony_ci } 81862306a36Sopenharmony_ci if (atmdebug) { 81962306a36Sopenharmony_ci dev_info(&card->dev->dev, "Received: port %d\n", port); 82062306a36Sopenharmony_ci dev_info(&card->dev->dev, "size: %d VPI: %d VCI: %d\n", 82162306a36Sopenharmony_ci size, le16_to_cpu(header->vpi), 82262306a36Sopenharmony_ci le16_to_cpu(header->vci)); 82362306a36Sopenharmony_ci print_buffer(skb); 82462306a36Sopenharmony_ci } 82562306a36Sopenharmony_ci 82662306a36Sopenharmony_ci switch (le16_to_cpu(header->type)) { 82762306a36Sopenharmony_ci case PKT_DATA: 82862306a36Sopenharmony_ci vcc = find_vcc(card->atmdev[port], le16_to_cpu(header->vpi), 82962306a36Sopenharmony_ci le16_to_cpu(header->vci)); 83062306a36Sopenharmony_ci if (!vcc) { 83162306a36Sopenharmony_ci if (net_ratelimit()) 83262306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Received packet for unknown VPI.VCI %d.%d on port %d\n", 83362306a36Sopenharmony_ci le16_to_cpu(header->vpi), le16_to_cpu(header->vci), 83462306a36Sopenharmony_ci port); 83562306a36Sopenharmony_ci dev_kfree_skb_any(skb); 83662306a36Sopenharmony_ci break; 83762306a36Sopenharmony_ci } 83862306a36Sopenharmony_ci atm_charge(vcc, skb->truesize); 83962306a36Sopenharmony_ci vcc->push(vcc, skb); 84062306a36Sopenharmony_ci atomic_inc(&vcc->stats->rx); 84162306a36Sopenharmony_ci break; 84262306a36Sopenharmony_ci 84362306a36Sopenharmony_ci case PKT_STATUS: 84462306a36Sopenharmony_ci if (process_status(card, port, skb) && 84562306a36Sopenharmony_ci net_ratelimit()) { 84662306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Bad status packet of %d bytes on port %d:\n", skb->len, port); 84762306a36Sopenharmony_ci print_buffer(skb); 84862306a36Sopenharmony_ci } 84962306a36Sopenharmony_ci dev_kfree_skb_any(skb); 85062306a36Sopenharmony_ci break; 85162306a36Sopenharmony_ci 85262306a36Sopenharmony_ci case PKT_COMMAND: 85362306a36Sopenharmony_ci default: /* FIXME: Not really, surely? */ 85462306a36Sopenharmony_ci if (process_command(card, port, skb)) 85562306a36Sopenharmony_ci break; 85662306a36Sopenharmony_ci spin_lock(&card->cli_queue_lock); 85762306a36Sopenharmony_ci if (skb_queue_len(&card->cli_queue[port]) > 10) { 85862306a36Sopenharmony_ci if (net_ratelimit()) 85962306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Dropping console response on port %d\n", 86062306a36Sopenharmony_ci port); 86162306a36Sopenharmony_ci dev_kfree_skb_any(skb); 86262306a36Sopenharmony_ci } else 86362306a36Sopenharmony_ci skb_queue_tail(&card->cli_queue[port], skb); 86462306a36Sopenharmony_ci spin_unlock(&card->cli_queue_lock); 86562306a36Sopenharmony_ci break; 86662306a36Sopenharmony_ci } 86762306a36Sopenharmony_ci } 86862306a36Sopenharmony_ci /* Allocate RX skbs for any ports which need them */ 86962306a36Sopenharmony_ci if (card->using_dma && card->atmdev[port] && 87062306a36Sopenharmony_ci !card->rx_skb[port]) { 87162306a36Sopenharmony_ci /* Unlike the MMIO case (qv) we can't add NET_IP_ALIGN 87262306a36Sopenharmony_ci * here; the FPGA can only DMA to addresses which are 87362306a36Sopenharmony_ci * aligned to 4 bytes. */ 87462306a36Sopenharmony_ci struct sk_buff *skb = dev_alloc_skb(RX_DMA_SIZE); 87562306a36Sopenharmony_ci if (skb) { 87662306a36Sopenharmony_ci SKB_CB(skb)->dma_addr = 87762306a36Sopenharmony_ci dma_map_single(&card->dev->dev, skb->data, 87862306a36Sopenharmony_ci RX_DMA_SIZE, DMA_FROM_DEVICE); 87962306a36Sopenharmony_ci iowrite32(SKB_CB(skb)->dma_addr, 88062306a36Sopenharmony_ci card->config_regs + RX_DMA_ADDR(port)); 88162306a36Sopenharmony_ci card->rx_skb[port] = skb; 88262306a36Sopenharmony_ci } else { 88362306a36Sopenharmony_ci if (net_ratelimit()) 88462306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate RX skb"); 88562306a36Sopenharmony_ci 88662306a36Sopenharmony_ci /* We'll have to try again later */ 88762306a36Sopenharmony_ci tasklet_schedule(&card->tlet); 88862306a36Sopenharmony_ci } 88962306a36Sopenharmony_ci } 89062306a36Sopenharmony_ci } 89162306a36Sopenharmony_ci if (rx_done) 89262306a36Sopenharmony_ci iowrite32(rx_done, card->config_regs + FLAGS_ADDR); 89362306a36Sopenharmony_ci 89462306a36Sopenharmony_ci return; 89562306a36Sopenharmony_ci} 89662306a36Sopenharmony_ci 89762306a36Sopenharmony_cistatic struct atm_vcc *find_vcc(struct atm_dev *dev, short vpi, int vci) 89862306a36Sopenharmony_ci{ 89962306a36Sopenharmony_ci struct hlist_head *head; 90062306a36Sopenharmony_ci struct atm_vcc *vcc = NULL; 90162306a36Sopenharmony_ci struct sock *s; 90262306a36Sopenharmony_ci 90362306a36Sopenharmony_ci read_lock(&vcc_sklist_lock); 90462306a36Sopenharmony_ci head = &vcc_hash[vci & (VCC_HTABLE_SIZE -1)]; 90562306a36Sopenharmony_ci sk_for_each(s, head) { 90662306a36Sopenharmony_ci vcc = atm_sk(s); 90762306a36Sopenharmony_ci if (vcc->dev == dev && vcc->vci == vci && 90862306a36Sopenharmony_ci vcc->vpi == vpi && vcc->qos.rxtp.traffic_class != ATM_NONE && 90962306a36Sopenharmony_ci test_bit(ATM_VF_READY, &vcc->flags)) 91062306a36Sopenharmony_ci goto out; 91162306a36Sopenharmony_ci } 91262306a36Sopenharmony_ci vcc = NULL; 91362306a36Sopenharmony_ci out: 91462306a36Sopenharmony_ci read_unlock(&vcc_sklist_lock); 91562306a36Sopenharmony_ci return vcc; 91662306a36Sopenharmony_ci} 91762306a36Sopenharmony_ci 91862306a36Sopenharmony_cistatic int popen(struct atm_vcc *vcc) 91962306a36Sopenharmony_ci{ 92062306a36Sopenharmony_ci struct solos_card *card = vcc->dev->dev_data; 92162306a36Sopenharmony_ci struct sk_buff *skb; 92262306a36Sopenharmony_ci struct pkt_hdr *header; 92362306a36Sopenharmony_ci 92462306a36Sopenharmony_ci if (vcc->qos.aal != ATM_AAL5) { 92562306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Unsupported ATM type %d\n", 92662306a36Sopenharmony_ci vcc->qos.aal); 92762306a36Sopenharmony_ci return -EINVAL; 92862306a36Sopenharmony_ci } 92962306a36Sopenharmony_ci 93062306a36Sopenharmony_ci skb = alloc_skb(sizeof(*header), GFP_KERNEL); 93162306a36Sopenharmony_ci if (!skb) { 93262306a36Sopenharmony_ci if (net_ratelimit()) 93362306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff in popen()\n"); 93462306a36Sopenharmony_ci return -ENOMEM; 93562306a36Sopenharmony_ci } 93662306a36Sopenharmony_ci header = skb_put(skb, sizeof(*header)); 93762306a36Sopenharmony_ci 93862306a36Sopenharmony_ci header->size = cpu_to_le16(0); 93962306a36Sopenharmony_ci header->vpi = cpu_to_le16(vcc->vpi); 94062306a36Sopenharmony_ci header->vci = cpu_to_le16(vcc->vci); 94162306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_POPEN); 94262306a36Sopenharmony_ci 94362306a36Sopenharmony_ci fpga_queue(card, SOLOS_CHAN(vcc->dev), skb, NULL); 94462306a36Sopenharmony_ci 94562306a36Sopenharmony_ci set_bit(ATM_VF_ADDR, &vcc->flags); 94662306a36Sopenharmony_ci set_bit(ATM_VF_READY, &vcc->flags); 94762306a36Sopenharmony_ci 94862306a36Sopenharmony_ci return 0; 94962306a36Sopenharmony_ci} 95062306a36Sopenharmony_ci 95162306a36Sopenharmony_cistatic void pclose(struct atm_vcc *vcc) 95262306a36Sopenharmony_ci{ 95362306a36Sopenharmony_ci struct solos_card *card = vcc->dev->dev_data; 95462306a36Sopenharmony_ci unsigned char port = SOLOS_CHAN(vcc->dev); 95562306a36Sopenharmony_ci struct sk_buff *skb, *tmpskb; 95662306a36Sopenharmony_ci struct pkt_hdr *header; 95762306a36Sopenharmony_ci 95862306a36Sopenharmony_ci /* Remove any yet-to-be-transmitted packets from the pending queue */ 95962306a36Sopenharmony_ci spin_lock_bh(&card->tx_queue_lock); 96062306a36Sopenharmony_ci skb_queue_walk_safe(&card->tx_queue[port], skb, tmpskb) { 96162306a36Sopenharmony_ci if (SKB_CB(skb)->vcc == vcc) { 96262306a36Sopenharmony_ci skb_unlink(skb, &card->tx_queue[port]); 96362306a36Sopenharmony_ci solos_pop(vcc, skb); 96462306a36Sopenharmony_ci } 96562306a36Sopenharmony_ci } 96662306a36Sopenharmony_ci spin_unlock_bh(&card->tx_queue_lock); 96762306a36Sopenharmony_ci 96862306a36Sopenharmony_ci skb = alloc_skb(sizeof(*header), GFP_KERNEL); 96962306a36Sopenharmony_ci if (!skb) { 97062306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff in pclose()\n"); 97162306a36Sopenharmony_ci return; 97262306a36Sopenharmony_ci } 97362306a36Sopenharmony_ci header = skb_put(skb, sizeof(*header)); 97462306a36Sopenharmony_ci 97562306a36Sopenharmony_ci header->size = cpu_to_le16(0); 97662306a36Sopenharmony_ci header->vpi = cpu_to_le16(vcc->vpi); 97762306a36Sopenharmony_ci header->vci = cpu_to_le16(vcc->vci); 97862306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_PCLOSE); 97962306a36Sopenharmony_ci 98062306a36Sopenharmony_ci skb_get(skb); 98162306a36Sopenharmony_ci fpga_queue(card, port, skb, NULL); 98262306a36Sopenharmony_ci 98362306a36Sopenharmony_ci if (!wait_event_timeout(card->param_wq, !skb_shared(skb), 5 * HZ)) 98462306a36Sopenharmony_ci dev_warn(&card->dev->dev, 98562306a36Sopenharmony_ci "Timeout waiting for VCC close on port %d\n", port); 98662306a36Sopenharmony_ci 98762306a36Sopenharmony_ci dev_kfree_skb(skb); 98862306a36Sopenharmony_ci 98962306a36Sopenharmony_ci /* Hold up vcc_destroy_socket() (our caller) until solos_bh() in the 99062306a36Sopenharmony_ci tasklet has finished processing any incoming packets (and, more to 99162306a36Sopenharmony_ci the point, using the vcc pointer). */ 99262306a36Sopenharmony_ci tasklet_unlock_wait(&card->tlet); 99362306a36Sopenharmony_ci 99462306a36Sopenharmony_ci clear_bit(ATM_VF_ADDR, &vcc->flags); 99562306a36Sopenharmony_ci 99662306a36Sopenharmony_ci return; 99762306a36Sopenharmony_ci} 99862306a36Sopenharmony_ci 99962306a36Sopenharmony_cistatic int print_buffer(struct sk_buff *buf) 100062306a36Sopenharmony_ci{ 100162306a36Sopenharmony_ci int len,i; 100262306a36Sopenharmony_ci char msg[500]; 100362306a36Sopenharmony_ci char item[10]; 100462306a36Sopenharmony_ci 100562306a36Sopenharmony_ci len = buf->len; 100662306a36Sopenharmony_ci for (i = 0; i < len; i++){ 100762306a36Sopenharmony_ci if(i % 8 == 0) 100862306a36Sopenharmony_ci sprintf(msg, "%02X: ", i); 100962306a36Sopenharmony_ci 101062306a36Sopenharmony_ci sprintf(item,"%02X ",*(buf->data + i)); 101162306a36Sopenharmony_ci strcat(msg, item); 101262306a36Sopenharmony_ci if(i % 8 == 7) { 101362306a36Sopenharmony_ci sprintf(item, "\n"); 101462306a36Sopenharmony_ci strcat(msg, item); 101562306a36Sopenharmony_ci printk(KERN_DEBUG "%s", msg); 101662306a36Sopenharmony_ci } 101762306a36Sopenharmony_ci } 101862306a36Sopenharmony_ci if (i % 8 != 0) { 101962306a36Sopenharmony_ci sprintf(item, "\n"); 102062306a36Sopenharmony_ci strcat(msg, item); 102162306a36Sopenharmony_ci printk(KERN_DEBUG "%s", msg); 102262306a36Sopenharmony_ci } 102362306a36Sopenharmony_ci printk(KERN_DEBUG "\n"); 102462306a36Sopenharmony_ci 102562306a36Sopenharmony_ci return 0; 102662306a36Sopenharmony_ci} 102762306a36Sopenharmony_ci 102862306a36Sopenharmony_cistatic void fpga_queue(struct solos_card *card, int port, struct sk_buff *skb, 102962306a36Sopenharmony_ci struct atm_vcc *vcc) 103062306a36Sopenharmony_ci{ 103162306a36Sopenharmony_ci int old_len; 103262306a36Sopenharmony_ci unsigned long flags; 103362306a36Sopenharmony_ci 103462306a36Sopenharmony_ci SKB_CB(skb)->vcc = vcc; 103562306a36Sopenharmony_ci 103662306a36Sopenharmony_ci spin_lock_irqsave(&card->tx_queue_lock, flags); 103762306a36Sopenharmony_ci old_len = skb_queue_len(&card->tx_queue[port]); 103862306a36Sopenharmony_ci skb_queue_tail(&card->tx_queue[port], skb); 103962306a36Sopenharmony_ci if (!old_len) 104062306a36Sopenharmony_ci card->tx_mask |= (1 << port); 104162306a36Sopenharmony_ci spin_unlock_irqrestore(&card->tx_queue_lock, flags); 104262306a36Sopenharmony_ci 104362306a36Sopenharmony_ci /* Theoretically we could just schedule the tasklet here, but 104462306a36Sopenharmony_ci that introduces latency we don't want -- it's noticeable */ 104562306a36Sopenharmony_ci if (!old_len) 104662306a36Sopenharmony_ci fpga_tx(card); 104762306a36Sopenharmony_ci} 104862306a36Sopenharmony_ci 104962306a36Sopenharmony_cistatic uint32_t fpga_tx(struct solos_card *card) 105062306a36Sopenharmony_ci{ 105162306a36Sopenharmony_ci uint32_t tx_pending, card_flags; 105262306a36Sopenharmony_ci uint32_t tx_started = 0; 105362306a36Sopenharmony_ci struct sk_buff *skb; 105462306a36Sopenharmony_ci struct atm_vcc *vcc; 105562306a36Sopenharmony_ci unsigned char port; 105662306a36Sopenharmony_ci unsigned long flags; 105762306a36Sopenharmony_ci 105862306a36Sopenharmony_ci spin_lock_irqsave(&card->tx_lock, flags); 105962306a36Sopenharmony_ci 106062306a36Sopenharmony_ci card_flags = ioread32(card->config_regs + FLAGS_ADDR); 106162306a36Sopenharmony_ci /* 106262306a36Sopenharmony_ci * The queue lock is required for _writing_ to tx_mask, but we're 106362306a36Sopenharmony_ci * OK to read it here without locking. The only potential update 106462306a36Sopenharmony_ci * that we could race with is in fpga_queue() where it sets a bit 106562306a36Sopenharmony_ci * for a new port... but it's going to call this function again if 106662306a36Sopenharmony_ci * it's doing that, anyway. 106762306a36Sopenharmony_ci */ 106862306a36Sopenharmony_ci tx_pending = card->tx_mask & ~card_flags; 106962306a36Sopenharmony_ci 107062306a36Sopenharmony_ci for (port = 0; tx_pending; tx_pending >>= 1, port++) { 107162306a36Sopenharmony_ci if (tx_pending & 1) { 107262306a36Sopenharmony_ci struct sk_buff *oldskb = card->tx_skb[port]; 107362306a36Sopenharmony_ci if (oldskb) { 107462306a36Sopenharmony_ci dma_unmap_single(&card->dev->dev, SKB_CB(oldskb)->dma_addr, 107562306a36Sopenharmony_ci oldskb->len, DMA_TO_DEVICE); 107662306a36Sopenharmony_ci card->tx_skb[port] = NULL; 107762306a36Sopenharmony_ci } 107862306a36Sopenharmony_ci spin_lock(&card->tx_queue_lock); 107962306a36Sopenharmony_ci skb = skb_dequeue(&card->tx_queue[port]); 108062306a36Sopenharmony_ci if (!skb) 108162306a36Sopenharmony_ci card->tx_mask &= ~(1 << port); 108262306a36Sopenharmony_ci spin_unlock(&card->tx_queue_lock); 108362306a36Sopenharmony_ci 108462306a36Sopenharmony_ci if (skb && !card->using_dma) { 108562306a36Sopenharmony_ci memcpy_toio(TX_BUF(card, port), skb->data, skb->len); 108662306a36Sopenharmony_ci tx_started |= 1 << port; 108762306a36Sopenharmony_ci oldskb = skb; /* We're done with this skb already */ 108862306a36Sopenharmony_ci } else if (skb && card->using_dma) { 108962306a36Sopenharmony_ci unsigned char *data = skb->data; 109062306a36Sopenharmony_ci if ((unsigned long)data & card->dma_alignment) { 109162306a36Sopenharmony_ci data = card->dma_bounce + (BUF_SIZE * port); 109262306a36Sopenharmony_ci memcpy(data, skb->data, skb->len); 109362306a36Sopenharmony_ci } 109462306a36Sopenharmony_ci SKB_CB(skb)->dma_addr = dma_map_single(&card->dev->dev, data, 109562306a36Sopenharmony_ci skb->len, DMA_TO_DEVICE); 109662306a36Sopenharmony_ci card->tx_skb[port] = skb; 109762306a36Sopenharmony_ci iowrite32(SKB_CB(skb)->dma_addr, 109862306a36Sopenharmony_ci card->config_regs + TX_DMA_ADDR(port)); 109962306a36Sopenharmony_ci } 110062306a36Sopenharmony_ci 110162306a36Sopenharmony_ci if (!oldskb) 110262306a36Sopenharmony_ci continue; 110362306a36Sopenharmony_ci 110462306a36Sopenharmony_ci /* Clean up and free oldskb now it's gone */ 110562306a36Sopenharmony_ci if (atmdebug) { 110662306a36Sopenharmony_ci struct pkt_hdr *header = (void *)oldskb->data; 110762306a36Sopenharmony_ci int size = le16_to_cpu(header->size); 110862306a36Sopenharmony_ci 110962306a36Sopenharmony_ci skb_pull(oldskb, sizeof(*header)); 111062306a36Sopenharmony_ci dev_info(&card->dev->dev, "Transmitted: port %d\n", 111162306a36Sopenharmony_ci port); 111262306a36Sopenharmony_ci dev_info(&card->dev->dev, "size: %d VPI: %d VCI: %d\n", 111362306a36Sopenharmony_ci size, le16_to_cpu(header->vpi), 111462306a36Sopenharmony_ci le16_to_cpu(header->vci)); 111562306a36Sopenharmony_ci print_buffer(oldskb); 111662306a36Sopenharmony_ci } 111762306a36Sopenharmony_ci 111862306a36Sopenharmony_ci vcc = SKB_CB(oldskb)->vcc; 111962306a36Sopenharmony_ci 112062306a36Sopenharmony_ci if (vcc) { 112162306a36Sopenharmony_ci atomic_inc(&vcc->stats->tx); 112262306a36Sopenharmony_ci solos_pop(vcc, oldskb); 112362306a36Sopenharmony_ci } else { 112462306a36Sopenharmony_ci dev_kfree_skb_irq(oldskb); 112562306a36Sopenharmony_ci wake_up(&card->param_wq); 112662306a36Sopenharmony_ci } 112762306a36Sopenharmony_ci } 112862306a36Sopenharmony_ci } 112962306a36Sopenharmony_ci /* For non-DMA TX, write the 'TX start' bit for all four ports simultaneously */ 113062306a36Sopenharmony_ci if (tx_started) 113162306a36Sopenharmony_ci iowrite32(tx_started, card->config_regs + FLAGS_ADDR); 113262306a36Sopenharmony_ci 113362306a36Sopenharmony_ci spin_unlock_irqrestore(&card->tx_lock, flags); 113462306a36Sopenharmony_ci return card_flags; 113562306a36Sopenharmony_ci} 113662306a36Sopenharmony_ci 113762306a36Sopenharmony_cistatic int psend(struct atm_vcc *vcc, struct sk_buff *skb) 113862306a36Sopenharmony_ci{ 113962306a36Sopenharmony_ci struct solos_card *card = vcc->dev->dev_data; 114062306a36Sopenharmony_ci struct pkt_hdr *header; 114162306a36Sopenharmony_ci int pktlen; 114262306a36Sopenharmony_ci 114362306a36Sopenharmony_ci pktlen = skb->len; 114462306a36Sopenharmony_ci if (pktlen > (BUF_SIZE - sizeof(*header))) { 114562306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Length of PDU is too large. Dropping PDU.\n"); 114662306a36Sopenharmony_ci solos_pop(vcc, skb); 114762306a36Sopenharmony_ci return 0; 114862306a36Sopenharmony_ci } 114962306a36Sopenharmony_ci 115062306a36Sopenharmony_ci if (!skb_clone_writable(skb, sizeof(*header))) { 115162306a36Sopenharmony_ci int expand_by = 0; 115262306a36Sopenharmony_ci int ret; 115362306a36Sopenharmony_ci 115462306a36Sopenharmony_ci if (skb_headroom(skb) < sizeof(*header)) 115562306a36Sopenharmony_ci expand_by = sizeof(*header) - skb_headroom(skb); 115662306a36Sopenharmony_ci 115762306a36Sopenharmony_ci ret = pskb_expand_head(skb, expand_by, 0, GFP_ATOMIC); 115862306a36Sopenharmony_ci if (ret) { 115962306a36Sopenharmony_ci dev_warn(&card->dev->dev, "pskb_expand_head failed.\n"); 116062306a36Sopenharmony_ci solos_pop(vcc, skb); 116162306a36Sopenharmony_ci return ret; 116262306a36Sopenharmony_ci } 116362306a36Sopenharmony_ci } 116462306a36Sopenharmony_ci 116562306a36Sopenharmony_ci header = skb_push(skb, sizeof(*header)); 116662306a36Sopenharmony_ci 116762306a36Sopenharmony_ci /* This does _not_ include the size of the header */ 116862306a36Sopenharmony_ci header->size = cpu_to_le16(pktlen); 116962306a36Sopenharmony_ci header->vpi = cpu_to_le16(vcc->vpi); 117062306a36Sopenharmony_ci header->vci = cpu_to_le16(vcc->vci); 117162306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_DATA); 117262306a36Sopenharmony_ci 117362306a36Sopenharmony_ci fpga_queue(card, SOLOS_CHAN(vcc->dev), skb, vcc); 117462306a36Sopenharmony_ci 117562306a36Sopenharmony_ci return 0; 117662306a36Sopenharmony_ci} 117762306a36Sopenharmony_ci 117862306a36Sopenharmony_cistatic const struct atmdev_ops fpga_ops = { 117962306a36Sopenharmony_ci .open = popen, 118062306a36Sopenharmony_ci .close = pclose, 118162306a36Sopenharmony_ci .ioctl = NULL, 118262306a36Sopenharmony_ci .send = psend, 118362306a36Sopenharmony_ci .send_oam = NULL, 118462306a36Sopenharmony_ci .phy_put = NULL, 118562306a36Sopenharmony_ci .phy_get = NULL, 118662306a36Sopenharmony_ci .change_qos = NULL, 118762306a36Sopenharmony_ci .proc_read = NULL, 118862306a36Sopenharmony_ci .owner = THIS_MODULE 118962306a36Sopenharmony_ci}; 119062306a36Sopenharmony_ci 119162306a36Sopenharmony_cistatic int fpga_probe(struct pci_dev *dev, const struct pci_device_id *id) 119262306a36Sopenharmony_ci{ 119362306a36Sopenharmony_ci int err; 119462306a36Sopenharmony_ci uint16_t fpga_ver; 119562306a36Sopenharmony_ci uint8_t major_ver, minor_ver; 119662306a36Sopenharmony_ci uint32_t data32; 119762306a36Sopenharmony_ci struct solos_card *card; 119862306a36Sopenharmony_ci 119962306a36Sopenharmony_ci card = kzalloc(sizeof(*card), GFP_KERNEL); 120062306a36Sopenharmony_ci if (!card) 120162306a36Sopenharmony_ci return -ENOMEM; 120262306a36Sopenharmony_ci 120362306a36Sopenharmony_ci card->dev = dev; 120462306a36Sopenharmony_ci init_waitqueue_head(&card->fw_wq); 120562306a36Sopenharmony_ci init_waitqueue_head(&card->param_wq); 120662306a36Sopenharmony_ci 120762306a36Sopenharmony_ci err = pci_enable_device(dev); 120862306a36Sopenharmony_ci if (err) { 120962306a36Sopenharmony_ci dev_warn(&dev->dev, "Failed to enable PCI device\n"); 121062306a36Sopenharmony_ci goto out; 121162306a36Sopenharmony_ci } 121262306a36Sopenharmony_ci 121362306a36Sopenharmony_ci err = dma_set_mask_and_coherent(&dev->dev, DMA_BIT_MASK(32)); 121462306a36Sopenharmony_ci if (err) { 121562306a36Sopenharmony_ci dev_warn(&dev->dev, "Failed to set 32-bit DMA mask\n"); 121662306a36Sopenharmony_ci goto out; 121762306a36Sopenharmony_ci } 121862306a36Sopenharmony_ci 121962306a36Sopenharmony_ci err = pci_request_regions(dev, "solos"); 122062306a36Sopenharmony_ci if (err) { 122162306a36Sopenharmony_ci dev_warn(&dev->dev, "Failed to request regions\n"); 122262306a36Sopenharmony_ci goto out; 122362306a36Sopenharmony_ci } 122462306a36Sopenharmony_ci 122562306a36Sopenharmony_ci card->config_regs = pci_iomap(dev, 0, CONFIG_RAM_SIZE); 122662306a36Sopenharmony_ci if (!card->config_regs) { 122762306a36Sopenharmony_ci dev_warn(&dev->dev, "Failed to ioremap config registers\n"); 122862306a36Sopenharmony_ci err = -ENOMEM; 122962306a36Sopenharmony_ci goto out_release_regions; 123062306a36Sopenharmony_ci } 123162306a36Sopenharmony_ci card->buffers = pci_iomap(dev, 1, DATA_RAM_SIZE); 123262306a36Sopenharmony_ci if (!card->buffers) { 123362306a36Sopenharmony_ci dev_warn(&dev->dev, "Failed to ioremap data buffers\n"); 123462306a36Sopenharmony_ci err = -ENOMEM; 123562306a36Sopenharmony_ci goto out_unmap_config; 123662306a36Sopenharmony_ci } 123762306a36Sopenharmony_ci 123862306a36Sopenharmony_ci if (reset) { 123962306a36Sopenharmony_ci iowrite32(1, card->config_regs + FPGA_MODE); 124062306a36Sopenharmony_ci ioread32(card->config_regs + FPGA_MODE); 124162306a36Sopenharmony_ci 124262306a36Sopenharmony_ci iowrite32(0, card->config_regs + FPGA_MODE); 124362306a36Sopenharmony_ci ioread32(card->config_regs + FPGA_MODE); 124462306a36Sopenharmony_ci } 124562306a36Sopenharmony_ci 124662306a36Sopenharmony_ci data32 = ioread32(card->config_regs + FPGA_VER); 124762306a36Sopenharmony_ci fpga_ver = (data32 & 0x0000FFFF); 124862306a36Sopenharmony_ci major_ver = ((data32 & 0xFF000000) >> 24); 124962306a36Sopenharmony_ci minor_ver = ((data32 & 0x00FF0000) >> 16); 125062306a36Sopenharmony_ci card->fpga_version = FPGA_VERSION(major_ver,minor_ver); 125162306a36Sopenharmony_ci if (card->fpga_version > LEGACY_BUFFERS) 125262306a36Sopenharmony_ci card->buffer_size = BUF_SIZE; 125362306a36Sopenharmony_ci else 125462306a36Sopenharmony_ci card->buffer_size = OLD_BUF_SIZE; 125562306a36Sopenharmony_ci dev_info(&dev->dev, "Solos FPGA Version %d.%02d svn-%d\n", 125662306a36Sopenharmony_ci major_ver, minor_ver, fpga_ver); 125762306a36Sopenharmony_ci 125862306a36Sopenharmony_ci if (fpga_ver < 37 && (fpga_upgrade || firmware_upgrade || 125962306a36Sopenharmony_ci db_fpga_upgrade || db_firmware_upgrade)) { 126062306a36Sopenharmony_ci dev_warn(&dev->dev, 126162306a36Sopenharmony_ci "FPGA too old; cannot upgrade flash. Use JTAG.\n"); 126262306a36Sopenharmony_ci fpga_upgrade = firmware_upgrade = 0; 126362306a36Sopenharmony_ci db_fpga_upgrade = db_firmware_upgrade = 0; 126462306a36Sopenharmony_ci } 126562306a36Sopenharmony_ci 126662306a36Sopenharmony_ci /* Stopped using Atmel flash after 0.03-38 */ 126762306a36Sopenharmony_ci if (fpga_ver < 39) 126862306a36Sopenharmony_ci card->atmel_flash = 1; 126962306a36Sopenharmony_ci else 127062306a36Sopenharmony_ci card->atmel_flash = 0; 127162306a36Sopenharmony_ci 127262306a36Sopenharmony_ci data32 = ioread32(card->config_regs + PORTS); 127362306a36Sopenharmony_ci card->nr_ports = (data32 & 0x000000FF); 127462306a36Sopenharmony_ci 127562306a36Sopenharmony_ci if (card->fpga_version >= DMA_SUPPORTED) { 127662306a36Sopenharmony_ci pci_set_master(dev); 127762306a36Sopenharmony_ci card->using_dma = 1; 127862306a36Sopenharmony_ci if (1) { /* All known FPGA versions so far */ 127962306a36Sopenharmony_ci card->dma_alignment = 3; 128062306a36Sopenharmony_ci card->dma_bounce = kmalloc_array(card->nr_ports, 128162306a36Sopenharmony_ci BUF_SIZE, GFP_KERNEL); 128262306a36Sopenharmony_ci if (!card->dma_bounce) { 128362306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate DMA bounce buffers\n"); 128462306a36Sopenharmony_ci err = -ENOMEM; 128562306a36Sopenharmony_ci /* Fallback to MMIO doesn't work */ 128662306a36Sopenharmony_ci goto out_unmap_both; 128762306a36Sopenharmony_ci } 128862306a36Sopenharmony_ci } 128962306a36Sopenharmony_ci } else { 129062306a36Sopenharmony_ci card->using_dma = 0; 129162306a36Sopenharmony_ci /* Set RX empty flag for all ports */ 129262306a36Sopenharmony_ci iowrite32(0xF0, card->config_regs + FLAGS_ADDR); 129362306a36Sopenharmony_ci } 129462306a36Sopenharmony_ci 129562306a36Sopenharmony_ci pci_set_drvdata(dev, card); 129662306a36Sopenharmony_ci 129762306a36Sopenharmony_ci tasklet_init(&card->tlet, solos_bh, (unsigned long)card); 129862306a36Sopenharmony_ci spin_lock_init(&card->tx_lock); 129962306a36Sopenharmony_ci spin_lock_init(&card->tx_queue_lock); 130062306a36Sopenharmony_ci spin_lock_init(&card->cli_queue_lock); 130162306a36Sopenharmony_ci spin_lock_init(&card->param_queue_lock); 130262306a36Sopenharmony_ci INIT_LIST_HEAD(&card->param_queue); 130362306a36Sopenharmony_ci 130462306a36Sopenharmony_ci err = request_irq(dev->irq, solos_irq, IRQF_SHARED, 130562306a36Sopenharmony_ci "solos-pci", card); 130662306a36Sopenharmony_ci if (err) { 130762306a36Sopenharmony_ci dev_dbg(&card->dev->dev, "Failed to request interrupt IRQ: %d\n", dev->irq); 130862306a36Sopenharmony_ci goto out_unmap_both; 130962306a36Sopenharmony_ci } 131062306a36Sopenharmony_ci 131162306a36Sopenharmony_ci iowrite32(1, card->config_regs + IRQ_EN_ADDR); 131262306a36Sopenharmony_ci 131362306a36Sopenharmony_ci if (fpga_upgrade) 131462306a36Sopenharmony_ci flash_upgrade(card, 0); 131562306a36Sopenharmony_ci 131662306a36Sopenharmony_ci if (firmware_upgrade) 131762306a36Sopenharmony_ci flash_upgrade(card, 1); 131862306a36Sopenharmony_ci 131962306a36Sopenharmony_ci if (db_fpga_upgrade) 132062306a36Sopenharmony_ci flash_upgrade(card, 2); 132162306a36Sopenharmony_ci 132262306a36Sopenharmony_ci if (db_firmware_upgrade) 132362306a36Sopenharmony_ci flash_upgrade(card, 3); 132462306a36Sopenharmony_ci 132562306a36Sopenharmony_ci err = atm_init(card, &dev->dev); 132662306a36Sopenharmony_ci if (err) 132762306a36Sopenharmony_ci goto out_free_irq; 132862306a36Sopenharmony_ci 132962306a36Sopenharmony_ci if (card->fpga_version >= DMA_SUPPORTED && 133062306a36Sopenharmony_ci sysfs_create_group(&card->dev->dev.kobj, &gpio_attr_group)) 133162306a36Sopenharmony_ci dev_err(&card->dev->dev, "Could not register parameter group for GPIOs\n"); 133262306a36Sopenharmony_ci 133362306a36Sopenharmony_ci return 0; 133462306a36Sopenharmony_ci 133562306a36Sopenharmony_ci out_free_irq: 133662306a36Sopenharmony_ci iowrite32(0, card->config_regs + IRQ_EN_ADDR); 133762306a36Sopenharmony_ci free_irq(dev->irq, card); 133862306a36Sopenharmony_ci tasklet_kill(&card->tlet); 133962306a36Sopenharmony_ci 134062306a36Sopenharmony_ci out_unmap_both: 134162306a36Sopenharmony_ci kfree(card->dma_bounce); 134262306a36Sopenharmony_ci pci_iounmap(dev, card->buffers); 134362306a36Sopenharmony_ci out_unmap_config: 134462306a36Sopenharmony_ci pci_iounmap(dev, card->config_regs); 134562306a36Sopenharmony_ci out_release_regions: 134662306a36Sopenharmony_ci pci_release_regions(dev); 134762306a36Sopenharmony_ci out: 134862306a36Sopenharmony_ci kfree(card); 134962306a36Sopenharmony_ci return err; 135062306a36Sopenharmony_ci} 135162306a36Sopenharmony_ci 135262306a36Sopenharmony_cistatic int atm_init(struct solos_card *card, struct device *parent) 135362306a36Sopenharmony_ci{ 135462306a36Sopenharmony_ci int i; 135562306a36Sopenharmony_ci 135662306a36Sopenharmony_ci for (i = 0; i < card->nr_ports; i++) { 135762306a36Sopenharmony_ci struct sk_buff *skb; 135862306a36Sopenharmony_ci struct pkt_hdr *header; 135962306a36Sopenharmony_ci 136062306a36Sopenharmony_ci skb_queue_head_init(&card->tx_queue[i]); 136162306a36Sopenharmony_ci skb_queue_head_init(&card->cli_queue[i]); 136262306a36Sopenharmony_ci 136362306a36Sopenharmony_ci card->atmdev[i] = atm_dev_register("solos-pci", parent, &fpga_ops, -1, NULL); 136462306a36Sopenharmony_ci if (!card->atmdev[i]) { 136562306a36Sopenharmony_ci dev_err(&card->dev->dev, "Could not register ATM device %d\n", i); 136662306a36Sopenharmony_ci atm_remove(card); 136762306a36Sopenharmony_ci return -ENODEV; 136862306a36Sopenharmony_ci } 136962306a36Sopenharmony_ci if (device_create_file(&card->atmdev[i]->class_dev, &dev_attr_console)) 137062306a36Sopenharmony_ci dev_err(&card->dev->dev, "Could not register console for ATM device %d\n", i); 137162306a36Sopenharmony_ci if (sysfs_create_group(&card->atmdev[i]->class_dev.kobj, &solos_attr_group)) 137262306a36Sopenharmony_ci dev_err(&card->dev->dev, "Could not register parameter group for ATM device %d\n", i); 137362306a36Sopenharmony_ci 137462306a36Sopenharmony_ci dev_info(&card->dev->dev, "Registered ATM device %d\n", card->atmdev[i]->number); 137562306a36Sopenharmony_ci 137662306a36Sopenharmony_ci card->atmdev[i]->ci_range.vpi_bits = 8; 137762306a36Sopenharmony_ci card->atmdev[i]->ci_range.vci_bits = 16; 137862306a36Sopenharmony_ci card->atmdev[i]->dev_data = card; 137962306a36Sopenharmony_ci card->atmdev[i]->phy_data = (void *)(unsigned long)i; 138062306a36Sopenharmony_ci atm_dev_signal_change(card->atmdev[i], ATM_PHY_SIG_FOUND); 138162306a36Sopenharmony_ci 138262306a36Sopenharmony_ci skb = alloc_skb(sizeof(*header), GFP_KERNEL); 138362306a36Sopenharmony_ci if (!skb) { 138462306a36Sopenharmony_ci dev_warn(&card->dev->dev, "Failed to allocate sk_buff in atm_init()\n"); 138562306a36Sopenharmony_ci continue; 138662306a36Sopenharmony_ci } 138762306a36Sopenharmony_ci 138862306a36Sopenharmony_ci header = skb_put(skb, sizeof(*header)); 138962306a36Sopenharmony_ci 139062306a36Sopenharmony_ci header->size = cpu_to_le16(0); 139162306a36Sopenharmony_ci header->vpi = cpu_to_le16(0); 139262306a36Sopenharmony_ci header->vci = cpu_to_le16(0); 139362306a36Sopenharmony_ci header->type = cpu_to_le16(PKT_STATUS); 139462306a36Sopenharmony_ci 139562306a36Sopenharmony_ci fpga_queue(card, i, skb, NULL); 139662306a36Sopenharmony_ci } 139762306a36Sopenharmony_ci return 0; 139862306a36Sopenharmony_ci} 139962306a36Sopenharmony_ci 140062306a36Sopenharmony_cistatic void atm_remove(struct solos_card *card) 140162306a36Sopenharmony_ci{ 140262306a36Sopenharmony_ci int i; 140362306a36Sopenharmony_ci 140462306a36Sopenharmony_ci for (i = 0; i < card->nr_ports; i++) { 140562306a36Sopenharmony_ci if (card->atmdev[i]) { 140662306a36Sopenharmony_ci struct sk_buff *skb; 140762306a36Sopenharmony_ci 140862306a36Sopenharmony_ci dev_info(&card->dev->dev, "Unregistering ATM device %d\n", card->atmdev[i]->number); 140962306a36Sopenharmony_ci 141062306a36Sopenharmony_ci sysfs_remove_group(&card->atmdev[i]->class_dev.kobj, &solos_attr_group); 141162306a36Sopenharmony_ci atm_dev_deregister(card->atmdev[i]); 141262306a36Sopenharmony_ci 141362306a36Sopenharmony_ci skb = card->rx_skb[i]; 141462306a36Sopenharmony_ci if (skb) { 141562306a36Sopenharmony_ci dma_unmap_single(&card->dev->dev, SKB_CB(skb)->dma_addr, 141662306a36Sopenharmony_ci RX_DMA_SIZE, DMA_FROM_DEVICE); 141762306a36Sopenharmony_ci dev_kfree_skb(skb); 141862306a36Sopenharmony_ci } 141962306a36Sopenharmony_ci skb = card->tx_skb[i]; 142062306a36Sopenharmony_ci if (skb) { 142162306a36Sopenharmony_ci dma_unmap_single(&card->dev->dev, SKB_CB(skb)->dma_addr, 142262306a36Sopenharmony_ci skb->len, DMA_TO_DEVICE); 142362306a36Sopenharmony_ci dev_kfree_skb(skb); 142462306a36Sopenharmony_ci } 142562306a36Sopenharmony_ci while ((skb = skb_dequeue(&card->tx_queue[i]))) 142662306a36Sopenharmony_ci dev_kfree_skb(skb); 142762306a36Sopenharmony_ci 142862306a36Sopenharmony_ci } 142962306a36Sopenharmony_ci } 143062306a36Sopenharmony_ci} 143162306a36Sopenharmony_ci 143262306a36Sopenharmony_cistatic void fpga_remove(struct pci_dev *dev) 143362306a36Sopenharmony_ci{ 143462306a36Sopenharmony_ci struct solos_card *card = pci_get_drvdata(dev); 143562306a36Sopenharmony_ci 143662306a36Sopenharmony_ci /* Disable IRQs */ 143762306a36Sopenharmony_ci iowrite32(0, card->config_regs + IRQ_EN_ADDR); 143862306a36Sopenharmony_ci 143962306a36Sopenharmony_ci /* Reset FPGA */ 144062306a36Sopenharmony_ci iowrite32(1, card->config_regs + FPGA_MODE); 144162306a36Sopenharmony_ci (void)ioread32(card->config_regs + FPGA_MODE); 144262306a36Sopenharmony_ci 144362306a36Sopenharmony_ci if (card->fpga_version >= DMA_SUPPORTED) 144462306a36Sopenharmony_ci sysfs_remove_group(&card->dev->dev.kobj, &gpio_attr_group); 144562306a36Sopenharmony_ci 144662306a36Sopenharmony_ci atm_remove(card); 144762306a36Sopenharmony_ci 144862306a36Sopenharmony_ci free_irq(dev->irq, card); 144962306a36Sopenharmony_ci tasklet_kill(&card->tlet); 145062306a36Sopenharmony_ci 145162306a36Sopenharmony_ci kfree(card->dma_bounce); 145262306a36Sopenharmony_ci 145362306a36Sopenharmony_ci /* Release device from reset */ 145462306a36Sopenharmony_ci iowrite32(0, card->config_regs + FPGA_MODE); 145562306a36Sopenharmony_ci (void)ioread32(card->config_regs + FPGA_MODE); 145662306a36Sopenharmony_ci 145762306a36Sopenharmony_ci pci_iounmap(dev, card->buffers); 145862306a36Sopenharmony_ci pci_iounmap(dev, card->config_regs); 145962306a36Sopenharmony_ci 146062306a36Sopenharmony_ci pci_release_regions(dev); 146162306a36Sopenharmony_ci pci_disable_device(dev); 146262306a36Sopenharmony_ci 146362306a36Sopenharmony_ci kfree(card); 146462306a36Sopenharmony_ci} 146562306a36Sopenharmony_ci 146662306a36Sopenharmony_cistatic const struct pci_device_id fpga_pci_tbl[] = { 146762306a36Sopenharmony_ci { 0x10ee, 0x0300, PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, 146862306a36Sopenharmony_ci { 0, } 146962306a36Sopenharmony_ci}; 147062306a36Sopenharmony_ci 147162306a36Sopenharmony_ciMODULE_DEVICE_TABLE(pci,fpga_pci_tbl); 147262306a36Sopenharmony_ci 147362306a36Sopenharmony_cistatic struct pci_driver fpga_driver = { 147462306a36Sopenharmony_ci .name = "solos", 147562306a36Sopenharmony_ci .id_table = fpga_pci_tbl, 147662306a36Sopenharmony_ci .probe = fpga_probe, 147762306a36Sopenharmony_ci .remove = fpga_remove, 147862306a36Sopenharmony_ci}; 147962306a36Sopenharmony_ci 148062306a36Sopenharmony_ci 148162306a36Sopenharmony_cistatic int __init solos_pci_init(void) 148262306a36Sopenharmony_ci{ 148362306a36Sopenharmony_ci BUILD_BUG_ON(sizeof(struct solos_skb_cb) > sizeof(((struct sk_buff *)0)->cb)); 148462306a36Sopenharmony_ci 148562306a36Sopenharmony_ci printk(KERN_INFO "Solos PCI Driver Version %s\n", VERSION); 148662306a36Sopenharmony_ci return pci_register_driver(&fpga_driver); 148762306a36Sopenharmony_ci} 148862306a36Sopenharmony_ci 148962306a36Sopenharmony_cistatic void __exit solos_pci_exit(void) 149062306a36Sopenharmony_ci{ 149162306a36Sopenharmony_ci pci_unregister_driver(&fpga_driver); 149262306a36Sopenharmony_ci printk(KERN_INFO "Solos PCI Driver %s Unloaded\n", VERSION); 149362306a36Sopenharmony_ci} 149462306a36Sopenharmony_ci 149562306a36Sopenharmony_cimodule_init(solos_pci_init); 149662306a36Sopenharmony_cimodule_exit(solos_pci_exit); 1497