xref: /aosp_15_r20/external/flashrom/nic3com.c (revision 0d6140be3aa665ecc836e8907834fcd3e3b018fc)
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