blob: 991507bfe256b3268fcfcc52e13f2b589bba3649 [file] [log] [blame]
Patrick Venture030b1a82019-01-18 08:33:02 -08001#include "io.hpp"
2
3#include "internal/sys.hpp"
4
5#include <fcntl.h>
6
7#include <cstdint>
8#include <cstring>
9#include <string>
10
11namespace host_tool
12{
13
14const std::string DevMemDevice::devMemPath = "/dev/mem";
15
Patrick Ventureac4ff972019-05-03 17:35:00 -070016bool DevMemDevice::read(const std::size_t offset, const std::size_t length,
17 void* const destination)
18{
Patrick Venture18bbe3c2019-05-14 11:40:57 -070019 devMemFd = sys->open(devMemPath.c_str(), O_RDONLY);
20 if (devMemFd < 0)
Patrick Ventureac4ff972019-05-03 17:35:00 -070021 {
Patrick Venture18bbe3c2019-05-14 11:40:57 -070022 return false;
Patrick Ventureac4ff972019-05-03 17:35:00 -070023 }
24
Patrick Venture206097b2019-05-07 13:37:01 -070025 /* Map based on aligned addresses - behind the scenes. */
26 const std::size_t alignedDiff = offset % sys->getpagesize();
27 const std::size_t alignedOffset = offset - alignedDiff;
28 const std::size_t alignedSize = length + alignedDiff;
29
Patrick Ventureac4ff972019-05-03 17:35:00 -070030 // addr, length, prot, flags, fd, offset
Patrick Venture206097b2019-05-07 13:37:01 -070031 devMemMapped = sys->mmap(0, alignedSize, PROT_READ, MAP_SHARED, devMemFd,
32 alignedOffset);
Patrick Ventureac4ff972019-05-03 17:35:00 -070033 if (devMemMapped == MAP_FAILED)
34 {
Patrick Venture213f2db2019-05-15 10:47:30 -070035 std::fprintf(stderr, "Failed to mmap at offset: 0x%zx, length: %zu\n",
Patrick Venture18bbe3c2019-05-14 11:40:57 -070036 offset, length);
37 sys->close(devMemFd);
38 return false;
Patrick Ventureac4ff972019-05-03 17:35:00 -070039 }
40
Patrick Venture206097b2019-05-07 13:37:01 -070041 void* alignedSource =
42 static_cast<std::uint8_t*>(devMemMapped) + alignedDiff;
43
Patrick Ventureac4ff972019-05-03 17:35:00 -070044 /* Copy the bytes. */
Patrick Venture206097b2019-05-07 13:37:01 -070045 std::memcpy(destination, alignedSource, length);
Patrick Ventureac4ff972019-05-03 17:35:00 -070046
47 /* Close the map between reads for now. */
48 sys->munmap(devMemMapped, length);
Patrick Venture18bbe3c2019-05-14 11:40:57 -070049 sys->close(devMemFd);
Patrick Ventureac4ff972019-05-03 17:35:00 -070050
51 return true;
52}
53
Patrick Venture030b1a82019-01-18 08:33:02 -080054bool DevMemDevice::write(const std::size_t offset, const std::size_t length,
55 const void* const source)
56{
Patrick Venture18bbe3c2019-05-14 11:40:57 -070057 devMemFd = sys->open(devMemPath.c_str(), O_RDWR);
58 if (devMemFd < 0)
Patrick Venture030b1a82019-01-18 08:33:02 -080059 {
Patrick Venture18bbe3c2019-05-14 11:40:57 -070060 std::fprintf(stderr, "Failed to open /dev/mem for writing\n");
61 return false;
Patrick Venture030b1a82019-01-18 08:33:02 -080062 }
63
Patrick Venture206097b2019-05-07 13:37:01 -070064 /* Map based on aligned addresses - behind the scenes. */
65 const std::size_t alignedDiff = offset % sys->getpagesize();
66 const std::size_t alignedOffset = offset - alignedDiff;
67 const std::size_t alignedSize = length + alignedDiff;
68
Patrick Venture030b1a82019-01-18 08:33:02 -080069 // addr, length, prot, flags, fd, offset
Patrick Venture206097b2019-05-07 13:37:01 -070070 devMemMapped = sys->mmap(0, alignedSize, PROT_WRITE, MAP_SHARED, devMemFd,
71 alignedOffset);
72
Patrick Venture030b1a82019-01-18 08:33:02 -080073 if (devMemMapped == MAP_FAILED)
74 {
Patrick Venture213f2db2019-05-15 10:47:30 -070075 std::fprintf(stderr, "Failed to mmap at offset: 0x%zx, length: %zu\n",
Patrick Venture18bbe3c2019-05-14 11:40:57 -070076 offset, length);
77 sys->close(devMemFd);
78 return false;
Patrick Venture030b1a82019-01-18 08:33:02 -080079 }
80
Patrick Venture206097b2019-05-07 13:37:01 -070081 void* alignedDestination =
82 static_cast<std::uint8_t*>(devMemMapped) + alignedDiff;
83
Patrick Venture030b1a82019-01-18 08:33:02 -080084 /* Copy the bytes. */
Patrick Venture206097b2019-05-07 13:37:01 -070085 std::memcpy(alignedDestination, source, length);
Patrick Venture030b1a82019-01-18 08:33:02 -080086
87 /* Close the map between writes for now. */
88 sys->munmap(devMemMapped, length);
Patrick Venture18bbe3c2019-05-14 11:40:57 -070089 sys->close(devMemFd);
Patrick Venture030b1a82019-01-18 08:33:02 -080090
91 return true;
92}
93
Brandon Kim286cc6a2019-11-05 16:39:47 -080094PpcMemDevice::~PpcMemDevice()
95{
96 // Attempt to close in case reads or writes didn't close themselves
97 if (ppcMemFd >= 0)
98 {
99 sys->close(ppcMemFd);
100 }
101}
102
103bool PpcMemDevice::read(const std::size_t offset, const std::size_t length,
104 void* const destination)
105{
106 ppcMemFd = sys->open(ppcMemPath.c_str(), O_RDWR);
107 if (ppcMemFd < 0)
108 {
109 std::fprintf(stderr, "Failed to open PPC LPC access path: %s",
110 ppcMemPath.c_str());
111 return false;
112 }
113
114 int ret = sys->pread(ppcMemFd, destination, length, offset);
115 if (ret < 0)
116 {
117 std::fprintf(stderr, "IO read failed at offset: 0x%zx, length: %zu\n",
118 offset, length);
119 sys->close(ppcMemFd);
120 return false;
121 }
122
123 sys->close(ppcMemFd);
124 return true;
125}
126
127bool PpcMemDevice::write(const std::size_t offset, const std::size_t length,
128 const void* const source)
129{
130 ppcMemFd = sys->open(ppcMemPath.c_str(), O_RDWR);
131 if (ppcMemFd < 0)
132 {
133 std::fprintf(stderr, "Failed to open PPC LPC access path: %s",
134 ppcMemPath.c_str());
135 return false;
136 }
137
138 ssize_t ret = sys->pwrite(ppcMemFd, source, length, offset);
139 if (ret < 0)
140 {
141 std::fprintf(stderr, "IO write failed at offset: 0x%zx, length: %zu\n",
142 offset, length);
143 sys->close(ppcMemFd);
144 return false;
145 }
146
147 sys->close(ppcMemFd);
148 return true;
149}
150
Patrick Venture030b1a82019-01-18 08:33:02 -0800151} // namespace host_tool