1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
|
/* $OpenBSD: smc91cxxreg.h,v 1.3 2006/01/23 14:42:55 martin Exp $ */
/* $NetBSD: smc91cxxreg.h,v 1.2 1997/09/02 00:10:58 thorpej Exp $ */
/*
* Copyright (c) 1996 Gardner Buchanan <gbuchanan@shl.com>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by Gardner Buchanan.
* 4. The name of Gardner Buchanan may not be used to endorse or promote
* products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* from FreeBSD Id: if_snreg.h,v 1.4 1996/03/18 15:47:30 gardner Exp
*/
/*
* This file contains register information and access macros for
* the SMC91xxx chipset.
*
* Information contained in this file was obtained from the SMC91C92
* and SMC91C94 manuals from SMC. You will need one of these in order
* to make any meaningful changes to this driver. Information about
* obtaining one can be found at http://www.smc.com in the components
* division.
*
* This FreeBSD driver is derived in part from the smc9194 Linux driver
* by Erik Stahlman.
*/
/*
* Wait time for memory to be free. This probably shouldn't be
* tuned that much, as waiting for this means nothing else happens
* in the system
*/
#define MEMORY_WAIT_TIME 1000
/* The SMC91xxx uses 16 I/O ports */
#define SMC_IOSIZE 16
/*
* A description of the SMC registers is probably in order here,
* although for details, the SMC datasheet is invaluable.
* The data sheet I (GB) am using is "SMC91C92 Single Chip Ethernet
* Controller With RAM", Rev. 12/0/94. Constant definitions I give
* here are loosely based on the mnemonic names given to them in the
* data sheet, but there are many exceptions.
*
* Basically, the chip has 4 banks of registers (0 to 3), which
* are accessed by writing a number into the BANK_SELECT register
* (I also use a SMC_SELECT_BANK macro for this). Registers are
* either Byte or Word sized. My constant definitions end in _B
* or _W as appropriate.
*
* The banks are arranged so that for most purposes, bank 2 is all
* that is needed for normal run time tasks.
*/
/*
* Bank Select Register. This also doubles as
* a chip identification register. This register
* is mapped at the same position in all banks.
*/
#define BANK_SELECT_REG_W 0x0e
#define BSR_DETECT_MASK 0xff00
#define BSR_DETECT_VALUE 0x3300
/*
* BANK 0
*/
/*
* Transmit Control Register controls some aspects of the transmit
* behavior of the Ethernet Protocol Handler.
*/
#define TXMIT_CONTROL_REG_W 0x00
#define TCR_ENABLE 0x0001 /* if this is 1, we can transmit */
#define TCR_LOOP 0x0002 /* Enable internal analogue loopback */
#define TCR_FORCOL 0x0004 /* Force Collision on next TX */
#define TCR_PAD_ENABLE 0x0080 /* Pad short packets to 64 bytes */
#define TCR_NOCRC 0x0100 /* Do not append CRC */
#define TCR_MON_CSN 0x0400 /* monitors the carrier status */
#define TCR_FDUPLX 0x0800 /* receive packets sent out */
#define TCR_STP_SQET 0x1000 /* stop transmitting if Signal quality error */
#define TCR_EPH_LOOP 0x2000 /* Enable internal digital loopback */
#define TCR_SWFDUP 0x8000 /* FEAST: Switched full-duplex (only w/ MII) */
/*
* Status of the last transmitted frame and instantaneous status of
* the Ethernet Protocol Handler jumbled together. In auto-release
* mode this information is simply discarded after each TX. This info
* is copied to the status word of in-memory packets after transmit
* where relevent statuses can be checked.
*/
#define EPH_STATUS_REG_W 0x02
#define EPHSR_TX_SUC 0x0001 /* Transmit was successful */
#define EPHSR_SNGLCOL 0x0002 /* Single collision occurred */
#define EPHSR_MULCOL 0x0004 /* Multiple Collisions occurred */
#define EPHSR_LTX_MULT 0x0008 /* Transmit was a multicast */
#define EPHSR_16COL 0x0010 /* 16 Collisions occurred, TX disabled */
#define EPHSR_SQET 0x0020 /* SQE Test failed, TX disabled */
#define EPHSR_LTX_BRD 0x0040 /* Transmit was a broadcast */
#define EPHSR_DEFR 0x0080 /* TX deferred due to carrier det. */
#define EPHSR_LATCOL 0x0200 /* Late collision detected, TX disabled */
#define EPHSR_LOST_CAR 0x0400 /* Lost carrier sense, TX disabled */
#define EPHSR_EXC_DEF 0x0800 /* Excessive deferrals in TX >2 MAXETHER
* times */
#define EPHSR_CTR_ROL 0x1000 /* Some ECR Counter(s) rolled over */
#define EPHSR_RX_OVRN 0x2000 /* Receiver overrun, packets dropped */
#define EPHSR_LINK_OK 0x4000 /* Link integrity is OK */
#define EPHSR_TXUNRN 0x8000 /* Transmit underrun */
/*
* Receiver Control Register controls some aspects of the receive
* behavior of the Ethernet Protocol Handler.
*/
#define RECV_CONTROL_REG_W 0x04
#define RCR_RX_ABORT 0x0001 /* Received huge packet */
#define RCR_PROMISC 0x0002 /* enable promiscuous mode */
#define RCR_ALMUL 0x0004 /* receive all multicast packets */
#define RCR_ENABLE 0x0100 /* IFF this is set, we can receive packets */
#define RCR_STRIP_CRC 0x0200 /* strips CRC */
#define RCR_GAIN_BITS 0x0c00 /* PLL Gain control (for testing) */
#define RCR_FILT_CAR 0x4000 /* Enable 12 bit carrier filter */
#define RCR_SOFTRESET 0x8000 /* Resets the EPH logic */
/*
* TX Statistics counters
*/
#define COUNTER_REG_W 0x06
#define ECR_COLN_MASK 0x000f /* Vanilla collisions */
#define ECR_MCOLN_MASK 0x00f0 /* Multiple collisions */
#define ECR_DTX_MASK 0x0f00 /* Deferred transmits */
#define ECR_EXDTX_MASK 0xf000 /* Excessively deferred transmits */
/*
* Memory Information
*/
#define MEM_INFO_REG_W 0x08
#define MIR_FREE_MASK 0xff00 /* Free memory pages available */
#define MIR_TOTAL_MASK 0x00ff /* Total memory pages available */
/*
* Memory Configuration
*/
#define MEM_CFG_REG_W 0x0a
#define MCR_MEM_MULT(x) (((x)>>9)&7) /* Memory size multiplier */
#define MCR_TXRSV_MASK 0x001f /* Count of pages reserved for transmit */
/*
* Bank 0, Register 0x0c is unused in the SMC91C92
*/
/*
* BANK 1
*/
/*
* Adapter configuration
*/
#define CONFIG_REG_W 0x00
#define CR_INT_SEL0 0x0002 /* Interrupt selector */
#define CR_INT_SEL1 0x0004 /* Interrupt selector */
#define CR_DIS_LINK 0x0040 /* Disable 10BaseT Link Test */
#define CR_16BIT 0x0080 /* Bus width */
#define CR_AUI_SELECT 0x0100 /* Use external (AUI) Transceiver */
#define CR_SET_SQLCH 0x0200 /* Squelch level */
#define CR_FULL_STEP 0x0400 /* AUI signalling mode */
#define CR_NOW_WAIT_ST 0x1000 /* Disable bus wait states */
#define CR_MII_SELECT 0x8000 /* FEAST: MII port selected */
/*
* The contents of this port are used by the adapter
* to decode its I/O address. We use it as a varification
* that the adapter is detected properly when probing.
*/
#define BASE_ADDR_REG_W 0x02 /* The selected I/O Base addr. */
/*
* These registers hold the Ethernet MAC address.
*/
#define IAR_ADDR0_REG_W 0x04 /* My Ethernet address */
#define IAR_ADDR1_REG_W 0x06 /* My Ethernet address */
#define IAR_ADDR2_REG_W 0x08 /* My Ethernet address */
/*
* General purpose register used for talking to the EEPROM.
*/
#define GENERAL_REG_W 0x0a
/*
* Control register used for talking to the EEPROM and
* setting some EPH functions.
*/
#define CONTROL_REG_W 0x0c
#define CTR_STORE 0x0001 /* Store something to EEPROM */
#define CTR_RELOAD 0x0002 /* Read EEPROM into registers */
#define CTR_EEPROM_SEL 0x0004 /* Select registers for Reload/Store */
#define CTR_TE_ENABLE 0x0020 /* Enable TX Error detection via EPH_INT */
#define CTR_CR_ENABLE 0x0040 /* Enable Counter Rollover via EPH_INT */
#define CTR_LE_ENABLE 0x0080 /* Enable Link Error detection via EPH_INT */
#define CTR_AUTO_RELEASE 0x0800 /* Enable auto release mode for TX */
#define CTR_POWERDOWN 0x2000 /* Enter powerdown mode */
#define CTR_RCV_BAD 0x4000 /* Enable receipt of frames with bad CRC */
/*
* BANK 2
*/
/*
* Memory Management Unit Control Register
* Controls allocation of memory to receive and
* transmit functions.
*/
#define MMU_CMD_REG_W 0x00
#define MMUCR_BUSY 0x0001 /* MMU busy performing a release */
/*
* MMU Commands:
*/
#define MMUCR_NOP 0x0000 /* Do nothing */
#define MMUCR_ALLOC 0x0020 /* Or with number of 256 byte packets - 1 */
#define MMUCR_RESET 0x0040 /* Reset MMU State */
#define MMUCR_REMOVE 0x0060 /* Dequeue (but not free) current RX packet */
#define MMUCR_RELEASE 0x0080 /* Dequeue and free the current RX packet */
#define MMUCR_FREEPKT 0x00a0 /* Release packet in PNR register */
#define MMUCR_ENQUEUE 0x00c0 /* Enqueue the packet for transmit */
#define MMUCR_RESETTX 0x00e0 /* Reset transmit queues */
/*
* Packet Number at TX Area
*/
#define PACKET_NUM_REG_B 0x02
/*
* Packet number resulting from MMUCR_ALLOC
*/
#define ALLOC_RESULT_REG_B 0x03
#define ARR_FAILED 0x80
/*
* Transmit and receive queue heads
*/
#define FIFO_PORTS_REG_W 0x04
#define FIFO_REMPTY 0x8000
#define FIFO_TEMPTY 0x0080
#define FIFO_RX_MASK 0x7f00
#define FIFO_TX_MASK 0x007f
/*
* The address within the packet for reading/writing. The
* PTR_RCV bit is tricky. When PTR_RCV==1, the packet number
* to be read is found in the FIFO_PORTS_REG_W, FIFO_RX_MASK.
* When PTR_RCV==0, the packet number to be written is found
* in the PACKET_NUM_REG_B.
*/
#define POINTER_REG_W 0x06
#define PTR_READ 0x2000 /* Intended access mode */
#define PTR_AUTOINC 0x4000 /* Do auto inc after read/write */
#define PTR_RCV 0x8000 /* FIFO_RX is packet, otherwise PNR is packet */
/*
* Data I/O register to be used in conjunction with
* The pointer register to read and write data from the
* card. The same register can be used for byte and word
* ops.
*/
#define DATA_REG_W 0x08
#define DATA_REG_B 0x08
#define DATA_1_REG_B 0x08
#define DATA_2_REG_B 0x0a
/*
* Sense interrupt status (READ)
*/
#define INTR_STAT_REG_B 0x0c
/*
* Acknowledge interrupt sources (WRITE)
*/
#define INTR_ACK_REG_B 0x0c
/*
* Interrupt mask. Bit set indicates interrupt allowed.
*/
#define INTR_MASK_REG_B 0x0d
/*
* Interrupts
*/
#define IM_RCV_INT 0x01 /* A packet has been received */
#define IM_TX_INT 0x02 /* Packet TX complete */
#define IM_TX_EMPTY_INT 0x04 /* No packets left to TX */
#define IM_ALLOC_INT 0x08 /* Memory allocation completed */
#define IM_RX_OVRN_INT 0x10 /* Receiver was overrun */
#define IM_EPH_INT 0x20 /* Misc. EPH conditions (see CONTROL_REG_W) */
#define IM_ERCV_INT 0x40 /* not on SMC9192 */
/*
* BANK 3
*/
/*
* Multicast subscriptions.
* The multicast handling in the SMC90Cxx is quite complicated. A table
* of multicast address subscriptions is provided and a clever way of
* speeding the search of that table by hashing is implemented in the
* hardware. I have ignored this and simply subscribed to all multicasts
* and let the kernel deal with the results.
*/
#define MULTICAST1_REG_W 0x00
#define MULTICAST2_REG_W 0x02
#define MULTICAST3_REG_W 0x04
#define MULTICAST4_REG_W 0x06
/*
* These registers do not exist on SMC9192, or at least
* are not documented in the SMC91C92 data sheet.
*
* The REVISION_REG_W register does however seem to work.
*
* On the FEAST, the low nibble controls the MII interface.
*/
#define MGMT_REG_W 0x08
#define MR_MDOE 0x08
#define MR_MCLK 0x04
#define MR_MDI 0x02
#define MR_MDO 0x01
#define REVISION_REG_W 0x0a /* (hi: chip id low: rev #) */
#define RR_REV(x) ((x) & 0x0f)
#define RR_ID(x) (((x) >> 4) & 0x0f)
#define ERCV_REG_W 0x0c
/*
* These are constants expected to be found in the
* chip id register.
*/
#define CHIP_9190 3
#define CHIP_9194 4
#define CHIP_9195 5
#define CHIP_91100 7
#define CHIP_91100FD 8
/*
* When packets are stuffed into the card or sucked out of the card
* they are set up more or less as follows:
*
* Addr msbyte lsbyte
* 00 SSSSSSSS SSSSSSSS - STATUS-WORD 16 bit TX or RX status
* 02 RRRRR - RESERVED (unused)
* 02 CCC CCCCCCCC - BYTE COUNT (RX: always even, TX: bit 0 ignored)
* 04 DDDDDDDD DDDDDDDD - DESTINATION ADDRESS
* 06 DDDDDDDD DDDDDDDD (48 bit Ethernet MAC Address)
* 08 DDDDDDDD DDDDDDDD
* 0A SSSSSSSS SSSSSSSS - SOURCE ADDRESS
* 0C SSSSSSSS SSSSSSSS (48 bit Ethernet MAC Address)
* 0E SSSSSSSS SSSSSSSS
* 10 PPPPPPPP PPPPPPPP
* .. PPPPPPPP PPPPPPPP
* C-2 CCCCCCCC - CONTROL BYTE
* C-2 PPPPPPPP - Last data byte (If odd length)
*
* The STATUS_WORD is derived from the EPH_STATUS_REG_W register
* during transmit and is composed of another set of bits described
* below during receive.
*/
/*
* Receive status bits. These values are found in the status word
* field of a received packet. For receive packets I use the RS_ODDFRAME
* to detect whether a frame has an extra byte on it. The CTLB_ODD
* bit of the control byte tells the same thing.
*/
#define RS_MULTICAST 0x0001 /* Packet is multicast */
#define RS_HASH_MASK 0x007e /* Mask of multicast hash value */
#define RS_TOOSHORT 0x0400 /* Frame was a runt, <64 bytes */
#define RS_TOOLONG 0x0800 /* Frame was giant, >1518 */
#define RS_ODDFRAME 0x1000 /* Frame is odd lengthed */
#define RS_BADCRC 0x2000 /* Frame had CRC error */
#define RS_ALGNERR 0x8000 /* Frame had alignment error */
#define RS_ERRORS (RS_ALGNERR | RS_BADCRC | RS_TOOLONG | RS_TOOSHORT)
#define RLEN_MASK 0x07ff /* Significant length bits in RX length */
/*
* The control byte has the following significant bits.
* For transmit, the CTLB_ODD bit specifies whether an extra byte
* is present in the frame. Bit 0 of the byte count field is
* ignored. I just pad every frame to even length and forget about
* it.
*/
#define CTLB_CRC 0x10 /* Add CRC for this packet (TX only) */
#define CTLB_ODD 0x20 /* The packet length is ODD */
|