13 #include <boost/filesystem.hpp>
14 #include <unordered_map>
23 using namespace boost::filesystem;
27 std::string
run(
const path &program,
const std::vector<std::string> &
args)
29 const char **argv =
new const char *[
args.size() + 2];
32 argv[index++] = program.c_str();
33 for (
auto it =
args.cbegin(); it !=
args.cend(); ++it, ++index)
35 argv[index] = it->c_str();
41 if (pipe(pipefd) < 0)
throw std::runtime_error(strerror(errno));
44 const pid_t pid = fork();
45 if(pid < 0)
throw std::runtime_error(strerror(errno));
48 dup2(pipefd[1], STDOUT_FILENO);
52 execvp(program.c_str(),
const_cast<char *
const *
>(argv));
64 int count = read(pipefd[0], buf,
sizeof buf);
65 if (count <= 0)
break;
67 out += std::string(buf, buf + count);
75 o << program <<
" returned " << status;
76 throw std::runtime_error(o.str());
94 std::string
extract_value(
const std::string &text,
const std::string &key)
97 const std::size_t index = text.find(key);
98 if (index == std::string::npos)
100 std::ostringstream o;
101 o <<
"Failed to find key \"" << key <<
"\"";
102 throw std::runtime_error(o.str());
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);
111 const std::string output =
run(
"/usr/bin/udevadm", {
"info", device_path.string() });
120 std::ostream &
operator <<(std::ostream &o,
const std::vector<T> &ts)
122 for (
const auto &t : ts)
124 o <<
" " << t << std::endl;
129 template<
typename K,
typename V>
130 std::ostream &
operator <<(std::ostream &o,
const std::unordered_map<K, V> &ts)
132 for (
const auto &t : ts)
134 o <<
" " << t.first <<
": " << t.second << std::endl;
155 return static_cast<std::uint8_t
>(microcontroller);
186 std::ostringstream o;
187 o <<
"KERNEL==\"ttyACM[0-9]*\", SUBSYSTEM==\"tty\", ATTRS{serial}==\"" << serial <<
"\", SYMLINK+=\"quori/" <<
microcontroller_filename(microcontroller) <<
"\", MODE=\"0666\"" << std::endl;
191 int main(
int argc,
char *argv[])
195 std::vector<DeviceInfo> teensys;
196 std::unordered_map<Microcontroller, std::string> serials;
198 for (directory_iterator it(
"/dev"); it != directory_iterator(); ++it)
200 if (it->path().filename().string().find(
"ttyACM") == std::string::npos)
continue;
206 if (info.
id_vendor ==
"STMicroelectronics")
212 if (info.
id_vendor !=
"Teensyduino")
continue;
214 teensys.push_back(info);
217 std::cout <<
"Found " << teensys.size() <<
" microcontrollers:" << std::endl << teensys;
219 boost::asio::io_service service;
223 std::cout <<
"Discovering mappings... ";
226 for (
const auto &teensy : teensys)
229 const auto serial_device = SerialDevice::open(service, teensy.device_path.string());
230 const auto info = serial_device->initialize();
233 const std::string &joint = info.joints[0].name;
234 std::cout << joint << std::endl;
235 if (joint.find(
"l_shoulder") != std::string::npos)
239 else if (joint.find(
"r_shoulder") != std::string::npos)
243 else if (joint.find(
"waist") != std::string::npos)
247 else if (joint.find(
"base") != std::string::npos)
254 std::ostringstream o;
255 o <<
"Unable to determine which quori microcontroller \"" << teensy <<
"\" is";
256 throw std::runtime_error(o.str());
259 serials.insert({ microcontroller, teensy.id_serial });
262 std::cout <<
"done!" << std::endl;
267 if (left_arm_it == serials.end())
269 std::ostringstream o;
270 o <<
"Didn't find Quori's Left Arm";
271 throw std::runtime_error(o.str());
275 if (right_arm_it == serials.end())
277 std::ostringstream o;
278 o <<
"Didn't find Quori's Right Arm";
279 throw std::runtime_error(o.str());
283 if (waist_it == serials.end())
285 std::ostringstream o;
286 o <<
"Didn't find Quori's Waist";
287 throw std::runtime_error(o.str());
291 if (base_it == serials.end())
293 std::ostringstream o;
294 o <<
"Didn't find Quori's Base";
295 throw std::runtime_error(o.str());
298 std::cout <<
"Found microcontroller mappings:" << std::endl << serials;
300 const static path RULES_PATH =
"/etc/udev/rules.d/80-quori.rules";
302 std::ofstream rules(RULES_PATH.string());
303 if (!rules.is_open())
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());
310 for (
const auto &mapping : serials)
312 rules <<
udev_rule(mapping.first, mapping.second);
316 rules <<
"KERNEL==\"ttyUSB[0-9]*\", SUBSYSTEM==\"usb\", MODE=\"0666\"" << std::endl;
320 std::cout <<
"Successfully wrote udev rules!" << std::endl;
322 bool should_reboot =
true;
325 std::cout <<
"You must reboot for changes to take effect. Would you like to reboot now? [Y/n]: ";
327 std::cin.getline(line,
sizeof line);
329 std::cout << line << std::endl;
331 if (line[0] == 0)
break;
333 if (line[0] ==
'Y' || line[0] ==
'y')
335 should_reboot =
true;
339 if (line[0] ==
'N' || line[0] ==
'n')
341 should_reboot =
false;
348 std::cout <<
"Rebooting..." << std::endl;
349 run(
"/sbin/reboot", {});