| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |  * SMSC 91C111 Ethernet interface emulation | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2005 CodeSourcery, LLC. | 
					
						
							|  |  |  |  * Written by Paul Brook | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-06-26 12:21:35 +10:00
										 |  |  |  * This code is licensed under the GPL | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-26 18:17:11 +00:00
										 |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-04 15:40:22 +01:00
										 |  |  | #include "hw/sysbus.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-24 08:43:34 +02:00
										 |  |  | #include "net/net.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-09 15:26:55 +01:00
										 |  |  | #include "hw/devices.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-26 17:50:41 +01:00
										 |  |  | #include "qemu/log.h"
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | /* For crc32 */ | 
					
						
							|  |  |  | #include <zlib.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Number of 2k memory pages available.  */ | 
					
						
							|  |  |  | #define NUM_PACKETS 4
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  | #define TYPE_SMC91C111 "smc91c111"
 | 
					
						
							|  |  |  | #define SMC91C111(obj) OBJECT_CHECK(smc91c111_state, (obj), TYPE_SMC91C111)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |     SysBusDevice parent_obj; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 18:49:23 +00:00
										 |  |  |     NICState *nic; | 
					
						
							| 
									
										
										
										
											2009-10-21 15:25:37 +02:00
										 |  |  |     NICConf conf; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     uint16_t tcr; | 
					
						
							|  |  |  |     uint16_t rcr; | 
					
						
							|  |  |  |     uint16_t cr; | 
					
						
							|  |  |  |     uint16_t ctr; | 
					
						
							|  |  |  |     uint16_t gpr; | 
					
						
							|  |  |  |     uint16_t ptr; | 
					
						
							|  |  |  |     uint16_t ercv; | 
					
						
							| 
									
										
										
										
											2007-04-07 18:14:41 +00:00
										 |  |  |     qemu_irq irq; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     int bank; | 
					
						
							|  |  |  |     int packet_num; | 
					
						
							|  |  |  |     int tx_alloc; | 
					
						
							|  |  |  |     /* Bitmask of allocated packets.  */ | 
					
						
							|  |  |  |     int allocated; | 
					
						
							|  |  |  |     int tx_fifo_len; | 
					
						
							|  |  |  |     int tx_fifo[NUM_PACKETS]; | 
					
						
							|  |  |  |     int rx_fifo_len; | 
					
						
							|  |  |  |     int rx_fifo[NUM_PACKETS]; | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |     int tx_fifo_done_len; | 
					
						
							|  |  |  |     int tx_fifo_done[NUM_PACKETS]; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     /* Packet buffer memory.  */ | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |     uint8_t data[NUM_PACKETS][2048]; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     uint8_t int_level; | 
					
						
							|  |  |  |     uint8_t int_mask; | 
					
						
							| 
									
										
										
										
											2011-08-24 18:28:21 +01:00
										 |  |  |     MemoryRegion mmio; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } smc91c111_state; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-23 17:19:58 +00:00
										 |  |  | static const VMStateDescription vmstate_smc91c111 = { | 
					
						
							|  |  |  |     .name = "smc91c111", | 
					
						
							|  |  |  |     .version_id = 1, | 
					
						
							|  |  |  |     .minimum_version_id = 1, | 
					
						
							| 
									
										
										
										
											2014-05-13 16:09:35 +01:00
										 |  |  |     .fields = (VMStateField[]) { | 
					
						
							| 
									
										
										
										
											2010-12-23 17:19:58 +00:00
										 |  |  |         VMSTATE_UINT16(tcr, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT16(rcr, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT16(cr, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT16(ctr, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT16(gpr, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT16(ptr, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT16(ercv, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32(bank, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32(packet_num, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32(tx_alloc, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32(allocated, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32(tx_fifo_len, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32_ARRAY(tx_fifo, smc91c111_state, NUM_PACKETS), | 
					
						
							|  |  |  |         VMSTATE_INT32(rx_fifo_len, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32_ARRAY(rx_fifo, smc91c111_state, NUM_PACKETS), | 
					
						
							|  |  |  |         VMSTATE_INT32(tx_fifo_done_len, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_INT32_ARRAY(tx_fifo_done, smc91c111_state, NUM_PACKETS), | 
					
						
							|  |  |  |         VMSTATE_BUFFER_UNSAFE(data, smc91c111_state, 0, NUM_PACKETS * 2048), | 
					
						
							|  |  |  |         VMSTATE_UINT8(int_level, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_UINT8(int_mask, smc91c111_state), | 
					
						
							|  |  |  |         VMSTATE_END_OF_LIST() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | #define RCR_SOFT_RST  0x8000
 | 
					
						
							|  |  |  | #define RCR_STRIP_CRC 0x0200
 | 
					
						
							|  |  |  | #define RCR_RXEN      0x0100
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define TCR_EPH_LOOP  0x2000
 | 
					
						
							|  |  |  | #define TCR_NOCRC     0x0100
 | 
					
						
							|  |  |  | #define TCR_PAD_EN    0x0080
 | 
					
						
							|  |  |  | #define TCR_FORCOL    0x0004
 | 
					
						
							|  |  |  | #define TCR_LOOP      0x0002
 | 
					
						
							|  |  |  | #define TCR_TXEN      0x0001
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define INT_MD        0x80
 | 
					
						
							|  |  |  | #define INT_ERCV      0x40
 | 
					
						
							|  |  |  | #define INT_EPH       0x20
 | 
					
						
							|  |  |  | #define INT_RX_OVRN   0x10
 | 
					
						
							|  |  |  | #define INT_ALLOC     0x08
 | 
					
						
							|  |  |  | #define INT_TX_EMPTY  0x04
 | 
					
						
							|  |  |  | #define INT_TX        0x02
 | 
					
						
							|  |  |  | #define INT_RCV       0x01
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CTR_AUTO_RELEASE  0x0800
 | 
					
						
							|  |  |  | #define CTR_RELOAD        0x0002
 | 
					
						
							|  |  |  | #define CTR_STORE         0x0001
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define RS_ALGNERR      0x8000
 | 
					
						
							|  |  |  | #define RS_BRODCAST     0x4000
 | 
					
						
							|  |  |  | #define RS_BADCRC       0x2000
 | 
					
						
							|  |  |  | #define RS_ODDFRAME     0x1000
 | 
					
						
							|  |  |  | #define RS_TOOLONG      0x0800
 | 
					
						
							|  |  |  | #define RS_TOOSHORT     0x0400
 | 
					
						
							|  |  |  | #define RS_MULTICAST    0x0001
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Update interrupt status.  */ | 
					
						
							|  |  |  | static void smc91c111_update(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int level; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s->tx_fifo_len == 0) | 
					
						
							|  |  |  |         s->int_level |= INT_TX_EMPTY; | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |     if (s->tx_fifo_done_len != 0) | 
					
						
							|  |  |  |         s->int_level |= INT_TX; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     level = (s->int_level & s->int_mask) != 0; | 
					
						
							| 
									
										
										
										
											2007-04-07 18:14:41 +00:00
										 |  |  |     qemu_set_irq(s->irq, level); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:43 -07:00
										 |  |  | static int smc91c111_can_receive(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if ((s->rcr & RCR_RXEN) == 0 || (s->rcr & RCR_SOFT_RST)) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:57 -07:00
										 |  |  |     if (s->allocated == (1 << NUM_PACKETS) - 1 || | 
					
						
							|  |  |  |         s->rx_fifo_len == NUM_PACKETS) { | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:43 -07:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void smc91c111_flush_queued_packets(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (smc91c111_can_receive(s)) { | 
					
						
							|  |  |  |         qemu_flush_queued_packets(qemu_get_queue(s->nic)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | /* Try to allocate a packet.  Returns 0x80 on failure.  */ | 
					
						
							|  |  |  | static int smc91c111_allocate_packet(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     if (s->allocated == (1 << NUM_PACKETS) - 1) { | 
					
						
							|  |  |  |         return 0x80; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < NUM_PACKETS; i++) { | 
					
						
							|  |  |  |         if ((s->allocated & (1 << i)) == 0) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     s->allocated |= 1 << i; | 
					
						
							|  |  |  |     return i; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Process a pending TX allocate.  */ | 
					
						
							|  |  |  | static void smc91c111_tx_alloc(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     s->tx_alloc = smc91c111_allocate_packet(s); | 
					
						
							|  |  |  |     if (s->tx_alloc == 0x80) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     s->int_level |= INT_ALLOC; | 
					
						
							|  |  |  |     smc91c111_update(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Remove and item from the RX FIFO.  */ | 
					
						
							|  |  |  | static void smc91c111_pop_rx_fifo(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s->rx_fifo_len--; | 
					
						
							|  |  |  |     if (s->rx_fifo_len) { | 
					
						
							|  |  |  |         for (i = 0; i < s->rx_fifo_len; i++) | 
					
						
							|  |  |  |             s->rx_fifo[i] = s->rx_fifo[i + 1]; | 
					
						
							|  |  |  |         s->int_level |= INT_RCV; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         s->int_level &= ~INT_RCV; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:57 -07:00
										 |  |  |     smc91c111_flush_queued_packets(s); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     smc91c111_update(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  | /* Remove an item from the TX completion FIFO.  */ | 
					
						
							|  |  |  | static void smc91c111_pop_tx_fifo_done(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s->tx_fifo_done_len == 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     s->tx_fifo_done_len--; | 
					
						
							|  |  |  |     for (i = 0; i < s->tx_fifo_done_len; i++) | 
					
						
							|  |  |  |         s->tx_fifo_done[i] = s->tx_fifo_done[i + 1]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | /* Release the memory allocated to a packet.  */ | 
					
						
							|  |  |  | static void smc91c111_release_packet(smc91c111_state *s, int packet) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     s->allocated &= ~(1 << packet); | 
					
						
							|  |  |  |     if (s->tx_alloc == 0x80) | 
					
						
							|  |  |  |         smc91c111_tx_alloc(s); | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:43 -07:00
										 |  |  |     smc91c111_flush_queued_packets(s); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Flush the TX FIFO.  */ | 
					
						
							|  |  |  | static void smc91c111_do_tx(smc91c111_state *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     int len; | 
					
						
							|  |  |  |     int control; | 
					
						
							|  |  |  |     int packetnum; | 
					
						
							|  |  |  |     uint8_t *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((s->tcr & TCR_TXEN) == 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     if (s->tx_fifo_len == 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     for (i = 0; i < s->tx_fifo_len; i++) { | 
					
						
							|  |  |  |         packetnum = s->tx_fifo[i]; | 
					
						
							|  |  |  |         p = &s->data[packetnum][0]; | 
					
						
							|  |  |  |         /* Set status word.  */ | 
					
						
							|  |  |  |         *(p++) = 0x01; | 
					
						
							|  |  |  |         *(p++) = 0x40; | 
					
						
							|  |  |  |         len = *(p++); | 
					
						
							|  |  |  |         len |= ((int)*(p++)) << 8; | 
					
						
							|  |  |  |         len -= 6; | 
					
						
							|  |  |  |         control = p[len + 1]; | 
					
						
							|  |  |  |         if (control & 0x20) | 
					
						
							|  |  |  |             len++; | 
					
						
							|  |  |  |         /* ??? This overwrites the data following the buffer.
 | 
					
						
							|  |  |  |            Don't know what real hardware does.  */ | 
					
						
							|  |  |  |         if (len < 64 && (s->tcr & TCR_PAD_EN)) { | 
					
						
							|  |  |  |             memset(p + len, 0, 64 - len); | 
					
						
							|  |  |  |             len = 64; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2010-04-25 19:31:06 +00:00
										 |  |  |         { | 
					
						
							|  |  |  |             int add_crc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* The card is supposed to append the CRC to the frame.
 | 
					
						
							|  |  |  |                However none of the other network traffic has the CRC | 
					
						
							|  |  |  |                appended.  Suspect this is low level ethernet detail we | 
					
						
							|  |  |  |                don't need to worry about.  */ | 
					
						
							|  |  |  |             add_crc = (control & 0x10) || (s->tcr & TCR_NOCRC) == 0; | 
					
						
							|  |  |  |             if (add_crc) { | 
					
						
							|  |  |  |                 uint32_t crc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 crc = crc32(~0, p, len); | 
					
						
							|  |  |  |                 memcpy(p + len, &crc, 4); | 
					
						
							|  |  |  |                 len += 4; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |         if (s->ctr & CTR_AUTO_RELEASE) | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |             /* Race?  */ | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |             smc91c111_release_packet(s, packetnum); | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |         else if (s->tx_fifo_done_len < NUM_PACKETS) | 
					
						
							|  |  |  |             s->tx_fifo_done[s->tx_fifo_done_len++] = packetnum; | 
					
						
							| 
									
										
										
										
											2013-01-30 19:12:22 +08:00
										 |  |  |         qemu_send_packet(qemu_get_queue(s->nic), p, len); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     s->tx_fifo_len = 0; | 
					
						
							|  |  |  |     smc91c111_update(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Add a packet to the TX FIFO.  */ | 
					
						
							|  |  |  | static void smc91c111_queue_tx(smc91c111_state *s, int packet) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (s->tx_fifo_len == NUM_PACKETS) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     s->tx_fifo[s->tx_fifo_len++] = packet; | 
					
						
							|  |  |  |     smc91c111_do_tx(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-31 17:48:23 +01:00
										 |  |  | static void smc91c111_reset(DeviceState *dev) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |     smc91c111_state *s = SMC91C111(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     s->bank = 0; | 
					
						
							|  |  |  |     s->tx_fifo_len = 0; | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |     s->tx_fifo_done_len = 0; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     s->rx_fifo_len = 0; | 
					
						
							|  |  |  |     s->allocated = 0; | 
					
						
							|  |  |  |     s->packet_num = 0; | 
					
						
							|  |  |  |     s->tx_alloc = 0; | 
					
						
							|  |  |  |     s->tcr = 0; | 
					
						
							|  |  |  |     s->rcr = 0; | 
					
						
							|  |  |  |     s->cr = 0xa0b1; | 
					
						
							|  |  |  |     s->ctr = 0x1210; | 
					
						
							|  |  |  |     s->ptr = 0; | 
					
						
							|  |  |  |     s->ercv = 0x1f; | 
					
						
							|  |  |  |     s->int_level = INT_TX_EMPTY; | 
					
						
							|  |  |  |     s->int_mask = 0; | 
					
						
							|  |  |  |     smc91c111_update(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define SET_LOW(name, val) s->name = (s->name & 0xff00) | val
 | 
					
						
							|  |  |  | #define SET_HIGH(name, val) s->name = (s->name & 0xff) | (val << 8)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 12:30:10 +02:00
										 |  |  | static void smc91c111_writeb(void *opaque, hwaddr offset, | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |                              uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     smc91c111_state *s = (smc91c111_state *)opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-29 15:16:58 +02:00
										 |  |  |     offset = offset & 0xf; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     if (offset == 14) { | 
					
						
							|  |  |  |         s->bank = value; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (offset == 15) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     switch (s->bank) { | 
					
						
							|  |  |  |     case 0: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: /* TCR */ | 
					
						
							|  |  |  |             SET_LOW(tcr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 1: | 
					
						
							|  |  |  |             SET_HIGH(tcr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 4: /* RCR */ | 
					
						
							|  |  |  |             SET_LOW(rcr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 5: | 
					
						
							|  |  |  |             SET_HIGH(rcr, value); | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |             if (s->rcr & RCR_SOFT_RST) { | 
					
						
							|  |  |  |                 smc91c111_reset(DEVICE(s)); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-09-10 21:24:12 -07:00
										 |  |  |             smc91c111_flush_queued_packets(s); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |             return; | 
					
						
							|  |  |  |         case 10: case 11: /* RPCR */ | 
					
						
							|  |  |  |             /* Ignored */ | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-03-29 15:17:18 +02:00
										 |  |  |         case 12: case 13: /* Reserved */ | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case 1: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: /* CONFIG */ | 
					
						
							|  |  |  |             SET_LOW(cr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 1: | 
					
						
							|  |  |  |             SET_HIGH(cr,value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 2: case 3: /* BASE */ | 
					
						
							|  |  |  |         case 4: case 5: case 6: case 7: case 8: case 9: /* IA */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 10: /* Genral Purpose */ | 
					
						
							|  |  |  |             SET_LOW(gpr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 11: | 
					
						
							|  |  |  |             SET_HIGH(gpr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 12: /* Control */ | 
					
						
							| 
									
										
										
										
											2018-06-26 17:50:41 +01:00
										 |  |  |             if (value & 1) { | 
					
						
							|  |  |  |                 qemu_log_mask(LOG_UNIMP, | 
					
						
							|  |  |  |                               "smc91c111: EEPROM store not implemented\n"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (value & 2) { | 
					
						
							|  |  |  |                 qemu_log_mask(LOG_UNIMP, | 
					
						
							|  |  |  |                               "smc91c111: EEPROM reload not implemented\n"); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |             value &= ~3; | 
					
						
							|  |  |  |             SET_LOW(ctr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 13: | 
					
						
							|  |  |  |             SET_HIGH(ctr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case 2: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: /* MMU Command */ | 
					
						
							|  |  |  |             switch (value >> 5) { | 
					
						
							|  |  |  |             case 0: /* no-op */ | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 1: /* Allocate for TX.  */ | 
					
						
							|  |  |  |                 s->tx_alloc = 0x80; | 
					
						
							|  |  |  |                 s->int_level &= ~INT_ALLOC; | 
					
						
							|  |  |  |                 smc91c111_update(s); | 
					
						
							|  |  |  |                 smc91c111_tx_alloc(s); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 2: /* Reset MMU.  */ | 
					
						
							|  |  |  |                 s->allocated = 0; | 
					
						
							|  |  |  |                 s->tx_fifo_len = 0; | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |                 s->tx_fifo_done_len = 0; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |                 s->rx_fifo_len = 0; | 
					
						
							|  |  |  |                 s->tx_alloc = 0; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 3: /* Remove from RX FIFO.  */ | 
					
						
							|  |  |  |                 smc91c111_pop_rx_fifo(s); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 4: /* Remove from RX FIFO and release.  */ | 
					
						
							|  |  |  |                 if (s->rx_fifo_len > 0) { | 
					
						
							|  |  |  |                     smc91c111_release_packet(s, s->rx_fifo[0]); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 smc91c111_pop_rx_fifo(s); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 5: /* Release.  */ | 
					
						
							|  |  |  |                 smc91c111_release_packet(s, s->packet_num); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 6: /* Add to TX FIFO.  */ | 
					
						
							|  |  |  |                 smc91c111_queue_tx(s, s->packet_num); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 7: /* Reset TX FIFO.  */ | 
					
						
							|  |  |  |                 s->tx_fifo_len = 0; | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |                 s->tx_fifo_done_len = 0; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 1: | 
					
						
							|  |  |  |             /* Ignore.  */ | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 2: /* Packet Number Register */ | 
					
						
							|  |  |  |             s->packet_num = value; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 3: case 4: case 5: | 
					
						
							|  |  |  |             /* Should be readonly, but linux writes to them anyway. Ignore.  */ | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 6: /* Pointer */ | 
					
						
							|  |  |  |             SET_LOW(ptr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 7: | 
					
						
							|  |  |  |             SET_HIGH(ptr, value); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 8: case 9: case 10: case 11: /* Data */ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 int p; | 
					
						
							|  |  |  |                 int n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (s->ptr & 0x8000) | 
					
						
							|  |  |  |                     n = s->rx_fifo[0]; | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     n = s->packet_num; | 
					
						
							|  |  |  |                 p = s->ptr & 0x07ff; | 
					
						
							|  |  |  |                 if (s->ptr & 0x4000) { | 
					
						
							|  |  |  |                     s->ptr = (s->ptr & 0xf800) | ((s->ptr + 1) & 0x7ff); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     p += (offset & 3); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 s->data[n][p] = value; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 12: /* Interrupt ACK.  */ | 
					
						
							|  |  |  |             s->int_level &= ~(value & 0xd6); | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |             if (value & INT_TX) | 
					
						
							|  |  |  |                 smc91c111_pop_tx_fifo_done(s); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |             smc91c111_update(s); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 13: /* Interrupt mask.  */ | 
					
						
							|  |  |  |             s->int_mask = value; | 
					
						
							|  |  |  |             smc91c111_update(s); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-11-29 16:52:38 +08:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case 3: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: case 1: case 2: case 3: case 4: case 5: case 6: case 7: | 
					
						
							|  |  |  |             /* Multicast table.  */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 8: case 9: /* Management Interface.  */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         case 12: /* Early receive.  */ | 
					
						
							|  |  |  |             s->ercv = value & 0x1f; | 
					
						
							| 
									
										
										
										
											2013-01-21 12:50:55 +00:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         case 13: | 
					
						
							|  |  |  |             /* Ignore.  */ | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-26 17:50:41 +01:00
										 |  |  |     qemu_log_mask(LOG_GUEST_ERROR, "smc91c111_write(bank:%d) Illegal register" | 
					
						
							|  |  |  |                                    " 0x%" HWADDR_PRIx " = 0x%x\n", | 
					
						
							|  |  |  |                   s->bank, offset, value); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-23 12:30:10 +02:00
										 |  |  | static uint32_t smc91c111_readb(void *opaque, hwaddr offset) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     smc91c111_state *s = (smc91c111_state *)opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-29 15:16:58 +02:00
										 |  |  |     offset = offset & 0xf; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     if (offset == 14) { | 
					
						
							|  |  |  |         return s->bank; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (offset == 15) | 
					
						
							|  |  |  |         return 0x33; | 
					
						
							|  |  |  |     switch (s->bank) { | 
					
						
							|  |  |  |     case 0: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: /* TCR */ | 
					
						
							|  |  |  |             return s->tcr & 0xff; | 
					
						
							|  |  |  |         case 1: | 
					
						
							|  |  |  |             return s->tcr >> 8; | 
					
						
							|  |  |  |         case 2: /* EPH Status */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         case 3: | 
					
						
							|  |  |  |             return 0x40; | 
					
						
							|  |  |  |         case 4: /* RCR */ | 
					
						
							|  |  |  |             return s->rcr & 0xff; | 
					
						
							|  |  |  |         case 5: | 
					
						
							|  |  |  |             return s->rcr >> 8; | 
					
						
							|  |  |  |         case 6: /* Counter */ | 
					
						
							|  |  |  |         case 7: | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2007-04-02 08:18:36 +00:00
										 |  |  |         case 8: /* Memory size.  */ | 
					
						
							|  |  |  |             return NUM_PACKETS; | 
					
						
							|  |  |  |         case 9: /* Free memory available.  */ | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |             { | 
					
						
							|  |  |  |                 int i; | 
					
						
							|  |  |  |                 int n; | 
					
						
							|  |  |  |                 n = 0; | 
					
						
							|  |  |  |                 for (i = 0; i < NUM_PACKETS; i++) { | 
					
						
							|  |  |  |                     if (s->allocated & (1 << i)) | 
					
						
							|  |  |  |                         n++; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 return n; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         case 10: case 11: /* RPCR */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2010-03-29 15:17:18 +02:00
										 |  |  |         case 12: case 13: /* Reserved */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case 1: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: /* CONFIG */ | 
					
						
							|  |  |  |             return s->cr & 0xff; | 
					
						
							|  |  |  |         case 1: | 
					
						
							|  |  |  |             return s->cr >> 8; | 
					
						
							|  |  |  |         case 2: case 3: /* BASE */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         case 4: case 5: case 6: case 7: case 8: case 9: /* IA */ | 
					
						
							| 
									
										
										
										
											2009-10-21 15:25:37 +02:00
										 |  |  |             return s->conf.macaddr.a[offset - 4]; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         case 10: /* General Purpose */ | 
					
						
							|  |  |  |             return s->gpr & 0xff; | 
					
						
							|  |  |  |         case 11: | 
					
						
							|  |  |  |             return s->gpr >> 8; | 
					
						
							|  |  |  |         case 12: /* Control */ | 
					
						
							|  |  |  |             return s->ctr & 0xff; | 
					
						
							|  |  |  |         case 13: | 
					
						
							|  |  |  |             return s->ctr >> 8; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case 2: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: case 1: /* MMUCR Busy bit.  */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         case 2: /* Packet Number.  */ | 
					
						
							|  |  |  |             return s->packet_num; | 
					
						
							|  |  |  |         case 3: /* Allocation Result.  */ | 
					
						
							|  |  |  |             return s->tx_alloc; | 
					
						
							|  |  |  |         case 4: /* TX FIFO */ | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |             if (s->tx_fifo_done_len == 0) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |                 return 0x80; | 
					
						
							|  |  |  |             else | 
					
						
							| 
									
										
										
										
											2005-12-18 17:39:52 +00:00
										 |  |  |                 return s->tx_fifo_done[0]; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         case 5: /* RX FIFO */ | 
					
						
							|  |  |  |             if (s->rx_fifo_len == 0) | 
					
						
							|  |  |  |                 return 0x80; | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 return s->rx_fifo[0]; | 
					
						
							|  |  |  |         case 6: /* Pointer */ | 
					
						
							|  |  |  |             return s->ptr & 0xff; | 
					
						
							|  |  |  |         case 7: | 
					
						
							|  |  |  |             return (s->ptr >> 8) & 0xf7; | 
					
						
							|  |  |  |         case 8: case 9: case 10: case 11: /* Data */ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 int p; | 
					
						
							|  |  |  |                 int n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (s->ptr & 0x8000) | 
					
						
							|  |  |  |                     n = s->rx_fifo[0]; | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     n = s->packet_num; | 
					
						
							|  |  |  |                 p = s->ptr & 0x07ff; | 
					
						
							|  |  |  |                 if (s->ptr & 0x4000) { | 
					
						
							|  |  |  |                     s->ptr = (s->ptr & 0xf800) | ((s->ptr + 1) & 0x07ff); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     p += (offset & 3); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 return s->data[n][p]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         case 12: /* Interrupt status.  */ | 
					
						
							|  |  |  |             return s->int_level; | 
					
						
							|  |  |  |         case 13: /* Interrupt mask.  */ | 
					
						
							|  |  |  |             return s->int_mask; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case 3: | 
					
						
							|  |  |  |         switch (offset) { | 
					
						
							|  |  |  |         case 0: case 1: case 2: case 3: case 4: case 5: case 6: case 7: | 
					
						
							|  |  |  |             /* Multicast table.  */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         case 8: /* Management Interface.  */ | 
					
						
							|  |  |  |             /* Not implemented.  */ | 
					
						
							|  |  |  |             return 0x30; | 
					
						
							|  |  |  |         case 9: | 
					
						
							|  |  |  |             return 0x33; | 
					
						
							|  |  |  |         case 10: /* Revision.  */ | 
					
						
							|  |  |  |             return 0x91; | 
					
						
							|  |  |  |         case 11: | 
					
						
							|  |  |  |             return 0x33; | 
					
						
							|  |  |  |         case 12: | 
					
						
							|  |  |  |             return s->ercv; | 
					
						
							|  |  |  |         case 13: | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-26 17:50:41 +01:00
										 |  |  |     qemu_log_mask(LOG_GUEST_ERROR, "smc91c111_read(bank:%d) Illegal register" | 
					
						
							|  |  |  |                                    " 0x%" HWADDR_PRIx "\n", | 
					
						
							|  |  |  |                   s->bank, offset); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-04 18:05:50 +01:00
										 |  |  | static uint64_t smc91c111_readfn(void *opaque, hwaddr addr, unsigned size) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-05-04 18:05:50 +01:00
										 |  |  |     int i; | 
					
						
							|  |  |  |     uint32_t val = 0; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-04 18:05:50 +01:00
										 |  |  |     for (i = 0; i < size; i++) { | 
					
						
							|  |  |  |         val |= smc91c111_readb(opaque, addr + i) << (i * 8); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     return val; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-04 18:05:50 +01:00
										 |  |  | static void smc91c111_writefn(void *opaque, hwaddr addr, | 
					
						
							|  |  |  |                                uint64_t value, unsigned size) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-05-04 18:05:50 +01:00
										 |  |  |     int i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* 32-bit writes to offset 0xc only actually write to the bank select
 | 
					
						
							|  |  |  |      * register (offset 0xe), so skip the first two bytes we would write. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     if (addr == 0xc && size == 4) { | 
					
						
							|  |  |  |         i += 2; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (; i < size; i++) { | 
					
						
							|  |  |  |         smc91c111_writeb(opaque, addr + i, | 
					
						
							|  |  |  |                          extract32(value, i * 8, 8)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:43 -07:00
										 |  |  | static int smc91c111_can_receive_nc(NetClientState *nc) | 
					
						
							| 
									
										
										
										
											2006-02-04 22:15:28 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-30 19:12:23 +08:00
										 |  |  |     smc91c111_state *s = qemu_get_nic_opaque(nc); | 
					
						
							| 
									
										
										
										
											2006-02-04 22:15:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:43 -07:00
										 |  |  |     return smc91c111_can_receive(s); | 
					
						
							| 
									
										
										
										
											2006-02-04 22:15:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-24 16:35:13 +01:00
										 |  |  | static ssize_t smc91c111_receive(NetClientState *nc, const uint8_t *buf, size_t size) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-30 19:12:23 +08:00
										 |  |  |     smc91c111_state *s = qemu_get_nic_opaque(nc); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     int status; | 
					
						
							|  |  |  |     int packetsize; | 
					
						
							|  |  |  |     uint32_t crc; | 
					
						
							|  |  |  |     int packetnum; | 
					
						
							|  |  |  |     uint8_t *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((s->rcr & RCR_RXEN) == 0 || (s->rcr & RCR_SOFT_RST)) | 
					
						
							| 
									
										
										
										
											2009-05-18 13:40:55 +01:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2006-12-07 18:28:42 +00:00
										 |  |  |     /* Short packets are padded with zeros.  Receiving a packet
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |        < 64 bytes long is considered an error condition.  */ | 
					
						
							|  |  |  |     if (size < 64) | 
					
						
							|  |  |  |         packetsize = 64; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         packetsize = (size & ~1); | 
					
						
							|  |  |  |     packetsize += 6; | 
					
						
							|  |  |  |     crc = (s->rcr & RCR_STRIP_CRC) == 0; | 
					
						
							|  |  |  |     if (crc) | 
					
						
							|  |  |  |         packetsize += 4; | 
					
						
							|  |  |  |     /* TODO: Flag overrun and receive errors.  */ | 
					
						
							|  |  |  |     if (packetsize > 2048) | 
					
						
							| 
									
										
										
										
											2009-05-18 13:40:55 +01:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     packetnum = smc91c111_allocate_packet(s); | 
					
						
							|  |  |  |     if (packetnum == 0x80) | 
					
						
							| 
									
										
										
										
											2009-05-18 13:40:55 +01:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     s->rx_fifo[s->rx_fifo_len++] = packetnum; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     p = &s->data[packetnum][0]; | 
					
						
							|  |  |  |     /* ??? Multicast packets?  */ | 
					
						
							|  |  |  |     status = 0; | 
					
						
							|  |  |  |     if (size > 1518) | 
					
						
							|  |  |  |         status |= RS_TOOLONG; | 
					
						
							|  |  |  |     if (size & 1) | 
					
						
							|  |  |  |         status |= RS_ODDFRAME; | 
					
						
							|  |  |  |     *(p++) = status & 0xff; | 
					
						
							|  |  |  |     *(p++) = status >> 8; | 
					
						
							|  |  |  |     *(p++) = packetsize & 0xff; | 
					
						
							|  |  |  |     *(p++) = packetsize >> 8; | 
					
						
							|  |  |  |     memcpy(p, buf, size & ~1); | 
					
						
							|  |  |  |     p += (size & ~1); | 
					
						
							|  |  |  |     /* Pad short packets.  */ | 
					
						
							|  |  |  |     if (size < 64) { | 
					
						
							|  |  |  |         int pad; | 
					
						
							| 
									
										
										
										
											2007-09-17 08:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |         if (size & 1) | 
					
						
							|  |  |  |             *(p++) = buf[size - 1]; | 
					
						
							|  |  |  |         pad = 64 - size; | 
					
						
							|  |  |  |         memset(p, 0, pad); | 
					
						
							|  |  |  |         p += pad; | 
					
						
							|  |  |  |         size = 64; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* It's not clear if the CRC should go before or after the last byte in
 | 
					
						
							|  |  |  |        odd sized packets.  Linux disables the CRC, so that's no help. | 
					
						
							|  |  |  |        The pictures in the documentation show the CRC aligned on a 16-bit | 
					
						
							|  |  |  |        boundary before the last odd byte, so that's what we do.  */ | 
					
						
							|  |  |  |     if (crc) { | 
					
						
							|  |  |  |         crc = crc32(~0, buf, size); | 
					
						
							|  |  |  |         *(p++) = crc & 0xff; crc >>= 8; | 
					
						
							|  |  |  |         *(p++) = crc & 0xff; crc >>= 8; | 
					
						
							|  |  |  |         *(p++) = crc & 0xff; crc >>= 8; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:31:06 +00:00
										 |  |  |         *(p++) = crc & 0xff; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (size & 1) { | 
					
						
							|  |  |  |         *(p++) = buf[size - 1]; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:31:06 +00:00
										 |  |  |         *p = 0x60; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         *(p++) = 0; | 
					
						
							| 
									
										
										
										
											2010-04-25 19:31:06 +00:00
										 |  |  |         *p = 0x40; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     /* TODO: Raise early RX interrupt?  */ | 
					
						
							|  |  |  |     s->int_level |= INT_RCV; | 
					
						
							|  |  |  |     smc91c111_update(s); | 
					
						
							| 
									
										
										
										
											2009-05-18 13:40:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return size; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-24 18:28:21 +01:00
										 |  |  | static const MemoryRegionOps smc91c111_mem_ops = { | 
					
						
							|  |  |  |     /* The special case for 32 bit writes to 0xc means we can't just
 | 
					
						
							|  |  |  |      * set .impl.min/max_access_size to 1, unfortunately | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2018-05-04 18:05:50 +01:00
										 |  |  |     .read = smc91c111_readfn, | 
					
						
							|  |  |  |     .write = smc91c111_writefn, | 
					
						
							|  |  |  |     .valid.min_access_size = 1, | 
					
						
							|  |  |  |     .valid.max_access_size = 4, | 
					
						
							| 
									
										
										
										
											2011-08-24 18:28:21 +01:00
										 |  |  |     .endianness = DEVICE_NATIVE_ENDIAN, | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 18:49:23 +00:00
										 |  |  | static NetClientInfo net_smc91c111_info = { | 
					
						
							| 
									
										
											  
											
												qapi: Change Netdev into a flat union
This is a mostly-mechanical conversion that creates a new flat
union 'Netdev' QAPI type that covers all the branches of the
former 'NetClientOptions' simple union, where the branches are
now listed in a new 'NetClientDriver' enum rather than generated
from the simple union.  The existence of a flat union has no
change to the command line syntax accepted for new code, and
will make it possible for a future patch to switch the QMP
command to parse a boxed union for no change to valid QMP; but
it does have some ripple effect on the C code when dealing with
the new types.
While making the conversion, note that the 'NetLegacy' type
remains unchanged: it applies only to legacy command line options,
and will not be ported to QMP, so it should remain a wrapper
around a simple union; to avoid confusion, the type named
'NetClientOptions' is now gone, and we introduce 'NetLegacyOptions'
in its place.  Then, in the C code, we convert from NetLegacy to
Netdev as soon as possible, so that the bulk of the net stack
only has to deal with one QAPI type, not two.  Note that since
the old legacy code always rejected 'hubport', we can just omit
that branch from the new 'NetLegacyOptions' simple union.
Based on an idea originally by Zoltán Kővágó <DirtY.iCE.hu@gmail.com>:
Message-Id: <01a527fbf1a5de880091f98cf011616a78adeeee.1441627176.git.DirtY.iCE.hu@gmail.com>
although the sed script in that patch no longer applies due to
other changes in the tree since then, and I also did some manual
cleanups (such as fixing whitespace to keep checkpatch happy).
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1468468228-27827-13-git-send-email-eblake@redhat.com>
Reviewed-by: Markus Armbruster <armbru@redhat.com>
[Fixup from Eric squashed in]
Signed-off-by: Markus Armbruster <armbru@redhat.com>
											
										 
											2016-07-13 21:50:23 -06:00
										 |  |  |     .type = NET_CLIENT_DRIVER_NIC, | 
					
						
							| 
									
										
										
										
											2009-11-25 18:49:23 +00:00
										 |  |  |     .size = sizeof(NICState), | 
					
						
							| 
									
										
										
										
											2015-09-10 21:23:43 -07:00
										 |  |  |     .can_receive = smc91c111_can_receive_nc, | 
					
						
							| 
									
										
										
										
											2009-11-25 18:49:23 +00:00
										 |  |  |     .receive = smc91c111_receive, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  | static int smc91c111_init1(SysBusDevice *sbd) | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |     DeviceState *dev = DEVICE(sbd); | 
					
						
							|  |  |  |     smc91c111_state *s = SMC91C111(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 21:25:08 -04:00
										 |  |  |     memory_region_init_io(&s->mmio, OBJECT(s), &smc91c111_mem_ops, s, | 
					
						
							| 
									
										
										
										
											2011-08-24 18:28:21 +01:00
										 |  |  |                           "smc91c111-mmio", 16); | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |     sysbus_init_mmio(sbd, &s->mmio); | 
					
						
							|  |  |  |     sysbus_init_irq(sbd, &s->irq); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:25:37 +02:00
										 |  |  |     qemu_macaddr_default_if_unset(&s->conf.macaddr); | 
					
						
							| 
									
										
										
										
											2009-11-25 18:49:23 +00:00
										 |  |  |     s->nic = qemu_new_nic(&net_smc91c111_info, &s->conf, | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |                           object_get_typename(OBJECT(dev)), dev->id, s); | 
					
						
							| 
									
										
										
										
											2013-01-30 19:12:22 +08:00
										 |  |  |     qemu_format_nic_info_str(qemu_get_queue(s->nic), s->conf.macaddr.a); | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  |     /* ??? Save/restore.  */ | 
					
						
							| 
									
										
										
										
											2009-08-14 10:36:05 +02:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2005-12-04 18:54:21 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-05-14 22:35:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-24 13:12:29 -06:00
										 |  |  | static Property smc91c111_properties[] = { | 
					
						
							|  |  |  |     DEFINE_NIC_PROPERTIES(smc91c111_state, conf), | 
					
						
							|  |  |  |     DEFINE_PROP_END_OF_LIST(), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void smc91c111_class_init(ObjectClass *klass, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     DeviceClass *dc = DEVICE_CLASS(klass); | 
					
						
							| 
									
										
										
										
											2012-01-24 13:12:29 -06:00
										 |  |  |     SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     k->init = smc91c111_init1; | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     dc->reset = smc91c111_reset; | 
					
						
							|  |  |  |     dc->vmsd = &vmstate_smc91c111; | 
					
						
							|  |  |  |     dc->props = smc91c111_properties; | 
					
						
							| 
									
										
										
										
											2012-01-24 13:12:29 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 16:19:07 +01:00
										 |  |  | static const TypeInfo smc91c111_info = { | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |     .name          = TYPE_SMC91C111, | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     .parent        = TYPE_SYS_BUS_DEVICE, | 
					
						
							|  |  |  |     .instance_size = sizeof(smc91c111_state), | 
					
						
							|  |  |  |     .class_init    = smc91c111_class_init, | 
					
						
							| 
									
										
										
										
											2009-10-21 15:25:37 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | static void smc91c111_register_types(void) | 
					
						
							| 
									
										
										
										
											2009-05-14 22:35:07 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     type_register_static(&smc91c111_info); | 
					
						
							| 
									
										
										
										
											2009-05-14 22:35:07 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Legacy helper function.  Should go away when machine config files are
 | 
					
						
							|  |  |  |    implemented.  */ | 
					
						
							|  |  |  | void smc91c111_init(NICInfo *nd, uint32_t base, qemu_irq irq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DeviceState *dev; | 
					
						
							|  |  |  |     SysBusDevice *s; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_check_nic_model(nd, "smc91c111"); | 
					
						
							| 
									
										
										
										
											2013-07-27 12:17:48 +02:00
										 |  |  |     dev = qdev_create(NULL, TYPE_SMC91C111); | 
					
						
							| 
									
										
										
										
											2009-10-21 15:25:37 +02:00
										 |  |  |     qdev_set_nic_properties(dev, nd); | 
					
						
							| 
									
										
										
										
											2009-10-07 01:15:58 +02:00
										 |  |  |     qdev_init_nofail(dev); | 
					
						
							| 
									
										
										
										
											2013-01-20 02:47:33 +01:00
										 |  |  |     s = SYS_BUS_DEVICE(dev); | 
					
						
							| 
									
										
										
										
											2009-05-14 22:35:07 +01:00
										 |  |  |     sysbus_mmio_map(s, 0, base); | 
					
						
							|  |  |  |     sysbus_connect_irq(s, 0, irq); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | type_init(smc91c111_register_types) |