VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DevEEPROM.cpp@ 25353

Last change on this file since 25353 was 24038, checked in by vboxsync, 15 years ago

DevEEPROM.cpp: it's sufficient to return the status of the last SSM operation in EEPROM93C46::load() - SSM remembers errors.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.0 KB
Line 
1/* $Id: DevEEPROM.cpp 24038 2009-10-23 14:17:37Z vboxsync $ */
2/** @file
3 * DevEEPROM - Microware-compatible 64x16-bit 93C46 EEPROM Emulation.
4 */
5
6/*
7 * Copyright (C) 2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#define LOG_GROUP LOG_GROUP_DEV_E1000 /// @todo Add a EEPROM logging group.
23#include <VBox/log.h>
24#include <VBox/pdmdev.h>
25#include <iprt/string.h>
26#include "DevEEPROM.h"
27
28#define E1kLog(a) Log(a)
29
30/**
31 * Initialize EEPROM device.
32 *
33 * @param pu16Initial Initial EEPROM content (optional). The size of initial
34 * content must be sizeof(uint16_t)*EEPROM93C46::SIZE
35 * bytes.
36 */
37void EEPROM93C46::init(const uint16_t *pu16Initial)
38{
39 if ( pu16Initial ) {
40 memcpy(this->m_au16Data, pu16Initial, sizeof(this->m_au16Data));
41 } else {
42 memset(this->m_au16Data, 0, sizeof(this->m_au16Data));
43 }
44 m_fWriteEnabled = false;
45 m_u32InternalWires = 0;
46 m_eState = STANDBY;
47}
48
49/**
50 * Writes one word to specified location if write is enabled.
51 *
52 * @param u32Addr Address to write at
53 * @param u16Value Value to store
54 */
55void EEPROM93C46::storeWord(uint32_t u32Addr, uint16_t u16Value)
56{
57 if (m_fWriteEnabled) {
58 E1kLog(("EEPROM: Stored word %04x at %08x\n", u16Value, u32Addr));
59 m_au16Data[u32Addr] = u16Value;
60 }
61 m_u16Mask = DATA_MSB;
62}
63
64/**
65 * Fetch next word pointer by m_u16Addr.
66 *
67 * m_u16Addr is advanced and mask is reset to support sequential reads.
68 *
69 * @returns New state
70 */
71EEPROM93C46::State EEPROM93C46::opRead()
72{
73 m_u16Word = m_au16Data[m_u16Addr++];
74 E1kLog(("EEPROM: Reading word %04x at %08x\n", m_u16Word, m_u16Addr-1));
75 m_u16Mask = DATA_MSB;
76 return WRITING_DO;
77}
78
79/**
80 * Write the value of m_u16Word to the location specified by m_u16Addr.
81 *
82 * @returns New state
83 *
84 * @remarks Need to wait for CS lower/raise to show busy/ready indication.
85 */
86EEPROM93C46::State EEPROM93C46::opWrite()
87{
88 storeWord(m_u16Addr, m_u16Word);
89 return WAITING_CS_FALL;
90}
91
92/**
93 * Overwrite the entire contents of EEPROM with the value of m_u16Word.
94 *
95 * @returns New state
96 *
97 * @remarks Need to wait for CS lower/raise to show busy/ready indication.
98 */
99EEPROM93C46::State EEPROM93C46::opWriteAll()
100{
101 for (int i = 0; i < SIZE; i++)
102 storeWord(i, m_u16Word);
103 return WAITING_CS_FALL;
104}
105
106/**
107 * Decode opcode and address from 'opAddr' member.
108 *
109 * Decodes operation and executes it immediately if possible; otherwise, stores
110 * the decoded operation and address.
111 *
112 * @returns New state
113 */
114EEPROM93C46::State EEPROM93C46::opDecode()
115{
116 switch (m_u16Word>>6) {
117 case 3: /* ERASE */
118 storeWord(m_u16Word & ADDR_MASK, 0xFFFF);
119 return WAITING_CS_FALL;
120 case 2: /* READ */
121 m_eOp = OP_READ;
122 m_u16Addr = m_u16Word & ADDR_MASK;
123 return opRead(); /* Load first word */
124 case 1: /* WRITE */
125 m_eOp = OP_WRITE;
126 m_u16Addr = m_u16Word & ADDR_MASK;
127 m_u16Word = 0;
128 m_u16Mask = DATA_MSB;
129 return READING_DI;
130 case 0:
131 switch (m_u16Word>>4) {
132 case 0: /* ERASE/WRITE DISABLE */
133 m_fWriteEnabled = false;
134 return STANDBY;
135 case 1: /* WRITE ALL */
136 m_eOp = OP_WRITE_ALL;
137 m_u16Word = 0;
138 m_u16Mask = DATA_MSB;
139 return READING_DI;
140 case 2: /* ERASE ALL */
141 /* Re-use opWriteAll */
142 m_u16Word = 0xFFFF;
143 return opWriteAll();
144 case 3: /* ERASE/WRITE ENABLE */
145 m_fWriteEnabled = true;
146 return STANDBY;
147 }
148 }
149 return m_eState;
150}
151
152/**
153 * Set bits in EEPROM 4-wire interface.
154 *
155 * @param u32Wires Values of DI, CS, SK.
156 * @remarks The value of DO bit in 'u32Wires' is ignored.
157 */
158void EEPROM93C46::write(uint32_t u32Wires)
159{
160 if (u32Wires & WIRES_CS) {
161 if (!(m_u32InternalWires & WIRES_SK) && (u32Wires & WIRES_SK)) {
162 /* Positive edge of clock */
163 if (m_eState == STANDBY) {
164 if (u32Wires & WIRES_DI) {
165 m_eState = READING_DI;
166 m_eOp = OP_DECODE;
167 m_u16Mask = OPADDR_MSB;
168 m_u16Word = 0;
169 }
170 }
171 else {
172 if (m_eState == READING_DI) {
173 if (u32Wires & WIRES_DI) {
174 m_u16Word |= m_u16Mask;
175 }
176 }
177 else if (m_eState == WRITING_DO) {
178 m_u32InternalWires &= ~WIRES_DO;
179 if (m_u16Word & m_u16Mask) {
180 m_u32InternalWires |= WIRES_DO;
181 }
182 }
183 else return;
184 /* Next bit */
185 m_u16Mask >>= 1;
186 if (m_u16Mask == 0)
187 {
188 switch (this->m_eOp)
189 {
190 case OP_READ:
191 m_eState = opRead();
192 break;
193 case OP_WRITE:
194 m_eState = opWrite();
195 break;
196 case OP_WRITE_ALL:
197 m_eState = opWriteAll();
198 break;
199 case OP_DECODE:
200 m_eState = opDecode();
201 break;
202 default:
203 ;
204 }
205 }
206 }
207 }
208 else if (m_eState == WAITING_CS_RISE) {
209 m_u32InternalWires |= WIRES_DO; /* ready */
210 m_eState = STANDBY;
211 }
212 }
213 else {
214 switch(m_eState) {
215 case WAITING_CS_FALL:
216 m_eState = WAITING_CS_RISE;
217 m_u32InternalWires &= ~WIRES_DO; /* busy */
218 break;
219 case WAITING_CS_RISE:
220 break;
221 case READING_DI:
222 m_u32InternalWires &= ~WIRES_DO; /* Clear ready/busy status from DO. */
223 /* Fall through! */
224 default:
225 m_eState = STANDBY;
226 break;
227 }
228 }
229 m_u32InternalWires &= WIRES_DO;
230 m_u32InternalWires |= u32Wires & ~WIRES_DO; /* Do not overwrite DO */
231}
232
233/**
234 * Read bits in EEPROM 4-wire interface.
235 *
236 * @returns Current values of DO, DI, CS, SK.
237 *
238 * @remarks Only DO is controlled by EEPROM, other bits are returned as they
239 * were written by 'write'.
240 */
241uint32_t EEPROM93C46::read()
242{
243 return m_u32InternalWires;
244}
245
246void EEPROM93C46::save(PSSMHANDLE pSSM)
247{
248 SSMR3PutU8( pSSM, EEPROM93C46_SAVEDSTATE_VERSION);
249 SSMR3PutU8( pSSM, m_eState);
250 SSMR3PutU8( pSSM, m_eOp);
251 SSMR3PutBool(pSSM, m_fWriteEnabled);
252 SSMR3PutU32( pSSM, m_u32InternalWires);
253 SSMR3PutU16( pSSM, m_u16Word);
254 SSMR3PutU16( pSSM, m_u16Mask);
255 SSMR3PutU16( pSSM, m_u16Addr);
256 SSMR3PutMem( pSSM, m_au16Data, sizeof(m_au16Data));
257}
258
259int EEPROM93C46::load(PSSMHANDLE pSSM)
260{
261 int rc;
262 uint8_t uVersion;
263
264 rc = SSMR3GetU8(pSSM, &uVersion);
265 AssertRCReturn(rc, rc);
266 if (uVersion != EEPROM93C46_SAVEDSTATE_VERSION)
267 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
268
269 SSMR3GetU8( pSSM, (uint8_t*)&m_eState);
270 SSMR3GetU8( pSSM, (uint8_t*)&m_eOp);
271 SSMR3GetBool(pSSM, &m_fWriteEnabled);
272 SSMR3GetU32( pSSM, &m_u32InternalWires);
273 SSMR3GetU16( pSSM, &m_u16Word);
274 SSMR3GetU16( pSSM, &m_u16Mask);
275 SSMR3GetU16( pSSM, &m_u16Addr);
276 return SSMR3GetMem( pSSM, m_au16Data, sizeof(m_au16Data));
277}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette