blob: 1b6b7ffa207a9936d20428b5b52252f639c77453 [file] [log] [blame]
Patrick Venture22e38752018-11-21 08:52:49 -08001/*
2 * Copyright 2018 Google Inc.
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Patrick Venture1cde5f92018-11-07 08:26:47 -080017#include "pci_handler.hpp"
18
Patrick Venture84778b82019-06-26 20:11:09 -070019#include "data.hpp"
20
Patrick Venture4289e712019-04-30 17:08:50 -070021#include <fcntl.h>
22#include <linux/aspeed-p2a-ctrl.h>
23
Patrick Venture1cde5f92018-11-07 08:26:47 -080024#include <cstdint>
Patrick Venturea8a99ae2018-11-09 11:14:58 -080025#include <cstring>
Patrick Venturebc40c612019-04-30 16:48:19 -070026#include <string>
Patrick Venture1cde5f92018-11-07 08:26:47 -080027#include <vector>
28
Patrick Venture1d5a31c2019-05-20 11:38:22 -070029namespace ipmi_flash
Patrick Venture1cde5f92018-11-07 08:26:47 -080030{
31
Patrick Venturebc40c612019-04-30 16:48:19 -070032const std::string PciDataHandler::p2aControlPath = "/dev/aspeed-p2a-ctrl";
33
Patrick Venture0d2a8132018-11-09 11:34:21 -080034bool PciDataHandler::open()
35{
Patrick Venture4289e712019-04-30 17:08:50 -070036 mappedFd = sys->open(p2aControlPath.c_str(), O_RDWR);
37 if (mappedFd == -1)
38 {
39 return false;
40 }
41
42 struct aspeed_p2a_ctrl_mapping map;
43 map.addr = regionAddress;
44 map.length = memoryRegionSize;
45 map.flags = ASPEED_P2A_CTRL_READWRITE;
46
47 if (sys->ioctl(mappedFd, ASPEED_P2A_CTRL_IOCTL_SET_WINDOW, &map))
48 {
49 sys->close(mappedFd);
50 mappedFd = -1;
51
52 return false;
53 }
54
55 if (sys->ioctl(mappedFd, ASPEED_P2A_CTRL_IOCTL_GET_MEMORY_CONFIG, &map))
56 {
57 sys->close(mappedFd);
58 mappedFd = -1;
59
60 return false;
61 }
62
63 /* The length of the region reserved is reported, and it's important
64 * because the offset + memory region could be beyond that reserved
65 * region.
Patrick Venture0fbabf22018-11-09 11:54:12 -080066 */
Patrick Venture4289e712019-04-30 17:08:50 -070067 std::uint64_t offset = regionAddress - map.addr;
68
69 mapped = reinterpret_cast<std::uint8_t*>(
70 mmap(0, memoryRegionSize, PROT_READ, MAP_SHARED, mappedFd, offset));
71 if (mapped == MAP_FAILED)
72 {
73 sys->close(mappedFd);
74 mappedFd = -1;
Patrick Ventureef4993e2019-06-24 09:56:37 -070075 mapped = nullptr;
Patrick Venture4289e712019-04-30 17:08:50 -070076
77 return false;
78 }
79
80 return true;
Patrick Venture0fbabf22018-11-09 11:54:12 -080081}
82
83bool PciDataHandler::close()
84{
85 /* TODO: Turn off the P2A bridge and region to disable host-side access.
Patrick Venture0d2a8132018-11-09 11:34:21 -080086 */
Patrick Venture4289e712019-04-30 17:08:50 -070087 if (mapped)
88 {
89 sys->munmap(mapped, memoryRegionSize);
90 mapped = nullptr;
91 }
92
93 if (mappedFd != -1)
94 {
95 sys->close(mappedFd);
96 mappedFd = -1;
97 }
98
99 return true;
Patrick Venture0d2a8132018-11-09 11:34:21 -0800100}
101
Patrick Venture1cde5f92018-11-07 08:26:47 -0800102std::vector<std::uint8_t> PciDataHandler::copyFrom(std::uint32_t length)
103{
Patrick Venture18bbe3c2019-05-14 11:40:57 -0700104 std::vector<std::uint8_t> results(length);
105 std::memcpy(results.data(), mapped, length);
106
107 return results;
Patrick Venture1cde5f92018-11-07 08:26:47 -0800108}
109
Patrick Venture74304642019-01-17 09:31:04 -0800110bool PciDataHandler::writeMeta(const std::vector<std::uint8_t>& configuration)
Patrick Venture8c535332018-11-08 15:58:00 -0800111{
112 /* PCI handler doesn't require configuration write, only read. */
113 return false;
114}
115
Patrick Venture74304642019-01-17 09:31:04 -0800116std::vector<std::uint8_t> PciDataHandler::readMeta()
Patrick Venture8c535332018-11-08 15:58:00 -0800117{
118 /* PCI handler does require returning a configuration from read. */
119 struct PciConfigResponse reply;
Patrick Ventureb0c84d02018-11-09 12:00:31 -0800120 reply.address = regionAddress;
Patrick Venture8c535332018-11-08 15:58:00 -0800121
122 std::vector<std::uint8_t> bytes;
123 bytes.resize(sizeof(reply));
Patrick Venturea8a99ae2018-11-09 11:14:58 -0800124 std::memcpy(bytes.data(), &reply, sizeof(reply));
Patrick Venture8c535332018-11-08 15:58:00 -0800125
126 return bytes;
127}
128
Patrick Venture1d5a31c2019-05-20 11:38:22 -0700129} // namespace ipmi_flash