gem5  v20.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
bus.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2012 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 #include "dev/i2c/bus.hh"
39 
40 #include "debug/Checkpoint.hh"
41 #include "dev/i2c/device.hh"
42 #include "mem/packet_access.hh"
43 
44 // clang complains about std::set being overloaded with Packet::set if
45 // we open up the entire namespace std
46 using std::vector;
47 using std::map;
48 
53 I2CBus::I2CBus(const I2CBusParams *p)
54  : BasicPioDevice(p, 0x1000), scl(1), sda(1), state(IDLE), currBit(7),
55  i2cAddr(0x00), message(0x00)
56 {
57  vector<I2CDevice*> devs = p->devices;
58 
59  for (auto d : p->devices) {
60  devices[d->i2cAddr()] = d;
61  }
62 }
63 
68 Tick
70 {
71  assert(pkt->getAddr() == pioAddr + SB_CONTROLS);
72 
73  pkt->setRaw<uint8_t>((sda << 1) | scl);
74  pkt->makeAtomicResponse();
75  return pioDelay;
76 }
77 
87 Tick
89 {
90  assert(pkt->getAddr() == pioAddr + SB_CONTROLS ||
91  pkt->getAddr() == pioAddr + SB_CONTROLC);
92 
93  updateSignals(pkt);
94 
95  // Check if the bus master is starting a new transmission.
96  if (isStart(pkt)) {
98  message = 0x00;
99  currBit = 7;
100  /* Most i2c devices expect something special (e.g., command,
101  * register address) in the first byte they receive so they
102  * must be notified somehow that this is a new transmission.
103  */
104  for (auto& d : devices) {
105  d.second->i2cStart();
106  }
107  return pioDelay;
108  }
109 
110  // Check if the bus master is ending a transmission.
111  if (isEnd(pkt)) {
112  state = IDLE;
113  return pioDelay;
114  }
115 
116  // Only change state when the clock is transitioning from low to high.
117  // This may not perfectly mimic physical i2c devices but the important
118  // part is to only do the following once per clock cycle.
119  if (isClockSet(pkt)) {
120  switch (state) {
121  case RECEIVING_ADDR:
122  if (currBit >= 0) {
123  message |= sda << currBit;
124  currBit--;
125  } else {
126  i2cAddr = message >> 1;
127  assert(devices.find(i2cAddr) != devices.end());
128  if (message & 0x01) {
130  message = devices[i2cAddr]->read();
131  } else {
133  message = 0x00;
134  }
135  currBit = 7;
136  sda = 0; /* Ack */
137  }
138  break;
139  case RECEIVING_DATA:
140  if (currBit >= 0) {
141  message |= sda << currBit;
142  currBit--;
143  } else {
144  devices[i2cAddr]->write(message);
145  message = 0x00;
146  currBit = 7;
147  sda = 0; /* Ack */
148  }
149  break;
150  case SENDING_DATA:
151  if (currBit >= 0) {
152  sda = (message >> currBit) & 0x01;
153  currBit--;
154  } else {
155  if (!sda) /* Check for ack from the bus master. */
156  message = devices[i2cAddr]->read();
157  currBit = 7;
158  }
159  break;
160  case IDLE:
161  default:
162  panic("Invalid state on posedge of clock in I2CBus::write.\n");
163  break;
164  }
165  }
166 
167  return pioDelay;
168 }
169 
170 void
172 {
173  uint8_t msg = pkt->getRaw<uint8_t>();
174  Addr daddr = pkt->getAddr() - pioAddr;
175 
176  switch (daddr) {
177  case SB_CONTROLS:
178  scl = (msg & 1) ? 1 : scl;
179  sda = (msg & 2) ? 1 : sda;
180  break;
181  case SB_CONTROLC:
182  scl = (msg & 1) ? 0 : scl;
183  sda = (msg & 2) ? 0 : sda;
184  break;
185  default:
186  break;
187  }
188 }
189 
190 bool
192 {
193  uint8_t msg = pkt->getRaw<uint8_t>();
194  Addr daddr = pkt->getAddr() - pioAddr;
195  return daddr == SB_CONTROLS && (msg & 1);
196 }
197 
198 bool
200 {
201  uint8_t msg = pkt->getRaw<uint8_t>();
202  Addr daddr = pkt->getAddr() - pioAddr;
203  return scl && (msg & 2) && daddr == SB_CONTROLC;
204 }
205 
206 bool
208 {
209  uint8_t msg = pkt->getRaw<uint8_t>();
210  Addr daddr = pkt->getAddr() - pioAddr;
211  return scl && (msg & 2) && daddr == SB_CONTROLS;
212 }
213 void
215 {
216  DPRINTF(Checkpoint, "Serializing I2C bus.\n");
223 }
224 
225 void
227 {
228  DPRINTF(Checkpoint, "Unserializing I2C bus.\n");
235 }
236 
237 I2CBus*
238 I2CBusParams::create()
239 {
240  return new I2CBus(this);
241 }
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
#define DPRINTF(x,...)
Definition: trace.hh:225
bool isStart(PacketPtr pkt) const
i2c start signal check
Definition: bus.cc:199
std::map< uint8_t, I2CDevice * > devices
All the slave i2c devices that are connected to this bus.
Definition: bus.hh:107
enum I2CState state
State used by I2CBus::write to determine what stage of an i2c transmission it is currently in...
Definition: bus.hh:84
uint8_t i2cAddr
Key used to access a device in the slave devices map.
Definition: bus.hh:97
void setRaw(T v)
Set the value in the data pointer to v without byte swapping.
Definition: cprintf.cc:40
static const int SB_CONTROLS
Read [and Set] serial control bits: Bit [0] is SCL Bit [1] is SDA.
Definition: bus.hh:71
STL vector class.
Definition: stl.hh:37
uint8_t message
8-bit buffer used to send and receive messages bit by bit.
Definition: bus.hh:100
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:770
All i2c devices should derive from this class.
#define SERIALIZE_ENUM(scalar)
Definition: serialize.hh:786
void makeAtomicResponse()
Definition: packet.hh:943
uint64_t Tick
Tick count type.
Definition: types.hh:61
I2CBus(const I2CBusParams *p)
4KB - see e.g.
Definition: bus.cc:53
void updateSignals(PacketPtr pkt)
Update data (sda) and clock (scl) to match any transitions specified by pkt.
Definition: bus.cc:171
Bitfield< 9 > d
Tick read(PacketPtr pkt) override
Reads will always be to SB_CONTROLS.
Definition: bus.cc:69
Tick write(PacketPtr pkt) override
The default i2c bus driver used by the realview pbx board writes to this device one bit at a time...
Definition: bus.cc:88
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: bus.cc:226
Addr getAddr() const
Definition: packet.hh:720
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
int currBit
Order of the bit of the current message that is being sent or received (0 - 7).
Definition: bus.hh:90
Bitfield< 15, 8 > vector
Definition: intmessage.hh:44
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:763
Definition: bus.hh:53
std::ostream CheckpointOut
Definition: serialize.hh:63
static const int SB_CONTROLC
Clear control bits.
Definition: bus.hh:73
T getRaw() const
Get the data in the packet without byte swapping.
bool isEnd(PacketPtr pkt) const
i2c end signal check
Definition: bus.cc:207
#define UNSERIALIZE_ENUM(scalar)
Definition: serialize.hh:793
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: bus.cc:214
Tick pioDelay
Delay that the device experinces on an access.
Definition: io_device.hh:160
uint8_t scl
I2C clock wire (0, 1).
Definition: bus.hh:76
Bitfield< 0 > p
Addr pioAddr
Address that the device listens to.
Definition: io_device.hh:154
uint8_t sda
I2C data wire (0, 1)
Definition: bus.hh:78
Implementiation of an i2c bus.
bool isClockSet(PacketPtr pkt) const
Clock set check.
Definition: bus.cc:191

Generated on Thu May 28 2020 16:21:32 for gem5 by doxygen 1.8.13