+++ /dev/null
-# Copyright (c) 2012 ARM Limited
-# All rights reserved.
-#
-# The license below extends only to copyright in the software and shall
-# not be construed as granting a license to any other intellectual
-# property including but not limited to intellectual property relating
-# to a hardware implementation of the functionality of the software
-# licensed hereunder. You may use the software subject to the license
-# terms below provided that you ensure that this notice is replicated
-# unmodified and in its entirety in all distributions of the software,
-# modified or unmodified, in source code or in binary form.
-#
-# Redistribution and use in source and binary forms, with or without
-# modification, are permitted provided that the following conditions are
-# met: redistributions of source code must retain the above copyright
-# notice, this list of conditions and the following disclaimer;
-# 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;
-# neither the name of the copyright holders nor the names of its
-# contributors may be used to endorse or promote products derived from
-# this software without specific prior written permission.
-#
-# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
-# "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 COPYRIGHT
-# OWNER OR CONTRIBUTORS 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.
-#
-# Authors: Peter Enns
-
-from m5.SimObject import SimObject
-from m5.params import *
-from Device import BasicPioDevice
-
-class I2CDevice(SimObject):
- type = 'I2CDevice'
- cxx_header = "dev/i2cdev.hh"
- abstract = True
- i2c_addr = Param.UInt8("Address of device on i2c bus")
-
-class I2CBus(BasicPioDevice):
- type = 'I2CBus'
- cxx_header = "dev/i2cbus.hh"
- devices = VectorParam.I2CDevice([], "Devices")
SimObject('BadDevice.py')
SimObject('DiskImage.py')
SimObject('Ethernet.py')
-SimObject('I2C.py')
SimObject('Ide.py')
SimObject('Platform.py')
SimObject('SimpleDisk.py')
Source('tcp_iface.cc')
Source('etherpkt.cc')
Source('ethertap.cc')
-Source('i2cbus.cc')
Source('i8254xGBe.cc')
Source('ide_ctrl.cc')
Source('ide_disk.cc')
--- /dev/null
+# Copyright (c) 2012 ARM Limited
+# All rights reserved.
+#
+# The license below extends only to copyright in the software and shall
+# not be construed as granting a license to any other intellectual
+# property including but not limited to intellectual property relating
+# to a hardware implementation of the functionality of the software
+# licensed hereunder. You may use the software subject to the license
+# terms below provided that you ensure that this notice is replicated
+# unmodified and in its entirety in all distributions of the software,
+# modified or unmodified, in source code or in binary form.
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are
+# met: redistributions of source code must retain the above copyright
+# notice, this list of conditions and the following disclaimer;
+# 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;
+# neither the name of the copyright holders nor the names of its
+# contributors may be used to endorse or promote products derived from
+# this software without specific prior written permission.
+#
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+# "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 COPYRIGHT
+# OWNER OR CONTRIBUTORS 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.
+#
+# Authors: Peter Enns
+
+from m5.SimObject import SimObject
+from m5.params import *
+from Device import BasicPioDevice
+
+class I2CDevice(SimObject):
+ type = 'I2CDevice'
+ cxx_header = "dev/i2c/device.hh"
+ abstract = True
+ i2c_addr = Param.UInt8("Address of device on i2c bus")
+
+class I2CBus(BasicPioDevice):
+ type = 'I2CBus'
+ cxx_header = "dev/i2c/bus.hh"
+ devices = VectorParam.I2CDevice([], "Devices")
--- /dev/null
+/*
+ * Copyright (c) 2012 ARM Limited
+ * All rights reserved
+ *
+ * The license below extends only to copyright in the software and shall
+ * not be construed as granting a license to any other intellectual
+ * property including but not limited to intellectual property relating
+ * to a hardware implementation of the functionality of the software
+ * licensed hereunder. You may use the software subject to the license
+ * terms below provided that you ensure that this notice is replicated
+ * unmodified and in its entirety in all distributions of the software,
+ * modified or unmodified, in source code or in binary form.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met: redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer;
+ * 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;
+ * neither the name of the copyright holders nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "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 COPYRIGHT
+ * OWNER OR CONTRIBUTORS 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.
+ *
+ * Authors: Peter Enns
+ */
+
+#include "dev/i2c/bus.hh"
+
+#include "debug/Checkpoint.hh"
+#include "dev/i2c/device.hh"
+#include "mem/packet_access.hh"
+
+// clang complains about std::set being overloaded with Packet::set if
+// we open up the entire namespace std
+using std::vector;
+using std::map;
+
+/**
+ * 4KB - see e.g.
+ * http://infocenter.arm.com/help/topic/com.arm.doc.dui0440b/Bbajihec.html
+ */
+I2CBus::I2CBus(const I2CBusParams *p)
+ : BasicPioDevice(p, 0xfff), scl(1), sda(1), state(IDLE), currBit(7),
+ i2cAddr(0x00), message(0x00)
+{
+ vector<I2CDevice*> devs = p->devices;
+
+ for (auto d : p->devices) {
+ devices[d->i2cAddr()] = d;
+ }
+}
+
+/**
+ * Reads will always be to SB_CONTROLS. The kernel wants to know the state
+ * of sda and scl.
+ */
+Tick
+I2CBus::read(PacketPtr pkt)
+{
+ assert(pkt->getAddr() == pioAddr + SB_CONTROLS);
+
+ pkt->set<uint8_t>((sda << 1) | scl);
+ pkt->makeAtomicResponse();
+ return pioDelay;
+}
+
+/**
+ * The default i2c bus driver used by the realview pbx board writes to
+ * this device one bit at a time. To facilitate making new i2c devices,
+ * i2cBus::write takes care of the low-level details of the i2c protocol.
+ * See the I2C Specification [1] for a detailed description of the
+ * protocol.
+ *
+ * [1] - http://www.nxp.com/documents/user_manual/UM10204.pdf
+ */
+Tick
+I2CBus::write(PacketPtr pkt)
+{
+ assert(pkt->getAddr() == pioAddr + SB_CONTROLS ||
+ pkt->getAddr() == pioAddr + SB_CONTROLC);
+
+ updateSignals(pkt);
+
+ // Check if the bus master is starting a new transmission.
+ if (isStart(pkt)) {
+ state = RECEIVING_ADDR;
+ message = 0x00;
+ currBit = 7;
+ /* Most i2c devices expect something special (e.g., command,
+ * register address) in the first byte they receive so they
+ * must be notified somehow that this is a new transmission.
+ */
+ for (auto& d : devices) {
+ d.second->i2cStart();
+ }
+ return pioDelay;
+ }
+
+ // Check if the bus master is ending a transmission.
+ if (isEnd(pkt)) {
+ state = IDLE;
+ return pioDelay;
+ }
+
+ // Only change state when the clock is transitioning from low to high.
+ // This may not perfectly mimic physical i2c devices but the important
+ // part is to only do the following once per clock cycle.
+ if (isClockSet(pkt)) {
+ switch (state) {
+ case RECEIVING_ADDR:
+ if (currBit >= 0) {
+ message |= sda << currBit;
+ currBit--;
+ } else {
+ i2cAddr = message >> 1;
+ assert(devices.find(i2cAddr) != devices.end());
+ if (message & 0x01) {
+ state = SENDING_DATA;
+ message = devices[i2cAddr]->read();
+ } else {
+ state = RECEIVING_DATA;
+ message = 0x00;
+ }
+ currBit = 7;
+ sda = 0; /* Ack */
+ }
+ break;
+ case RECEIVING_DATA:
+ if (currBit >= 0) {
+ message |= sda << currBit;
+ currBit--;
+ } else {
+ devices[i2cAddr]->write(message);
+ message = 0x00;
+ currBit = 7;
+ sda = 0; /* Ack */
+ }
+ break;
+ case SENDING_DATA:
+ if (currBit >= 0) {
+ sda = (message >> currBit) & 0x01;
+ currBit--;
+ } else {
+ if (!sda) /* Check for ack from the bus master. */
+ message = devices[i2cAddr]->read();
+ currBit = 7;
+ }
+ break;
+ case IDLE:
+ default:
+ panic("Invalid state on posedge of clock in I2CBus::write.\n");
+ break;
+ }
+ }
+
+ return pioDelay;
+}
+
+void
+I2CBus::updateSignals(PacketPtr pkt)
+{
+ uint8_t msg = pkt->get<uint8_t>();
+ Addr daddr = pkt->getAddr() - pioAddr;
+
+ switch (daddr) {
+ case SB_CONTROLS:
+ scl = (msg & 1) ? 1 : scl;
+ sda = (msg & 2) ? 1 : sda;
+ break;
+ case SB_CONTROLC:
+ scl = (msg & 1) ? 0 : scl;
+ sda = (msg & 2) ? 0 : sda;
+ break;
+ default:
+ break;
+ }
+}
+
+bool
+I2CBus::isClockSet(PacketPtr pkt) const
+{
+ uint8_t msg = pkt->get<uint8_t>();
+ Addr daddr = pkt->getAddr() - pioAddr;
+ return daddr == SB_CONTROLS && (msg & 1);
+}
+
+bool
+I2CBus::isStart(PacketPtr pkt) const
+{
+ uint8_t msg = pkt->get<uint8_t>();
+ Addr daddr = pkt->getAddr() - pioAddr;
+ return scl && (msg & 2) && daddr == SB_CONTROLC;
+}
+
+bool
+I2CBus::isEnd(PacketPtr pkt) const
+{
+ uint8_t msg = pkt->get<uint8_t>();
+ Addr daddr = pkt->getAddr() - pioAddr;
+ return scl && (msg & 2) && daddr == SB_CONTROLS;
+}
+void
+I2CBus::serialize(CheckpointOut &cp) const
+{
+ DPRINTF(Checkpoint, "Serializing I2C bus.\n");
+ SERIALIZE_SCALAR(scl);
+ SERIALIZE_SCALAR(sda);
+ SERIALIZE_ENUM(state);
+ SERIALIZE_SCALAR(currBit);
+ SERIALIZE_SCALAR(i2cAddr);
+ SERIALIZE_SCALAR(message);
+}
+
+void
+I2CBus::unserialize(CheckpointIn &cp)
+{
+ DPRINTF(Checkpoint, "Unserializing I2C bus.\n");
+ UNSERIALIZE_SCALAR(scl);
+ UNSERIALIZE_SCALAR(sda);
+ UNSERIALIZE_ENUM(state);
+ UNSERIALIZE_SCALAR(currBit);
+ UNSERIALIZE_SCALAR(i2cAddr);
+ UNSERIALIZE_SCALAR(message);
+}
+
+I2CBus*
+I2CBusParams::create()
+{
+ return new I2CBus(this);
+}
--- /dev/null
+/*
+ * Copyright (c) 2012 ARM Limited
+ * All rights reserved
+ *
+ * The license below extends only to copyright in the software and shall
+ * not be construed as granting a license to any other intellectual
+ * property including but not limited to intellectual property relating
+ * to a hardware implementation of the functionality of the software
+ * licensed hereunder. You may use the software subject to the license
+ * terms below provided that you ensure that this notice is replicated
+ * unmodified and in its entirety in all distributions of the software,
+ * modified or unmodified, in source code or in binary form.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met: redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer;
+ * 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;
+ * neither the name of the copyright holders nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "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 COPYRIGHT
+ * OWNER OR CONTRIBUTORS 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.
+ *
+ * Authors: Peter Enns
+ */
+
+
+/** @file
+ * Implementiation of an i2c bus
+ */
+
+#ifndef __DEV_I2C_BUS_HH__
+#define __DEV_I2C_BUS_HH__
+
+#include <map>
+
+#include "dev/io_device.hh"
+#include "params/I2CBus.hh"
+
+class I2CDevice;
+
+class I2CBus : public BasicPioDevice
+{
+ protected:
+
+ enum I2CState {
+ IDLE,
+ RECEIVING_ADDR,
+ RECEIVING_DATA,
+ SENDING_DATA,
+ };
+
+ /**
+ * Read [and Set] serial control bits:
+ * Bit [0] is SCL
+ * Bit [1] is SDA
+ *
+ * http://infocenter.arm.com/help/topic/com.arm.doc.dui0440b/Bbajdjeg.html
+ */
+ static const int SB_CONTROLS = 0x0;
+ /** Clear control bits. Analogous to SB_CONTROLS */
+ static const int SB_CONTROLC = 0x4;
+
+ /** I2C clock wire (0, 1). */
+ uint8_t scl;
+ /** I2C data wire (0, 1) */
+ uint8_t sda;
+
+ /**
+ * State used by I2CBus::write to determine what stage of an i2c
+ * transmission it is currently in.
+ */
+ enum I2CState state;
+
+ /**
+ * Order of the bit of the current message that is being sent or
+ * received (0 - 7).
+ */
+ int currBit;
+
+ /**
+ * Key used to access a device in the slave devices map. This
+ * is the same address that is specified in kernel board
+ * initialization code (e.g., arch/arm/mach-realview/core.c).
+ */
+ uint8_t i2cAddr;
+
+ /** 8-bit buffer used to send and receive messages bit by bit. */
+ uint8_t message;
+
+ /**
+ * All the slave i2c devices that are connected to this
+ * bus. Each device has an address that points to the actual
+ * device.
+ */
+ std::map<uint8_t, I2CDevice*> devices;
+
+ /**
+ * Update data (sda) and clock (scl) to match any transitions
+ * specified by pkt.
+ *
+ * @param pkt memory request packet
+ */
+ void updateSignals(PacketPtr pkt);
+
+ /**
+ * Clock set check
+ *
+ * @param pkt memory request packet
+ * @return true if pkt indicates that scl transition from 0 to 1
+ */
+ bool isClockSet(PacketPtr pkt) const;
+
+ /**
+ * i2c start signal check
+ *
+ * @param pkt memory request packet
+ * @return true if pkt indicates a new transmission
+ */
+ bool isStart(PacketPtr pkt) const;
+
+ /**
+ * i2c end signal check
+ *
+ * @param pkt memory request packet
+ * @return true if pkt indicates stopping the current transmission
+ */
+ bool isEnd(PacketPtr pkt) const;
+
+ public:
+
+ I2CBus(const I2CBusParams* p);
+
+ Tick read(PacketPtr pkt) override;
+ Tick write(PacketPtr pkt) override;
+
+ void serialize(CheckpointOut &cp) const override;
+ void unserialize(CheckpointIn &cp) override;
+};
+
+#endif // __DEV_I2C_BUS_HH__
--- /dev/null
+/*
+ * Copyright (c) 2012 ARM Limited
+ * All rights reserved
+ *
+ * The license below extends only to copyright in the software and shall
+ * not be construed as granting a license to any other intellectual
+ * property including but not limited to intellectual property relating
+ * to a hardware implementation of the functionality of the software
+ * licensed hereunder. You may use the software subject to the license
+ * terms below provided that you ensure that this notice is replicated
+ * unmodified and in its entirety in all distributions of the software,
+ * modified or unmodified, in source code or in binary form.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met: redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer;
+ * 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;
+ * neither the name of the copyright holders nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "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 COPYRIGHT
+ * OWNER OR CONTRIBUTORS 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.*
+ *
+ * Authors: Peter Enns
+ */
+
+
+/** @file
+ * All i2c devices should derive from this class.
+ */
+
+#ifndef __DEV_I2C_DEVICE_HH__
+#define __DEV_I2C_DEVICE_HH__
+
+#include "base/types.hh"
+#include "params/I2CDevice.hh"
+#include "sim/sim_object.hh"
+
+class I2CDevice : public SimObject
+{
+
+ protected:
+
+ uint8_t _addr;
+
+ public:
+
+ I2CDevice(const I2CDeviceParams* p)
+ : SimObject(p), _addr(p->i2c_addr)
+ { }
+
+ virtual ~I2CDevice() { }
+
+ /**
+ * Return the next message that the device expects to send. This
+ * will likely have side effects (e.g., incrementing a register
+ * pointer).
+ *
+ * @return 8-bit message the device has been set up to send
+ */
+ virtual uint8_t read() = 0;
+
+ /**
+ * Perform any actions triggered by an i2c write (save msg in a
+ * register, perform an interrupt, update a register pointer or
+ * command register, etc...)
+ *
+ * @param msg 8-bit message from master
+ */
+ virtual void write(uint8_t msg) = 0;
+
+ /**
+ * Perform any initialization necessary for the device when it
+ * received a start signal from the bus master (devices frequently
+ * expect the first write to be a register address)
+ */
+ virtual void i2cStart() = 0;
+
+ uint8_t i2cAddr() const { return _addr; }
+
+};
+
+#endif // __DEV_I2C_DEVICE__
+++ /dev/null
-/*
- * Copyright (c) 2012 ARM Limited
- * All rights reserved
- *
- * The license below extends only to copyright in the software and shall
- * not be construed as granting a license to any other intellectual
- * property including but not limited to intellectual property relating
- * to a hardware implementation of the functionality of the software
- * licensed hereunder. You may use the software subject to the license
- * terms below provided that you ensure that this notice is replicated
- * unmodified and in its entirety in all distributions of the software,
- * modified or unmodified, in source code or in binary form.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are
- * met: redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer;
- * 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;
- * neither the name of the copyright holders nor the names of its
- * contributors may be used to endorse or promote products derived from
- * this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "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 COPYRIGHT
- * OWNER OR CONTRIBUTORS 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.
- *
- * Authors: Peter Enns
- */
-
-#include "dev/i2cbus.hh"
-
-#include "debug/Checkpoint.hh"
-#include "mem/packet_access.hh"
-
-// clang complains about std::set being overloaded with Packet::set if
-// we open up the entire namespace std
-using std::vector;
-using std::map;
-
-/**
- * 4KB - see e.g.
- * http://infocenter.arm.com/help/topic/com.arm.doc.dui0440b/Bbajihec.html
- */
-I2CBus::I2CBus(const I2CBusParams *p)
- : BasicPioDevice(p, 0xfff), scl(1), sda(1), state(IDLE), currBit(7),
- i2cAddr(0x00), message(0x00)
-{
- vector<I2CDevice*> devs = p->devices;
-
- for (auto d : p->devices) {
- devices[d->i2cAddr()] = d;
- }
-}
-
-/**
- * Reads will always be to SB_CONTROLS. The kernel wants to know the state
- * of sda and scl.
- */
-Tick
-I2CBus::read(PacketPtr pkt)
-{
- assert(pkt->getAddr() == pioAddr + SB_CONTROLS);
-
- pkt->set<uint8_t>((sda << 1) | scl);
- pkt->makeAtomicResponse();
- return pioDelay;
-}
-
-/**
- * The default i2c bus driver used by the realview pbx board writes to
- * this device one bit at a time. To facilitate making new i2c devices,
- * i2cBus::write takes care of the low-level details of the i2c protocol.
- * See the I2C Specification [1] for a detailed description of the
- * protocol.
- *
- * [1] - http://www.nxp.com/documents/user_manual/UM10204.pdf
- */
-Tick
-I2CBus::write(PacketPtr pkt)
-{
- assert(pkt->getAddr() == pioAddr + SB_CONTROLS ||
- pkt->getAddr() == pioAddr + SB_CONTROLC);
-
- updateSignals(pkt);
-
- // Check if the bus master is starting a new transmission.
- if (isStart(pkt)) {
- state = RECEIVING_ADDR;
- message = 0x00;
- currBit = 7;
- /* Most i2c devices expect something special (e.g., command,
- * register address) in the first byte they receive so they
- * must be notified somehow that this is a new transmission.
- */
- for (auto& d : devices) {
- d.second->i2cStart();
- }
- return pioDelay;
- }
-
- // Check if the bus master is ending a transmission.
- if (isEnd(pkt)) {
- state = IDLE;
- return pioDelay;
- }
-
- // Only change state when the clock is transitioning from low to high.
- // This may not perfectly mimic physical i2c devices but the important
- // part is to only do the following once per clock cycle.
- if (isClockSet(pkt)) {
- switch (state) {
- case RECEIVING_ADDR:
- if (currBit >= 0) {
- message |= sda << currBit;
- currBit--;
- } else {
- i2cAddr = message >> 1;
- assert(devices.find(i2cAddr) != devices.end());
- if (message & 0x01) {
- state = SENDING_DATA;
- message = devices[i2cAddr]->read();
- } else {
- state = RECEIVING_DATA;
- message = 0x00;
- }
- currBit = 7;
- sda = 0; /* Ack */
- }
- break;
- case RECEIVING_DATA:
- if (currBit >= 0) {
- message |= sda << currBit;
- currBit--;
- } else {
- devices[i2cAddr]->write(message);
- message = 0x00;
- currBit = 7;
- sda = 0; /* Ack */
- }
- break;
- case SENDING_DATA:
- if (currBit >= 0) {
- sda = (message >> currBit) & 0x01;
- currBit--;
- } else {
- if (!sda) /* Check for ack from the bus master. */
- message = devices[i2cAddr]->read();
- currBit = 7;
- }
- break;
- case IDLE:
- default:
- panic("Invalid state on posedge of clock in I2CBus::write.\n");
- break;
- }
- }
-
- return pioDelay;
-}
-
-void
-I2CBus::updateSignals(PacketPtr pkt)
-{
- uint8_t msg = pkt->get<uint8_t>();
- Addr daddr = pkt->getAddr() - pioAddr;
-
- switch (daddr) {
- case SB_CONTROLS:
- scl = (msg & 1) ? 1 : scl;
- sda = (msg & 2) ? 1 : sda;
- break;
- case SB_CONTROLC:
- scl = (msg & 1) ? 0 : scl;
- sda = (msg & 2) ? 0 : sda;
- break;
- default:
- break;
- }
-}
-
-bool
-I2CBus::isClockSet(PacketPtr pkt) const
-{
- uint8_t msg = pkt->get<uint8_t>();
- Addr daddr = pkt->getAddr() - pioAddr;
- return daddr == SB_CONTROLS && (msg & 1);
-}
-
-bool
-I2CBus::isStart(PacketPtr pkt) const
-{
- uint8_t msg = pkt->get<uint8_t>();
- Addr daddr = pkt->getAddr() - pioAddr;
- return scl && (msg & 2) && daddr == SB_CONTROLC;
-}
-
-bool
-I2CBus::isEnd(PacketPtr pkt) const
-{
- uint8_t msg = pkt->get<uint8_t>();
- Addr daddr = pkt->getAddr() - pioAddr;
- return scl && (msg & 2) && daddr == SB_CONTROLS;
-}
-void
-I2CBus::serialize(CheckpointOut &cp) const
-{
- DPRINTF(Checkpoint, "Serializing I2C bus.\n");
- SERIALIZE_SCALAR(scl);
- SERIALIZE_SCALAR(sda);
- SERIALIZE_ENUM(state);
- SERIALIZE_SCALAR(currBit);
- SERIALIZE_SCALAR(i2cAddr);
- SERIALIZE_SCALAR(message);
-}
-
-void
-I2CBus::unserialize(CheckpointIn &cp)
-{
- DPRINTF(Checkpoint, "Unserializing I2C bus.\n");
- UNSERIALIZE_SCALAR(scl);
- UNSERIALIZE_SCALAR(sda);
- UNSERIALIZE_ENUM(state);
- UNSERIALIZE_SCALAR(currBit);
- UNSERIALIZE_SCALAR(i2cAddr);
- UNSERIALIZE_SCALAR(message);
-}
-
-I2CBus*
-I2CBusParams::create()
-{
- return new I2CBus(this);
-}
+++ /dev/null
-/*
- * Copyright (c) 2012 ARM Limited
- * All rights reserved
- *
- * The license below extends only to copyright in the software and shall
- * not be construed as granting a license to any other intellectual
- * property including but not limited to intellectual property relating
- * to a hardware implementation of the functionality of the software
- * licensed hereunder. You may use the software subject to the license
- * terms below provided that you ensure that this notice is replicated
- * unmodified and in its entirety in all distributions of the software,
- * modified or unmodified, in source code or in binary form.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are
- * met: redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer;
- * 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;
- * neither the name of the copyright holders nor the names of its
- * contributors may be used to endorse or promote products derived from
- * this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "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 COPYRIGHT
- * OWNER OR CONTRIBUTORS 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.
- *
- * Authors: Peter Enns
- */
-
-
-/** @file
- * Implementiation of an i2c bus
- */
-
-#ifndef __DEV_I2CBUS__
-#define __DEV_I2CBUS__
-
-#include <map>
-
-#include "dev/i2cdev.hh"
-#include "dev/io_device.hh"
-#include "params/I2CBus.hh"
-
-class I2CBus : public BasicPioDevice
-{
- protected:
-
- enum I2CState {
- IDLE,
- RECEIVING_ADDR,
- RECEIVING_DATA,
- SENDING_DATA,
- };
-
- /**
- * Read [and Set] serial control bits:
- * Bit [0] is SCL
- * Bit [1] is SDA
- *
- * http://infocenter.arm.com/help/topic/com.arm.doc.dui0440b/Bbajdjeg.html
- */
- static const int SB_CONTROLS = 0x0;
- /** Clear control bits. Analogous to SB_CONTROLS */
- static const int SB_CONTROLC = 0x4;
-
- /** I2C clock wire (0, 1). */
- uint8_t scl;
- /** I2C data wire (0, 1) */
- uint8_t sda;
-
- /**
- * State used by I2CBus::write to determine what stage of an i2c
- * transmission it is currently in.
- */
- enum I2CState state;
-
- /**
- * Order of the bit of the current message that is being sent or
- * received (0 - 7).
- */
- int currBit;
-
- /**
- * Key used to access a device in the slave devices map. This
- * is the same address that is specified in kernel board
- * initialization code (e.g., arch/arm/mach-realview/core.c).
- */
- uint8_t i2cAddr;
-
- /** 8-bit buffer used to send and receive messages bit by bit. */
- uint8_t message;
-
- /**
- * All the slave i2c devices that are connected to this
- * bus. Each device has an address that points to the actual
- * device.
- */
- std::map<uint8_t, I2CDevice*> devices;
-
- /**
- * Update data (sda) and clock (scl) to match any transitions
- * specified by pkt.
- *
- * @param pkt memory request packet
- */
- void updateSignals(PacketPtr pkt);
-
- /**
- * Clock set check
- *
- * @param pkt memory request packet
- * @return true if pkt indicates that scl transition from 0 to 1
- */
- bool isClockSet(PacketPtr pkt) const;
-
- /**
- * i2c start signal check
- *
- * @param pkt memory request packet
- * @return true if pkt indicates a new transmission
- */
- bool isStart(PacketPtr pkt) const;
-
- /**
- * i2c end signal check
- *
- * @param pkt memory request packet
- * @return true if pkt indicates stopping the current transmission
- */
- bool isEnd(PacketPtr pkt) const;
-
- public:
-
- I2CBus(const I2CBusParams* p);
-
- Tick read(PacketPtr pkt) override;
- Tick write(PacketPtr pkt) override;
-
- void serialize(CheckpointOut &cp) const override;
- void unserialize(CheckpointIn &cp) override;
-};
-
-#endif //__DEV_I2CBUS
+++ /dev/null
-/*
- * Copyright (c) 2012 ARM Limited
- * All rights reserved
- *
- * The license below extends only to copyright in the software and shall
- * not be construed as granting a license to any other intellectual
- * property including but not limited to intellectual property relating
- * to a hardware implementation of the functionality of the software
- * licensed hereunder. You may use the software subject to the license
- * terms below provided that you ensure that this notice is replicated
- * unmodified and in its entirety in all distributions of the software,
- * modified or unmodified, in source code or in binary form.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are
- * met: redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer;
- * 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;
- * neither the name of the copyright holders nor the names of its
- * contributors may be used to endorse or promote products derived from
- * this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "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 COPYRIGHT
- * OWNER OR CONTRIBUTORS 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.*
- *
- * Authors: Peter Enns
- */
-
-
-/** @file
- * All i2c devices should derive from this class.
- */
-
-#ifndef __DEV_I2CDEV__
-#define __DEV_I2CDEV__
-
-#include "base/types.hh"
-#include "params/I2CDevice.hh"
-#include "sim/sim_object.hh"
-
-class I2CDevice : public SimObject
-{
-
- protected:
-
- uint8_t _addr;
-
- public:
-
- I2CDevice(const I2CDeviceParams* p)
- : SimObject(p), _addr(p->i2c_addr)
- { }
-
- virtual ~I2CDevice() { }
-
- /**
- * Return the next message that the device expects to send. This
- * will likely have side effects (e.g., incrementing a register
- * pointer).
- *
- * @return 8-bit message the device has been set up to send
- */
- virtual uint8_t read() = 0;
-
- /**
- * Perform any actions triggered by an i2c write (save msg in a
- * register, perform an interrupt, update a register pointer or
- * command register, etc...)
- *
- * @param msg 8-bit message from master
- */
- virtual void write(uint8_t msg) = 0;
-
- /**
- * Perform any initialization necessary for the device when it
- * received a start signal from the bus master (devices frequently
- * expect the first write to be a register address)
- */
- virtual void i2cStart() = 0;
-
- uint8_t i2cAddr() const { return _addr; }
-
-};
-
-#endif // __DEV_I2CDEV__