init.cpp
Go to the documentation of this file.
1 // This program creates the udev rules necessary for quori's microcontrollers.
2 // It does this by first scanning for Arduinos, then connecting to each arduino
3 // and reading its published joints. We record the serial numbers of each device
4 // and write out the correct udev rules.
5 
6 #include <unistd.h>
7 #include <sys/types.h>
8 #include <sys/wait.h>
9 #include <string.h>
10 #include <sstream>
11 #include <iostream>
12 #include <vector>
13 #include <boost/filesystem.hpp>
14 #include <unordered_map>
15 
16 #include <fstream>
17 
18 #include "SerialDevice.hpp"
19 #include "message.hpp"
20 
21 
22 
23 using namespace boost::filesystem;
24 
25 using namespace quori_controller;
26 
27 std::string run(const path &program, const std::vector<std::string> &args)
28 {
29  const char **argv = new const char *[args.size() + 2];
30 
31  size_t index = 0;
32  argv[index++] = program.c_str();
33  for (auto it = args.cbegin(); it != args.cend(); ++it, ++index)
34  {
35  argv[index] = it->c_str();
36  }
37  argv[index] = NULL;
38 
39  int pipefd[2];
40 
41  if (pipe(pipefd) < 0) throw std::runtime_error(strerror(errno));
42 
43 
44  const pid_t pid = fork();
45  if(pid < 0) throw std::runtime_error(strerror(errno));
46 
47  if (pid == 0) {
48  dup2(pipefd[1], STDOUT_FILENO);
49  close(pipefd[0]);
50  close(pipefd[1]);
51 
52  execvp(program.c_str(), const_cast<char *const *>(argv));
53  _exit(1);
54  }
55 
56  delete[] argv;
57 
58  close(pipefd[1]);
59 
60  std::string out;
61  for (;;)
62  {
63  char buf[1024];
64  int count = read(pipefd[0], buf, sizeof buf);
65  if (count <= 0) break;
66 
67  out += std::string(buf, buf + count);
68  }
69 
70  int status;
71  wait(&status);
72 
73  if (status != 0) {
74  std::ostringstream o;
75  o << program << " returned " << status;
76  throw std::runtime_error(o.str());
77  }
78 
79  return out;
80 }
81 
82 struct DeviceInfo
83 {
85  std::string id_vendor;
86  std::string id_serial;
87 };
88 
89 std::ostream &operator <<(std::ostream &o, const DeviceInfo &v)
90 {
91  return o << v.device_path << " (id_serial: " << v.id_serial << ", id_vendor: " << v.id_vendor << ")";
92 }
93 
94 std::string extract_value(const std::string &text, const std::string &key)
95 {
96 
97  const std::size_t index = text.find(key);
98  if (index == std::string::npos)
99  {
100  std::ostringstream o;
101  o << "Failed to find key \"" << key << "\"";
102  throw std::runtime_error(o.str());
103  }
104  const std::size_t begin = index + key.size();
105  const std::size_t end = text.find("\n", begin);
106  return text.substr(begin, end - begin);
107 }
108 
109 DeviceInfo device_info(const path &device_path)
110 {
111  const std::string output = run("/usr/bin/udevadm", { "info", device_path.string() });
112  DeviceInfo ret;
113  ret.device_path = device_path;
114  ret.id_vendor = extract_value(output, "ID_VENDOR=");
115  ret.id_serial = extract_value(output, "ID_SERIAL_SHORT=");
116  return ret;
117 }
118 
119 template<typename T>
120 std::ostream &operator <<(std::ostream &o, const std::vector<T> &ts)
121 {
122  for (const auto &t : ts)
123  {
124  o << " " << t << std::endl;
125  }
126  return o;
127 }
128 
129 template<typename K, typename V>
130 std::ostream &operator <<(std::ostream &o, const std::unordered_map<K, V> &ts)
131 {
132  for (const auto &t : ts)
133  {
134  o << " " << t.first << ": " << t.second << std::endl;
135  }
136  return o;
137 }
138 
139 enum class Microcontroller : std::uint8_t
140 {
141  Unknown,
142  LeftArm,
143  RightArm,
144  Waist,
145  Base
146 };
147 
148 namespace std
149 {
150  template<>
151  struct hash<Microcontroller>
152  {
153  size_t operator() (const Microcontroller &microcontroller) const noexcept
154  {
155  return static_cast<std::uint8_t>(microcontroller);
156  }
157  };
158 }
159 
160 std::ostream &operator <<(std::ostream &o, const Microcontroller v)
161 {
162  switch (v)
163  {
164  case Microcontroller::Unknown: return o << "Unknown";
165  case Microcontroller::LeftArm: return o << "Left Arm";
166  case Microcontroller::RightArm: return o << "Right Arm";
167  case Microcontroller::Waist: return o << "Waist";
168  case Microcontroller::Base: return o << "Base";
169  }
170 }
171 
173 {
174  switch (v)
175  {
176  case Microcontroller::Unknown: return "unknown";
177  case Microcontroller::LeftArm: return "left_arm";
178  case Microcontroller::RightArm: return "right_arm";
179  case Microcontroller::Waist: return "waist";
180  case Microcontroller::Base: return "base";
181  }
182 }
183 
184 std::string udev_rule(const Microcontroller microcontroller, const std::string &serial)
185 {
186  std::ostringstream o;
187  o << "KERNEL==\"ttyACM[0-9]*\", SUBSYSTEM==\"tty\", ATTRS{serial}==\"" << serial << "\", SYMLINK+=\"quori/" << microcontroller_filename(microcontroller) << "\", MODE=\"0666\"" << std::endl;
188  return o.str();
189 }
190 
191 int main(int argc, char *argv[])
192 {
193  std::cout << "sizeof " << sizeof(quori::message::Initialize) << std::endl;
194  // Iterate through the /dev directory, finding devices of the form "ttyACM*"
195  std::vector<DeviceInfo> teensys;
196  std::unordered_map<Microcontroller, std::string> serials;
197 
198  for (directory_iterator it("/dev"); it != directory_iterator(); ++it)
199  {
200  if (it->path().filename().string().find("ttyACM") == std::string::npos) continue;
201 
202  // Each USB device has a VENDOR property we can introspect on.
203  // We only want Arduino Teensys
204  const DeviceInfo info = device_info(it->path());
205 
206  if (info.id_vendor == "STMicroelectronics")
207  {
208  serials.insert({ Microcontroller::Base, info.id_serial });
209  continue;
210  }
211 
212  if (info.id_vendor != "Teensyduino") continue;
213 
214  teensys.push_back(info);
215  }
216 
217  std::cout << "Found " << teensys.size() << " microcontrollers:" << std::endl << teensys;
218 
219  boost::asio::io_service service;
220 
221 
222 
223  std::cout << "Discovering mappings... ";
224  std::cout.flush();
225  // Connect to each teensy and determine which joints it has
226  for (const auto &teensy : teensys)
227  {
228  // std::cout << teensy.device_path << std::endl;
229  const auto serial_device = SerialDevice::open(service, teensy.device_path.string());
230  const auto info = serial_device->initialize();
231 
232  Microcontroller microcontroller = Microcontroller::Unknown;
233  const std::string &joint = info.joints[0].name;
234  std::cout << joint << std::endl;
235  if (joint.find("l_shoulder") != std::string::npos)
236  {
237  microcontroller = Microcontroller::LeftArm;
238  }
239  else if (joint.find("r_shoulder") != std::string::npos)
240  {
241  microcontroller = Microcontroller::RightArm;
242  }
243  else if (joint.find("waist") != std::string::npos)
244  {
245  microcontroller = Microcontroller::Waist;
246  }
247  else if (joint.find("base") != std::string::npos)
248  {
249  microcontroller = Microcontroller::Base;
250  }
251 
252  if (microcontroller == Microcontroller::Unknown)
253  {
254  std::ostringstream o;
255  o << "Unable to determine which quori microcontroller \"" << teensy << "\" is";
256  throw std::runtime_error(o.str());
257  }
258 
259  serials.insert({ microcontroller, teensy.id_serial });
260  }
261 
262  std::cout << "done!" << std::endl;
263 
264 
265 
266  const auto left_arm_it = serials.find(Microcontroller::LeftArm);
267  if (left_arm_it == serials.end())
268  {
269  std::ostringstream o;
270  o << "Didn't find Quori's Left Arm";
271  throw std::runtime_error(o.str());
272  }
273 
274  const auto right_arm_it = serials.find(Microcontroller::RightArm);
275  if (right_arm_it == serials.end())
276  {
277  std::ostringstream o;
278  o << "Didn't find Quori's Right Arm";
279  throw std::runtime_error(o.str());
280  }
281 
282  const auto waist_it = serials.find(Microcontroller::Waist);
283  if (waist_it == serials.end())
284  {
285  std::ostringstream o;
286  o << "Didn't find Quori's Waist";
287  throw std::runtime_error(o.str());
288  }
289 
290  const auto base_it = serials.find(Microcontroller::Base);
291  if (base_it == serials.end())
292  {
293  std::ostringstream o;
294  o << "Didn't find Quori's Base";
295  throw std::runtime_error(o.str());
296  }
297 
298  std::cout << "Found microcontroller mappings:" << std::endl << serials;
299 
300  const static path RULES_PATH = "/etc/udev/rules.d/80-quori.rules";
301 
302  std::ofstream rules(RULES_PATH.string());
303  if (!rules.is_open())
304  {
305  std::ostringstream o;
306  o << "Unable to open \"" << RULES_PATH << "\" for writing. Make sure you run this program as superuser!";
307  throw std::runtime_error(o.str());
308  }
309 
310  for (const auto &mapping : serials)
311  {
312  rules << udev_rule(mapping.first, mapping.second);
313  }
314 
315  // Make all ttyUSB* devices user-rw as well
316  rules << "KERNEL==\"ttyUSB[0-9]*\", SUBSYSTEM==\"usb\", MODE=\"0666\"" << std::endl;
317 
318  rules.close();
319 
320  std::cout << "Successfully wrote udev rules!" << std::endl;
321 
322  bool should_reboot = true;
323  for (;;)
324  {
325  std::cout << "You must reboot for changes to take effect. Would you like to reboot now? [Y/n]: ";
326  char line[2];
327  std::cin.getline(line, sizeof line);
328 
329  std::cout << line << std::endl;
330 
331  if (line[0] == 0) break;
332 
333  if (line[0] == 'Y' || line[0] == 'y')
334  {
335  should_reboot = true;
336  break;
337  }
338 
339  if (line[0] == 'N' || line[0] == 'n')
340  {
341  should_reboot = false;
342  break;
343  }
344  }
345 
346  if (should_reboot)
347  {
348  std::cout << "Rebooting..." << std::endl;
349  run("/sbin/reboot", {});
350  }
351 
352  return EXIT_SUCCESS;
353 }
operator<<
std::ostream & operator<<(std::ostream &o, const DeviceInfo &v)
Definition: init.cpp:89
message.hpp
Microcontroller::LeftArm
@ LeftArm
Microcontroller::Unknown
@ Unknown
quori::message::Initialize
Definition: message.hpp:29
DeviceInfo::device_path
path device_path
Definition: init.cpp:84
wait
void wait(int seconds)
DeviceInfo::id_vendor
std::string id_vendor
Definition: init.cpp:85
run
std::string run(const path &program, const std::vector< std::string > &args)
Definition: init.cpp:27
quori_controller
Definition: Csv.hpp:10
device_info
DeviceInfo device_info(const path &device_path)
Definition: init.cpp:109
Base
DeviceInfo
Definition: init.cpp:82
udev_rule
std::string udev_rule(const Microcontroller microcontroller, const std::string &serial)
Definition: init.cpp:184
extract_value
std::string extract_value(const std::string &text, const std::string &key)
Definition: init.cpp:94
DeviceInfo::id_serial
std::string id_serial
Definition: init.cpp:86
Microcontroller::Waist
@ Waist
Microcontroller::RightArm
@ RightArm
std
Microcontroller::Base
@ Base
main
int main(int argc, char *argv[])
Definition: init.cpp:191
SerialDevice.hpp
args
microcontroller_filename
std::string microcontroller_filename(const Microcontroller v)
Definition: init.cpp:172
Microcontroller
Microcontroller
Definition: init.cpp:139


quori_controller
Author(s):
autogenerated on Wed Mar 2 2022 00:53:16