1*0d6140beSAndroid Build Coastguard Worker /*
2*0d6140beSAndroid Build Coastguard Worker * This file is part of the flashrom project.
3*0d6140beSAndroid Build Coastguard Worker *
4*0d6140beSAndroid Build Coastguard Worker * Copyright (C) 2009 Uwe Hermann <[email protected]>
5*0d6140beSAndroid Build Coastguard Worker *
6*0d6140beSAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or modify
7*0d6140beSAndroid Build Coastguard Worker * it under the terms of the GNU General Public License as published by
8*0d6140beSAndroid Build Coastguard Worker * the Free Software Foundation; either version 2 of the License, or
9*0d6140beSAndroid Build Coastguard Worker * (at your option) any later version.
10*0d6140beSAndroid Build Coastguard Worker *
11*0d6140beSAndroid Build Coastguard Worker * This program is distributed in the hope that it will be useful,
12*0d6140beSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*0d6140beSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14*0d6140beSAndroid Build Coastguard Worker * GNU General Public License for more details.
15*0d6140beSAndroid Build Coastguard Worker */
16*0d6140beSAndroid Build Coastguard Worker
17*0d6140beSAndroid Build Coastguard Worker #include <stdlib.h>
18*0d6140beSAndroid Build Coastguard Worker #include "flash.h"
19*0d6140beSAndroid Build Coastguard Worker #include "programmer.h"
20*0d6140beSAndroid Build Coastguard Worker #include "hwaccess_x86_io.h"
21*0d6140beSAndroid Build Coastguard Worker #include "platform/pci.h"
22*0d6140beSAndroid Build Coastguard Worker
23*0d6140beSAndroid Build Coastguard Worker #define BIOS_ROM_ADDR 0x04
24*0d6140beSAndroid Build Coastguard Worker #define BIOS_ROM_DATA 0x08
25*0d6140beSAndroid Build Coastguard Worker #define INT_STATUS 0x0e
26*0d6140beSAndroid Build Coastguard Worker #define INTERNAL_CONFIG 0x00
27*0d6140beSAndroid Build Coastguard Worker #define SELECT_REG_WINDOW 0x800
28*0d6140beSAndroid Build Coastguard Worker
29*0d6140beSAndroid Build Coastguard Worker #define PCI_VENDOR_ID_3COM 0x10b7
30*0d6140beSAndroid Build Coastguard Worker
31*0d6140beSAndroid Build Coastguard Worker struct nic3com_data {
32*0d6140beSAndroid Build Coastguard Worker uint32_t io_base_addr;
33*0d6140beSAndroid Build Coastguard Worker uint32_t internal_conf;
34*0d6140beSAndroid Build Coastguard Worker uint16_t id;
35*0d6140beSAndroid Build Coastguard Worker };
36*0d6140beSAndroid Build Coastguard Worker
37*0d6140beSAndroid Build Coastguard Worker static const struct dev_entry nics_3com[] = {
38*0d6140beSAndroid Build Coastguard Worker /* 3C90xB */
39*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9055, OK, "3COM", "3C90xB: PCI 10/100 Mbps; shared 10BASE-T/100BASE-TX"},
40*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9001, NT, "3COM", "3C90xB: PCI 10/100 Mbps; shared 10BASE-T/100BASE-T4" },
41*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9004, OK, "3COM", "3C90xB: PCI 10BASE-T (TPO)" },
42*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9005, NT, "3COM", "3C90xB: PCI 10BASE-T/10BASE2/AUI (COMBO)" },
43*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9006, OK, "3COM", "3C90xB: PCI 10BASE-T/10BASE2 (TPC)" },
44*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x900a, NT, "3COM", "3C90xB: PCI 10BASE-FL" },
45*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x905a, NT, "3COM", "3C90xB: PCI 10BASE-FX" },
46*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9058, OK, "3COM", "3C905B: Cyclone 10/100/BNC" },
47*0d6140beSAndroid Build Coastguard Worker
48*0d6140beSAndroid Build Coastguard Worker /* 3C905C */
49*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9200, OK, "3COM", "3C905C: EtherLink 10/100 PCI (TX)" },
50*0d6140beSAndroid Build Coastguard Worker
51*0d6140beSAndroid Build Coastguard Worker /* 3C980C */
52*0d6140beSAndroid Build Coastguard Worker {0x10b7, 0x9805, NT, "3COM", "3C980C: EtherLink Server 10/100 PCI (TX)" },
53*0d6140beSAndroid Build Coastguard Worker
54*0d6140beSAndroid Build Coastguard Worker {0},
55*0d6140beSAndroid Build Coastguard Worker };
56*0d6140beSAndroid Build Coastguard Worker
nic3com_chip_writeb(const struct flashctx * flash,uint8_t val,chipaddr addr)57*0d6140beSAndroid Build Coastguard Worker static void nic3com_chip_writeb(const struct flashctx *flash, uint8_t val,
58*0d6140beSAndroid Build Coastguard Worker chipaddr addr)
59*0d6140beSAndroid Build Coastguard Worker {
60*0d6140beSAndroid Build Coastguard Worker struct nic3com_data *data = flash->mst->par.data;
61*0d6140beSAndroid Build Coastguard Worker
62*0d6140beSAndroid Build Coastguard Worker OUTL((uint32_t)addr, data->io_base_addr + BIOS_ROM_ADDR);
63*0d6140beSAndroid Build Coastguard Worker OUTB(val, data->io_base_addr + BIOS_ROM_DATA);
64*0d6140beSAndroid Build Coastguard Worker }
65*0d6140beSAndroid Build Coastguard Worker
nic3com_chip_readb(const struct flashctx * flash,const chipaddr addr)66*0d6140beSAndroid Build Coastguard Worker static uint8_t nic3com_chip_readb(const struct flashctx *flash,
67*0d6140beSAndroid Build Coastguard Worker const chipaddr addr)
68*0d6140beSAndroid Build Coastguard Worker {
69*0d6140beSAndroid Build Coastguard Worker struct nic3com_data *data = flash->mst->par.data;
70*0d6140beSAndroid Build Coastguard Worker
71*0d6140beSAndroid Build Coastguard Worker OUTL((uint32_t)addr, data->io_base_addr + BIOS_ROM_ADDR);
72*0d6140beSAndroid Build Coastguard Worker return INB(data->io_base_addr + BIOS_ROM_DATA);
73*0d6140beSAndroid Build Coastguard Worker }
74*0d6140beSAndroid Build Coastguard Worker
nic3com_shutdown(void * par_data)75*0d6140beSAndroid Build Coastguard Worker static int nic3com_shutdown(void *par_data)
76*0d6140beSAndroid Build Coastguard Worker {
77*0d6140beSAndroid Build Coastguard Worker struct nic3com_data *data = par_data;
78*0d6140beSAndroid Build Coastguard Worker const uint16_t id = data->id;
79*0d6140beSAndroid Build Coastguard Worker
80*0d6140beSAndroid Build Coastguard Worker /* 3COM 3C90xB cards need a special fixup. */
81*0d6140beSAndroid Build Coastguard Worker if (id == 0x9055 || id == 0x9001 || id == 0x9004 || id == 0x9005
82*0d6140beSAndroid Build Coastguard Worker || id == 0x9006 || id == 0x900a || id == 0x905a || id == 0x9058) {
83*0d6140beSAndroid Build Coastguard Worker /* Select register window 3 and restore the receiver status. */
84*0d6140beSAndroid Build Coastguard Worker OUTW(SELECT_REG_WINDOW + 3, data->io_base_addr + INT_STATUS);
85*0d6140beSAndroid Build Coastguard Worker OUTL(data->internal_conf, data->io_base_addr + INTERNAL_CONFIG);
86*0d6140beSAndroid Build Coastguard Worker }
87*0d6140beSAndroid Build Coastguard Worker
88*0d6140beSAndroid Build Coastguard Worker free(data);
89*0d6140beSAndroid Build Coastguard Worker return 0;
90*0d6140beSAndroid Build Coastguard Worker }
91*0d6140beSAndroid Build Coastguard Worker
92*0d6140beSAndroid Build Coastguard Worker static const struct par_master par_master_nic3com = {
93*0d6140beSAndroid Build Coastguard Worker .chip_readb = nic3com_chip_readb,
94*0d6140beSAndroid Build Coastguard Worker .chip_writeb = nic3com_chip_writeb,
95*0d6140beSAndroid Build Coastguard Worker .shutdown = nic3com_shutdown,
96*0d6140beSAndroid Build Coastguard Worker };
97*0d6140beSAndroid Build Coastguard Worker
nic3com_init(const struct programmer_cfg * cfg)98*0d6140beSAndroid Build Coastguard Worker static int nic3com_init(const struct programmer_cfg *cfg)
99*0d6140beSAndroid Build Coastguard Worker {
100*0d6140beSAndroid Build Coastguard Worker struct pci_dev *dev = NULL;
101*0d6140beSAndroid Build Coastguard Worker uint32_t io_base_addr = 0;
102*0d6140beSAndroid Build Coastguard Worker uint32_t internal_conf = 0;
103*0d6140beSAndroid Build Coastguard Worker uint16_t id;
104*0d6140beSAndroid Build Coastguard Worker
105*0d6140beSAndroid Build Coastguard Worker if (rget_io_perms())
106*0d6140beSAndroid Build Coastguard Worker return 1;
107*0d6140beSAndroid Build Coastguard Worker
108*0d6140beSAndroid Build Coastguard Worker dev = pcidev_init(cfg, nics_3com, PCI_BASE_ADDRESS_0);
109*0d6140beSAndroid Build Coastguard Worker if (!dev)
110*0d6140beSAndroid Build Coastguard Worker return 1;
111*0d6140beSAndroid Build Coastguard Worker
112*0d6140beSAndroid Build Coastguard Worker io_base_addr = pcidev_readbar(dev, PCI_BASE_ADDRESS_0);
113*0d6140beSAndroid Build Coastguard Worker if (!io_base_addr)
114*0d6140beSAndroid Build Coastguard Worker return 1;
115*0d6140beSAndroid Build Coastguard Worker
116*0d6140beSAndroid Build Coastguard Worker id = dev->device_id;
117*0d6140beSAndroid Build Coastguard Worker
118*0d6140beSAndroid Build Coastguard Worker /* 3COM 3C90xB cards need a special fixup. */
119*0d6140beSAndroid Build Coastguard Worker if (id == 0x9055 || id == 0x9001 || id == 0x9004 || id == 0x9005
120*0d6140beSAndroid Build Coastguard Worker || id == 0x9006 || id == 0x900a || id == 0x905a || id == 0x9058) {
121*0d6140beSAndroid Build Coastguard Worker /* Select register window 3 and save the receiver status. */
122*0d6140beSAndroid Build Coastguard Worker OUTW(SELECT_REG_WINDOW + 3, io_base_addr + INT_STATUS);
123*0d6140beSAndroid Build Coastguard Worker internal_conf = INL(io_base_addr + INTERNAL_CONFIG);
124*0d6140beSAndroid Build Coastguard Worker
125*0d6140beSAndroid Build Coastguard Worker /* Set receiver type to MII for full BIOS ROM access. */
126*0d6140beSAndroid Build Coastguard Worker OUTL((internal_conf & 0xf00fffff) | 0x00600000, io_base_addr);
127*0d6140beSAndroid Build Coastguard Worker }
128*0d6140beSAndroid Build Coastguard Worker
129*0d6140beSAndroid Build Coastguard Worker /*
130*0d6140beSAndroid Build Coastguard Worker * The lowest 16 bytes of the I/O mapped register space of (most) 3COM
131*0d6140beSAndroid Build Coastguard Worker * cards form a 'register window' into one of multiple (usually 8)
132*0d6140beSAndroid Build Coastguard Worker * register banks. For 3C90xB/3C90xC we need register window/bank 0.
133*0d6140beSAndroid Build Coastguard Worker */
134*0d6140beSAndroid Build Coastguard Worker OUTW(SELECT_REG_WINDOW + 0, io_base_addr + INT_STATUS);
135*0d6140beSAndroid Build Coastguard Worker
136*0d6140beSAndroid Build Coastguard Worker struct nic3com_data *data = calloc(1, sizeof(*data));
137*0d6140beSAndroid Build Coastguard Worker if (!data) {
138*0d6140beSAndroid Build Coastguard Worker msg_perr("Unable to allocate space for PAR master data\n");
139*0d6140beSAndroid Build Coastguard Worker goto init_err_cleanup_exit;
140*0d6140beSAndroid Build Coastguard Worker }
141*0d6140beSAndroid Build Coastguard Worker data->io_base_addr = io_base_addr;
142*0d6140beSAndroid Build Coastguard Worker data->internal_conf = internal_conf;
143*0d6140beSAndroid Build Coastguard Worker data->id = id;
144*0d6140beSAndroid Build Coastguard Worker
145*0d6140beSAndroid Build Coastguard Worker max_rom_decode.parallel = 128 * 1024;
146*0d6140beSAndroid Build Coastguard Worker
147*0d6140beSAndroid Build Coastguard Worker return register_par_master(&par_master_nic3com, BUS_PARALLEL, data);
148*0d6140beSAndroid Build Coastguard Worker
149*0d6140beSAndroid Build Coastguard Worker init_err_cleanup_exit:
150*0d6140beSAndroid Build Coastguard Worker /* 3COM 3C90xB cards need a special fixup. */
151*0d6140beSAndroid Build Coastguard Worker if (id == 0x9055 || id == 0x9001 || id == 0x9004 || id == 0x9005
152*0d6140beSAndroid Build Coastguard Worker || id == 0x9006 || id == 0x900a || id == 0x905a || id == 0x9058) {
153*0d6140beSAndroid Build Coastguard Worker /* Select register window 3 and restore the receiver status. */
154*0d6140beSAndroid Build Coastguard Worker OUTW(SELECT_REG_WINDOW + 3, io_base_addr + INT_STATUS);
155*0d6140beSAndroid Build Coastguard Worker OUTL(internal_conf, io_base_addr + INTERNAL_CONFIG);
156*0d6140beSAndroid Build Coastguard Worker }
157*0d6140beSAndroid Build Coastguard Worker return 1;
158*0d6140beSAndroid Build Coastguard Worker }
159*0d6140beSAndroid Build Coastguard Worker
160*0d6140beSAndroid Build Coastguard Worker const struct programmer_entry programmer_nic3com = {
161*0d6140beSAndroid Build Coastguard Worker .name = "nic3com",
162*0d6140beSAndroid Build Coastguard Worker .type = PCI,
163*0d6140beSAndroid Build Coastguard Worker .devs.dev = nics_3com,
164*0d6140beSAndroid Build Coastguard Worker .init = nic3com_init,
165*0d6140beSAndroid Build Coastguard Worker };
166