1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker * Copyright 2022, The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker *
4*4d7e907cSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker *
8*4d7e907cSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker *
10*4d7e907cSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker */
16*4d7e907cSAndroid Build Coastguard Worker
17*4d7e907cSAndroid Build Coastguard Worker #include "CanController.h"
18*4d7e907cSAndroid Build Coastguard Worker
19*4d7e907cSAndroid Build Coastguard Worker #include "CanBusNative.h"
20*4d7e907cSAndroid Build Coastguard Worker #include "CanBusSlcan.h"
21*4d7e907cSAndroid Build Coastguard Worker #include "CanBusVirtual.h"
22*4d7e907cSAndroid Build Coastguard Worker
23*4d7e907cSAndroid Build Coastguard Worker #include <android-base/format.h>
24*4d7e907cSAndroid Build Coastguard Worker #include <android-base/logging.h>
25*4d7e907cSAndroid Build Coastguard Worker
26*4d7e907cSAndroid Build Coastguard Worker #include <filesystem>
27*4d7e907cSAndroid Build Coastguard Worker #include <fstream>
28*4d7e907cSAndroid Build Coastguard Worker #include <regex>
29*4d7e907cSAndroid Build Coastguard Worker
30*4d7e907cSAndroid Build Coastguard Worker namespace aidl::android::hardware::automotive::can {
31*4d7e907cSAndroid Build Coastguard Worker
32*4d7e907cSAndroid Build Coastguard Worker namespace fs = ::std::filesystem;
33*4d7e907cSAndroid Build Coastguard Worker
34*4d7e907cSAndroid Build Coastguard Worker namespace fsErrors {
35*4d7e907cSAndroid Build Coastguard Worker static const std::error_code ok;
36*4d7e907cSAndroid Build Coastguard Worker static const std::error_code eperm(EPERM, std::generic_category());
37*4d7e907cSAndroid Build Coastguard Worker static const std::error_code enoent(ENOENT, std::generic_category());
38*4d7e907cSAndroid Build Coastguard Worker static const std::error_code eacces(EACCES, std::generic_category());
39*4d7e907cSAndroid Build Coastguard Worker } // namespace fsErrors
40*4d7e907cSAndroid Build Coastguard Worker
41*4d7e907cSAndroid Build Coastguard Worker /* In the /sys/devices tree, there are files called "serial", which contain the serial numbers
42*4d7e907cSAndroid Build Coastguard Worker * for various devices. The exact location inside of this directory is dependent upon the
43*4d7e907cSAndroid Build Coastguard Worker * hardware we are running on, so we have to start from /sys/devices and work our way down. */
44*4d7e907cSAndroid Build Coastguard Worker static const fs::path kDevPath("/sys/devices/");
45*4d7e907cSAndroid Build Coastguard Worker static const std::regex kTtyRe("^tty[A-Z]+[0-9]+$");
46*4d7e907cSAndroid Build Coastguard Worker static constexpr auto kOpts = ~(fs::directory_options::follow_directory_symlink |
47*4d7e907cSAndroid Build Coastguard Worker fs::directory_options::skip_permission_denied);
48*4d7e907cSAndroid Build Coastguard Worker
49*4d7e907cSAndroid Build Coastguard Worker constexpr auto ok = &ndk::ScopedAStatus::ok;
50*4d7e907cSAndroid Build Coastguard Worker
51*4d7e907cSAndroid Build Coastguard Worker /**
52*4d7e907cSAndroid Build Coastguard Worker * A helper object to associate the interface name and type of a USB to CAN adapter.
53*4d7e907cSAndroid Build Coastguard Worker */
54*4d7e907cSAndroid Build Coastguard Worker struct UsbCanIface {
55*4d7e907cSAndroid Build Coastguard Worker InterfaceType iftype;
56*4d7e907cSAndroid Build Coastguard Worker std::string ifaceName;
57*4d7e907cSAndroid Build Coastguard Worker };
58*4d7e907cSAndroid Build Coastguard Worker
isValidName(const std::string & name)59*4d7e907cSAndroid Build Coastguard Worker static bool isValidName(const std::string& name) {
60*4d7e907cSAndroid Build Coastguard Worker static const std::regex nameRE("^[a-zA-Z0-9_]{1,32}$");
61*4d7e907cSAndroid Build Coastguard Worker return std::regex_match(name, nameRE);
62*4d7e907cSAndroid Build Coastguard Worker }
63*4d7e907cSAndroid Build Coastguard Worker
64*4d7e907cSAndroid Build Coastguard Worker /**
65*4d7e907cSAndroid Build Coastguard Worker * Given a path, get the last element from it.
66*4d7e907cSAndroid Build Coastguard Worker *
67*4d7e907cSAndroid Build Coastguard Worker * \param itrPath - the path we want the last element of
68*4d7e907cSAndroid Build Coastguard Worker * \return - the last element in the path (in string form).
69*4d7e907cSAndroid Build Coastguard Worker */
getLeaf(const fs::path & itrPath)70*4d7e907cSAndroid Build Coastguard Worker static std::string getLeaf(const fs::path& itrPath) {
71*4d7e907cSAndroid Build Coastguard Worker /* end() returns an iterator one past the leaf of the path, so we've overshot
72*4d7e907cSAndroid Build Coastguard Worker decrement (--) to go back one to the leaf
73*4d7e907cSAndroid Build Coastguard Worker dereference and now we have our leaf. */
74*4d7e907cSAndroid Build Coastguard Worker return *(--(itrPath.end()));
75*4d7e907cSAndroid Build Coastguard Worker }
76*4d7e907cSAndroid Build Coastguard Worker
resultToStatus(Result res,const std::string & msg="")77*4d7e907cSAndroid Build Coastguard Worker static ndk::ScopedAStatus resultToStatus(Result res, const std::string& msg = "") {
78*4d7e907cSAndroid Build Coastguard Worker if (msg.empty()) {
79*4d7e907cSAndroid Build Coastguard Worker return ndk::ScopedAStatus(AStatus_fromServiceSpecificError(static_cast<int>(res)));
80*4d7e907cSAndroid Build Coastguard Worker }
81*4d7e907cSAndroid Build Coastguard Worker return ndk::ScopedAStatus(
82*4d7e907cSAndroid Build Coastguard Worker AStatus_fromServiceSpecificErrorWithMessage(static_cast<int>(res), msg.c_str()));
83*4d7e907cSAndroid Build Coastguard Worker }
84*4d7e907cSAndroid Build Coastguard Worker
85*4d7e907cSAndroid Build Coastguard Worker /**
86*4d7e907cSAndroid Build Coastguard Worker * Given a UsbCanIface object, get the ifaceName given the serialPath.
87*4d7e907cSAndroid Build Coastguard Worker *
88*4d7e907cSAndroid Build Coastguard Worker * \param serialPath - Absolute path to a "serial" file for a given device in /sys.
89*4d7e907cSAndroid Build Coastguard Worker * \return A populated UsbCanIface. On failure, nullopt is returned.
90*4d7e907cSAndroid Build Coastguard Worker */
getIfaceName(const fs::path & serialPath)91*4d7e907cSAndroid Build Coastguard Worker static std::optional<UsbCanIface> getIfaceName(const fs::path& serialPath) {
92*4d7e907cSAndroid Build Coastguard Worker std::error_code fsStatus;
93*4d7e907cSAndroid Build Coastguard Worker // Since the path is to a file called "serial", we need to search its parent directory.
94*4d7e907cSAndroid Build Coastguard Worker fs::recursive_directory_iterator fsItr(serialPath.parent_path(), kOpts, fsStatus);
95*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
96*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed to open " << serialPath.parent_path();
97*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
98*4d7e907cSAndroid Build Coastguard Worker }
99*4d7e907cSAndroid Build Coastguard Worker
100*4d7e907cSAndroid Build Coastguard Worker for (; fsStatus == fsErrors::ok && fsItr != fs::recursive_directory_iterator();
101*4d7e907cSAndroid Build Coastguard Worker fsItr.increment(fsStatus)) {
102*4d7e907cSAndroid Build Coastguard Worker /* We want either a directory called "net" or a directory that looks like tty<something>, so
103*4d7e907cSAndroid Build Coastguard Worker * skip files. */
104*4d7e907cSAndroid Build Coastguard Worker bool isDir = fsItr->is_directory(fsStatus);
105*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok || !isDir) continue;
106*4d7e907cSAndroid Build Coastguard Worker
107*4d7e907cSAndroid Build Coastguard Worker std::string currentDir = getLeaf(fsItr->path());
108*4d7e907cSAndroid Build Coastguard Worker if (currentDir == "net") {
109*4d7e907cSAndroid Build Coastguard Worker /* This device is a SocketCAN device. The iface name is the only directory under
110*4d7e907cSAndroid Build Coastguard Worker * net/. Multiple directories under net/ is an error.*/
111*4d7e907cSAndroid Build Coastguard Worker fs::directory_iterator netItr(fsItr->path(), kOpts, fsStatus);
112*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
113*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed to open " << fsItr->path() << " to get net name!";
114*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
115*4d7e907cSAndroid Build Coastguard Worker }
116*4d7e907cSAndroid Build Coastguard Worker
117*4d7e907cSAndroid Build Coastguard Worker // The leaf of our path should be the interface name.
118*4d7e907cSAndroid Build Coastguard Worker std::string netName = getLeaf(netItr->path());
119*4d7e907cSAndroid Build Coastguard Worker
120*4d7e907cSAndroid Build Coastguard Worker // Check if there is more than one item in net/
121*4d7e907cSAndroid Build Coastguard Worker netItr.increment(fsStatus);
122*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
123*4d7e907cSAndroid Build Coastguard Worker // It's possible we have a valid net name, but this is most likely an error.
124*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed to verify " << fsItr->path() << " has valid net name!";
125*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
126*4d7e907cSAndroid Build Coastguard Worker }
127*4d7e907cSAndroid Build Coastguard Worker if (netItr != fs::directory_iterator()) {
128*4d7e907cSAndroid Build Coastguard Worker // There should never be more than one name under net/
129*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Found more than one net name in " << fsItr->path() << "!";
130*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
131*4d7e907cSAndroid Build Coastguard Worker }
132*4d7e907cSAndroid Build Coastguard Worker return {{InterfaceType::NATIVE, netName}};
133*4d7e907cSAndroid Build Coastguard Worker } else if (std::regex_match(currentDir, kTtyRe)) {
134*4d7e907cSAndroid Build Coastguard Worker // This device is a USB serial device, and currentDir is the tty name.
135*4d7e907cSAndroid Build Coastguard Worker return {{InterfaceType::SLCAN, "/dev/" + currentDir}};
136*4d7e907cSAndroid Build Coastguard Worker }
137*4d7e907cSAndroid Build Coastguard Worker }
138*4d7e907cSAndroid Build Coastguard Worker
139*4d7e907cSAndroid Build Coastguard Worker // check if the loop above exited due to a c++fs error.
140*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
141*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed search filesystem: " << fsStatus;
142*4d7e907cSAndroid Build Coastguard Worker }
143*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
144*4d7e907cSAndroid Build Coastguard Worker }
145*4d7e907cSAndroid Build Coastguard Worker
146*4d7e907cSAndroid Build Coastguard Worker /**
147*4d7e907cSAndroid Build Coastguard Worker * A helper function to read the serial number from a "serial" file in /sys/devices/
148*4d7e907cSAndroid Build Coastguard Worker *
149*4d7e907cSAndroid Build Coastguard Worker * \param serialnoPath - path to the file to read.
150*4d7e907cSAndroid Build Coastguard Worker * \return the serial number, or nullopt on failure.
151*4d7e907cSAndroid Build Coastguard Worker */
readSerialNo(const std::string & serialnoPath)152*4d7e907cSAndroid Build Coastguard Worker static std::optional<std::string> readSerialNo(const std::string& serialnoPath) {
153*4d7e907cSAndroid Build Coastguard Worker std::ifstream serialnoStream(serialnoPath);
154*4d7e907cSAndroid Build Coastguard Worker std::string serialno;
155*4d7e907cSAndroid Build Coastguard Worker if (!serialnoStream.good()) {
156*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed to read serial number from " << serialnoPath;
157*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
158*4d7e907cSAndroid Build Coastguard Worker }
159*4d7e907cSAndroid Build Coastguard Worker std::getline(serialnoStream, serialno);
160*4d7e907cSAndroid Build Coastguard Worker return serialno;
161*4d7e907cSAndroid Build Coastguard Worker }
162*4d7e907cSAndroid Build Coastguard Worker
163*4d7e907cSAndroid Build Coastguard Worker /**
164*4d7e907cSAndroid Build Coastguard Worker * Searches for USB devices found in /sys/devices/, and attempts to find a device matching the
165*4d7e907cSAndroid Build Coastguard Worker * provided list of serial numbers.
166*4d7e907cSAndroid Build Coastguard Worker *
167*4d7e907cSAndroid Build Coastguard Worker * \param configSerialnos - a list of serial number (suffixes) from the HAL config.
168*4d7e907cSAndroid Build Coastguard Worker * \param iftype - the type of the interface to be located.
169*4d7e907cSAndroid Build Coastguard Worker * \return a matching USB device. On failure, std::nullopt is returned.
170*4d7e907cSAndroid Build Coastguard Worker */
findUsbDevice(const std::vector<std::string> & configSerialnos)171*4d7e907cSAndroid Build Coastguard Worker static std::optional<UsbCanIface> findUsbDevice(const std::vector<std::string>& configSerialnos) {
172*4d7e907cSAndroid Build Coastguard Worker std::error_code fsStatus;
173*4d7e907cSAndroid Build Coastguard Worker fs::recursive_directory_iterator fsItr(kDevPath, kOpts, fsStatus);
174*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
175*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed to open " << kDevPath;
176*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
177*4d7e907cSAndroid Build Coastguard Worker }
178*4d7e907cSAndroid Build Coastguard Worker
179*4d7e907cSAndroid Build Coastguard Worker for (; fsStatus == fsErrors::ok && fsItr != fs::recursive_directory_iterator();
180*4d7e907cSAndroid Build Coastguard Worker fsItr.increment(fsStatus)) {
181*4d7e907cSAndroid Build Coastguard Worker // We want to find a file called "serial", which is in a directory somewhere. Skip files.
182*4d7e907cSAndroid Build Coastguard Worker bool isDir = fsItr->is_directory(fsStatus);
183*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
184*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Failed check if " << fsStatus;
185*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
186*4d7e907cSAndroid Build Coastguard Worker }
187*4d7e907cSAndroid Build Coastguard Worker if (!isDir) continue;
188*4d7e907cSAndroid Build Coastguard Worker
189*4d7e907cSAndroid Build Coastguard Worker auto serialnoPath = fsItr->path() / "serial";
190*4d7e907cSAndroid Build Coastguard Worker bool isReg = fs::is_regular_file(serialnoPath, fsStatus);
191*4d7e907cSAndroid Build Coastguard Worker
192*4d7e907cSAndroid Build Coastguard Worker /* Make sure we have permissions to this directory, ignore enoent, since the file
193*4d7e907cSAndroid Build Coastguard Worker * "serial" may not exist, which is ok. */
194*4d7e907cSAndroid Build Coastguard Worker if (fsStatus == fsErrors::eperm || fsStatus == fsErrors::eacces) {
195*4d7e907cSAndroid Build Coastguard Worker /* This means we don't have access to this directory. If we recurse into it, this
196*4d7e907cSAndroid Build Coastguard Worker * will cause the iterator to loose its state and we'll crash. */
197*4d7e907cSAndroid Build Coastguard Worker fsItr.disable_recursion_pending();
198*4d7e907cSAndroid Build Coastguard Worker continue;
199*4d7e907cSAndroid Build Coastguard Worker }
200*4d7e907cSAndroid Build Coastguard Worker if (fsStatus == fsErrors::enoent) continue;
201*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
202*4d7e907cSAndroid Build Coastguard Worker LOG(WARNING) << "An unexpected error occurred while checking for serialno: "
203*4d7e907cSAndroid Build Coastguard Worker << fsStatus;
204*4d7e907cSAndroid Build Coastguard Worker continue;
205*4d7e907cSAndroid Build Coastguard Worker }
206*4d7e907cSAndroid Build Coastguard Worker if (!isReg) continue;
207*4d7e907cSAndroid Build Coastguard Worker
208*4d7e907cSAndroid Build Coastguard Worker // we found a serial number
209*4d7e907cSAndroid Build Coastguard Worker auto serialno = readSerialNo(serialnoPath);
210*4d7e907cSAndroid Build Coastguard Worker if (!serialno.has_value()) continue;
211*4d7e907cSAndroid Build Coastguard Worker
212*4d7e907cSAndroid Build Coastguard Worker // see if the serial number exists in the config
213*4d7e907cSAndroid Build Coastguard Worker for (auto&& cfgSn : configSerialnos) {
214*4d7e907cSAndroid Build Coastguard Worker if (serialno->ends_with(std::string(cfgSn))) {
215*4d7e907cSAndroid Build Coastguard Worker auto ifaceInfo = getIfaceName(serialnoPath);
216*4d7e907cSAndroid Build Coastguard Worker if (!ifaceInfo.has_value()) break;
217*4d7e907cSAndroid Build Coastguard Worker return ifaceInfo;
218*4d7e907cSAndroid Build Coastguard Worker }
219*4d7e907cSAndroid Build Coastguard Worker }
220*4d7e907cSAndroid Build Coastguard Worker }
221*4d7e907cSAndroid Build Coastguard Worker if (fsStatus != fsErrors::ok) {
222*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Error searching filesystem: " << fsStatus;
223*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
224*4d7e907cSAndroid Build Coastguard Worker }
225*4d7e907cSAndroid Build Coastguard Worker return std::nullopt;
226*4d7e907cSAndroid Build Coastguard Worker }
227*4d7e907cSAndroid Build Coastguard Worker
getSupportedInterfaceTypes(std::vector<InterfaceType> * supportedTypes)228*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus CanController::getSupportedInterfaceTypes(
229*4d7e907cSAndroid Build Coastguard Worker std::vector<InterfaceType>* supportedTypes) {
230*4d7e907cSAndroid Build Coastguard Worker *supportedTypes = {InterfaceType::VIRTUAL, InterfaceType::NATIVE, InterfaceType::SLCAN};
231*4d7e907cSAndroid Build Coastguard Worker return ok();
232*4d7e907cSAndroid Build Coastguard Worker }
233*4d7e907cSAndroid Build Coastguard Worker
getInterfaceName(const std::string & busName,std::string * ifaceName)234*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus CanController::getInterfaceName(const std::string& busName,
235*4d7e907cSAndroid Build Coastguard Worker std::string* ifaceName) {
236*4d7e907cSAndroid Build Coastguard Worker *ifaceName = {};
237*4d7e907cSAndroid Build Coastguard Worker if (mBusesByName.find(busName) == mBusesByName.end()) {
238*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(Result::BAD_BUS_NAME, fmt::format("{} doesn't exist", busName));
239*4d7e907cSAndroid Build Coastguard Worker }
240*4d7e907cSAndroid Build Coastguard Worker *ifaceName = std::string(mBusesByName[busName]->getIfaceName());
241*4d7e907cSAndroid Build Coastguard Worker return ok();
242*4d7e907cSAndroid Build Coastguard Worker }
243*4d7e907cSAndroid Build Coastguard Worker
upBus(const BusConfig & config,std::string * ifaceName)244*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus CanController::upBus(const BusConfig& config, std::string* ifaceName) {
245*4d7e907cSAndroid Build Coastguard Worker if (!isValidName(config.name)) {
246*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Bus name " << config.name << " is invalid";
247*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(Result::BAD_BUS_NAME,
248*4d7e907cSAndroid Build Coastguard Worker fmt::format("{} is not a valid bus name", config.name));
249*4d7e907cSAndroid Build Coastguard Worker } else if (mBusesByName.find(config.name) != mBusesByName.end()) {
250*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "A bus named " << config.name << " already exists!";
251*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(Result::INVALID_STATE,
252*4d7e907cSAndroid Build Coastguard Worker fmt::format("A bus named {} already exists", config.name));
253*4d7e907cSAndroid Build Coastguard Worker }
254*4d7e907cSAndroid Build Coastguard Worker
255*4d7e907cSAndroid Build Coastguard Worker if (config.interfaceId.getTag() == BusConfig::InterfaceId::Tag::virtualif) {
256*4d7e907cSAndroid Build Coastguard Worker auto& virtualif = config.interfaceId.get<BusConfig::InterfaceId::Tag::virtualif>();
257*4d7e907cSAndroid Build Coastguard Worker mBusesByName[config.name] = std::make_unique<CanBusVirtual>(virtualif.ifname);
258*4d7e907cSAndroid Build Coastguard Worker }
259*4d7e907cSAndroid Build Coastguard Worker
260*4d7e907cSAndroid Build Coastguard Worker else if (config.interfaceId.getTag() == BusConfig::InterfaceId::Tag::nativeif) {
261*4d7e907cSAndroid Build Coastguard Worker auto& nativeif = config.interfaceId.get<BusConfig::InterfaceId::Tag::nativeif>();
262*4d7e907cSAndroid Build Coastguard Worker std::string ifaceName;
263*4d7e907cSAndroid Build Coastguard Worker if (nativeif.interfaceId.getTag() == NativeInterface::InterfaceId::Tag::serialno) {
264*4d7e907cSAndroid Build Coastguard Worker // Configure by serial number.
265*4d7e907cSAndroid Build Coastguard Worker auto selectedDevice = findUsbDevice(
266*4d7e907cSAndroid Build Coastguard Worker nativeif.interfaceId.get<NativeInterface::InterfaceId::Tag::serialno>());
267*4d7e907cSAndroid Build Coastguard Worker // verify the returned device is the correct one
268*4d7e907cSAndroid Build Coastguard Worker if (!selectedDevice.has_value() || selectedDevice->iftype != InterfaceType::NATIVE) {
269*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(
270*4d7e907cSAndroid Build Coastguard Worker Result::BAD_INTERFACE_ID,
271*4d7e907cSAndroid Build Coastguard Worker "Couldn't find a native socketcan device with the given serial number(s)");
272*4d7e907cSAndroid Build Coastguard Worker }
273*4d7e907cSAndroid Build Coastguard Worker ifaceName = selectedDevice->ifaceName;
274*4d7e907cSAndroid Build Coastguard Worker } else {
275*4d7e907cSAndroid Build Coastguard Worker // configure by iface name.
276*4d7e907cSAndroid Build Coastguard Worker ifaceName = nativeif.interfaceId.get<NativeInterface::InterfaceId::Tag::ifname>();
277*4d7e907cSAndroid Build Coastguard Worker }
278*4d7e907cSAndroid Build Coastguard Worker mBusesByName[config.name] = std::make_unique<CanBusNative>(ifaceName, config.bitrate);
279*4d7e907cSAndroid Build Coastguard Worker }
280*4d7e907cSAndroid Build Coastguard Worker
281*4d7e907cSAndroid Build Coastguard Worker else if (config.interfaceId.getTag() == BusConfig::InterfaceId::Tag::slcan) {
282*4d7e907cSAndroid Build Coastguard Worker auto& slcanif = config.interfaceId.get<BusConfig::InterfaceId::Tag::slcan>();
283*4d7e907cSAndroid Build Coastguard Worker std::string ttyName;
284*4d7e907cSAndroid Build Coastguard Worker if (slcanif.interfaceId.getTag() == SlcanInterface::InterfaceId::Tag::serialno) {
285*4d7e907cSAndroid Build Coastguard Worker // Configure by serial number.
286*4d7e907cSAndroid Build Coastguard Worker auto selectedDevice = findUsbDevice(
287*4d7e907cSAndroid Build Coastguard Worker slcanif.interfaceId.get<SlcanInterface::InterfaceId::Tag::serialno>());
288*4d7e907cSAndroid Build Coastguard Worker if (!selectedDevice.has_value() || selectedDevice->iftype != InterfaceType::SLCAN) {
289*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(
290*4d7e907cSAndroid Build Coastguard Worker Result::BAD_INTERFACE_ID,
291*4d7e907cSAndroid Build Coastguard Worker "Couldn't find a slcan device with the given serial number(s)");
292*4d7e907cSAndroid Build Coastguard Worker }
293*4d7e907cSAndroid Build Coastguard Worker ttyName = selectedDevice->ifaceName;
294*4d7e907cSAndroid Build Coastguard Worker } else {
295*4d7e907cSAndroid Build Coastguard Worker // Configure by tty name.
296*4d7e907cSAndroid Build Coastguard Worker ttyName = slcanif.interfaceId.get<SlcanInterface::InterfaceId::Tag::ttyname>();
297*4d7e907cSAndroid Build Coastguard Worker }
298*4d7e907cSAndroid Build Coastguard Worker mBusesByName[config.name] = std::make_unique<CanBusSlcan>(ttyName, config.bitrate);
299*4d7e907cSAndroid Build Coastguard Worker }
300*4d7e907cSAndroid Build Coastguard Worker
301*4d7e907cSAndroid Build Coastguard Worker else if (config.interfaceId.getTag() == BusConfig::InterfaceId::Tag::indexed) {
302*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(Result::NOT_SUPPORTED,
303*4d7e907cSAndroid Build Coastguard Worker "Indexed devices are not supported in this implementation");
304*4d7e907cSAndroid Build Coastguard Worker } else {
305*4d7e907cSAndroid Build Coastguard Worker // this shouldn't happen.
306*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(Result::UNKNOWN_ERROR, "Unknown interface id type");
307*4d7e907cSAndroid Build Coastguard Worker }
308*4d7e907cSAndroid Build Coastguard Worker
309*4d7e907cSAndroid Build Coastguard Worker Result result = mBusesByName[config.name]->up();
310*4d7e907cSAndroid Build Coastguard Worker if (result != Result::OK) {
311*4d7e907cSAndroid Build Coastguard Worker // the bus failed to come up, don't leave a broken entry in the map.
312*4d7e907cSAndroid Build Coastguard Worker mBusesByName.erase(config.name);
313*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(result, fmt::format("CanBus::up failed for {}", config.name));
314*4d7e907cSAndroid Build Coastguard Worker }
315*4d7e907cSAndroid Build Coastguard Worker
316*4d7e907cSAndroid Build Coastguard Worker *ifaceName = mBusesByName[config.name]->getIfaceName();
317*4d7e907cSAndroid Build Coastguard Worker return ok();
318*4d7e907cSAndroid Build Coastguard Worker }
319*4d7e907cSAndroid Build Coastguard Worker
downBus(const std::string & busName)320*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus CanController::downBus(const std::string& busName) {
321*4d7e907cSAndroid Build Coastguard Worker if (mBusesByName.find(busName) == mBusesByName.end()) {
322*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(
323*4d7e907cSAndroid Build Coastguard Worker Result::UNKNOWN_ERROR,
324*4d7e907cSAndroid Build Coastguard Worker fmt::format("Couldn't bring down {}, because it doesn't exist", busName));
325*4d7e907cSAndroid Build Coastguard Worker }
326*4d7e907cSAndroid Build Coastguard Worker Result result = mBusesByName[busName]->down();
327*4d7e907cSAndroid Build Coastguard Worker if (result != Result::OK) {
328*4d7e907cSAndroid Build Coastguard Worker return resultToStatus(result, fmt::format("Couldn't bring down {}!", busName));
329*4d7e907cSAndroid Build Coastguard Worker }
330*4d7e907cSAndroid Build Coastguard Worker mBusesByName.erase(busName);
331*4d7e907cSAndroid Build Coastguard Worker return ok();
332*4d7e907cSAndroid Build Coastguard Worker }
333*4d7e907cSAndroid Build Coastguard Worker } // namespace aidl::android::hardware::automotive::can
334