stty.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609
  1. /*
  2. * Copyright (c) 2021, Daniel Bertalan <dani@danielbertalan.dev>
  3. * Copyright (c) 2022, Alex Major
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #define __USE_MISC
  8. #define TTYDEFCHARS
  9. #include <AK/Array.h>
  10. #include <AK/Optional.h>
  11. #include <AK/Result.h>
  12. #include <AK/ScopeGuard.h>
  13. #include <AK/String.h>
  14. #include <AK/StringView.h>
  15. #include <AK/Vector.h>
  16. #include <LibCore/System.h>
  17. #include <LibMain/Main.h>
  18. #include <ctype.h>
  19. #include <fcntl.h>
  20. #include <getopt.h>
  21. #include <stdio.h>
  22. #include <sys/ioctl.h>
  23. #include <sys/ttydefaults.h>
  24. #include <termios.h>
  25. #include <unistd.h>
  26. constexpr option long_options[] = {
  27. { "all", no_argument, 0, 'a' },
  28. { "save", no_argument, 0, 'g' },
  29. { "file", required_argument, 0, 'F' },
  30. { 0, 0, 0, 0 }
  31. };
  32. struct TermiosFlag {
  33. StringView name;
  34. tcflag_t value;
  35. tcflag_t mask;
  36. };
  37. struct BaudRate {
  38. speed_t speed;
  39. unsigned long numeric_value;
  40. };
  41. struct ControlCharacter {
  42. StringView name;
  43. unsigned index;
  44. };
  45. constexpr TermiosFlag all_iflags[] = {
  46. { "ignbrk", IGNBRK, IGNBRK },
  47. { "brkint", BRKINT, BRKINT },
  48. { "ignpar", IGNPAR, IGNPAR },
  49. { "parmer", PARMRK, PARMRK },
  50. { "inpck", INPCK, INPCK },
  51. { "istrip", ISTRIP, ISTRIP },
  52. { "inlcr", INLCR, INLCR },
  53. { "igncr", IGNCR, IGNCR },
  54. { "icrnl", ICRNL, ICRNL },
  55. { "iuclc", IUCLC, IUCLC },
  56. { "ixon", IXON, IXON },
  57. { "ixany", IXANY, IXANY },
  58. { "ixoff", IXOFF, IXOFF },
  59. { "imaxbel", IMAXBEL, IMAXBEL },
  60. { "iutf8", IUTF8, IUTF8 }
  61. };
  62. constexpr TermiosFlag all_oflags[] = {
  63. { "opost", OPOST, OPOST },
  64. { "olcuc", OLCUC, OPOST },
  65. { "onlcr", ONLCR, ONLCR },
  66. { "onlret", ONLRET, ONLRET },
  67. { "ofill", OFILL, OFILL },
  68. { "ofdel", OFDEL, OFDEL },
  69. };
  70. constexpr TermiosFlag all_cflags[] = {
  71. { "cs5", CS5, CSIZE },
  72. { "cs6", CS6, CSIZE },
  73. { "cs7", CS7, CSIZE },
  74. { "cs8", CS8, CSIZE },
  75. { "cstopb", CSTOPB, CSTOPB },
  76. { "cread", CREAD, CREAD },
  77. { "parenb", PARENB, PARENB },
  78. { "parodd", PARODD, PARODD },
  79. { "hupcl", HUPCL, HUPCL },
  80. { "clocal", CLOCAL, CLOCAL },
  81. };
  82. constexpr TermiosFlag all_lflags[] = {
  83. { "isig", ISIG, ISIG },
  84. { "icanon", ICANON, ICANON },
  85. { "echo", ECHO, ECHO },
  86. { "echoe", ECHOE, ECHOE },
  87. { "echok", ECHOK, ECHOK },
  88. { "echonl", ECHONL, ECHONL },
  89. { "noflsh", NOFLSH, NOFLSH },
  90. { "tostop", TOSTOP, TOSTOP },
  91. { "iexten", IEXTEN, IEXTEN }
  92. };
  93. constexpr BaudRate baud_rates[] = {
  94. { B0, 0 },
  95. { B50, 50 },
  96. { B75, 75 },
  97. { B110, 110 },
  98. { B134, 134 },
  99. { B150, 150 },
  100. { B200, 200 },
  101. { B300, 300 },
  102. { B600, 600 },
  103. { B1200, 1200 },
  104. { B1800, 1800 },
  105. { B2400, 2400 },
  106. { B4800, 4800 },
  107. { B9600, 9600 },
  108. { B19200, 19200 },
  109. { B38400, 38400 },
  110. { B57600, 57600 },
  111. { B115200, 115200 },
  112. { B230400, 230400 },
  113. { B460800, 460800 },
  114. { B500000, 500000 },
  115. { B576000, 576000 },
  116. { B921600, 921600 },
  117. { B1000000, 1000000 },
  118. { B1152000, 1152000 },
  119. { B1500000, 1500000 },
  120. { B2000000, 2000000 },
  121. { B2500000, 2500000 },
  122. { B3000000, 3000000 },
  123. { B3500000, 3500000 },
  124. { B4000000, 4000000 }
  125. };
  126. constexpr ControlCharacter control_characters[] = {
  127. { "intr", VINTR },
  128. { "quit", VQUIT },
  129. { "erase", VERASE },
  130. { "kill", VKILL },
  131. { "eof", VEOF },
  132. /* time and min are handled separately */
  133. { "swtc", VSWTC },
  134. { "start", VSTART },
  135. { "stop", VSTOP },
  136. { "susp", VSUSP },
  137. { "eol", VEOL },
  138. { "reprint", VREPRINT },
  139. { "discard", VDISCARD },
  140. { "werase", VWERASE },
  141. { "lnext", VLNEXT },
  142. { "eol2", VEOL2 }
  143. };
  144. Optional<speed_t> numeric_value_to_speed(unsigned long);
  145. Optional<unsigned long> speed_to_numeric_value(speed_t);
  146. void print_stty_readable(termios const&);
  147. void print_human_readable(termios const&, winsize const&, bool);
  148. Result<void, int> apply_stty_readable_modes(StringView, termios&);
  149. Result<void, int> apply_modes(size_t, char**, termios&, winsize&);
  150. Optional<speed_t> numeric_value_to_speed(unsigned long numeric_value)
  151. {
  152. for (auto rate : baud_rates) {
  153. if (rate.numeric_value == numeric_value)
  154. return rate.speed;
  155. }
  156. return {};
  157. }
  158. Optional<unsigned long> speed_to_numeric_value(speed_t speed)
  159. {
  160. for (auto rate : baud_rates) {
  161. if (rate.speed == speed)
  162. return rate.numeric_value;
  163. }
  164. return {};
  165. }
  166. void print_stty_readable(termios const& modes)
  167. {
  168. out("{:x}:{:x}:{:x}:{:x}", modes.c_iflag, modes.c_oflag, modes.c_cflag, modes.c_lflag);
  169. for (size_t i = 0; i < NCCS; ++i)
  170. out(":{:x}", modes.c_cc[i]);
  171. out(":{:x}:{:x}\n", modes.c_ispeed, modes.c_ospeed);
  172. }
  173. void print_human_readable(termios const& modes, winsize const& ws, bool verbose_mode)
  174. {
  175. auto print_speed = [&] {
  176. if (verbose_mode && modes.c_ispeed != modes.c_ospeed) {
  177. out("ispeed {} baud; ospeed {} baud;", speed_to_numeric_value(modes.c_ispeed).value(), speed_to_numeric_value(modes.c_ospeed).value());
  178. } else {
  179. out("speed {} baud;", speed_to_numeric_value(modes.c_ispeed).value());
  180. }
  181. };
  182. auto print_winsize = [&] {
  183. out("rows {}; columns {};", ws.ws_row, ws.ws_col);
  184. };
  185. auto escape_character = [&](u8 ch) {
  186. StringBuilder sb;
  187. if (ch <= 0x20) {
  188. sb.append("^");
  189. sb.append(ch + 0x40);
  190. } else if (ch == 0x7f) {
  191. sb.append("^?");
  192. } else {
  193. sb.append(ch);
  194. }
  195. return sb.to_string();
  196. };
  197. auto print_control_characters = [&] {
  198. bool first_in_line = true;
  199. for (auto cc : control_characters) {
  200. if (verbose_mode || modes.c_cc[cc.index] != ttydefchars[cc.index]) {
  201. out("{}{} = {};", (first_in_line) ? "" : " ", cc.name, escape_character(modes.c_cc[cc.index]));
  202. first_in_line = false;
  203. }
  204. }
  205. if (!first_in_line)
  206. out("\n");
  207. };
  208. auto print_flags_of_type = [&](const TermiosFlag flags[], size_t flag_count, tcflag_t field_value, tcflag_t field_default) {
  209. bool first_in_line = true;
  210. for (size_t i = 0; i < flag_count; ++i) {
  211. auto& flag = flags[i];
  212. if (verbose_mode || (field_value & flag.mask) != (field_default & flag.mask)) {
  213. bool set = (field_value & flag.mask) == flag.value;
  214. out("{}{}{}", first_in_line ? "" : " ", set ? "" : "-", flag.name);
  215. first_in_line = false;
  216. }
  217. }
  218. if (!first_in_line)
  219. out("\n");
  220. };
  221. auto print_flags = [&] {
  222. print_flags_of_type(all_cflags, sizeof(all_cflags) / sizeof(TermiosFlag), modes.c_cflag, TTYDEF_CFLAG);
  223. print_flags_of_type(all_oflags, sizeof(all_oflags) / sizeof(TermiosFlag), modes.c_oflag, TTYDEF_OFLAG);
  224. print_flags_of_type(all_iflags, sizeof(all_iflags) / sizeof(TermiosFlag), modes.c_iflag, TTYDEF_IFLAG);
  225. print_flags_of_type(all_lflags, sizeof(all_lflags) / sizeof(TermiosFlag), modes.c_lflag, TTYDEF_LFLAG);
  226. };
  227. print_speed();
  228. out(" ");
  229. print_winsize();
  230. out("\n");
  231. print_control_characters();
  232. print_flags();
  233. }
  234. Result<void, int> apply_stty_readable_modes(StringView mode_string, termios& t)
  235. {
  236. auto split = mode_string.split_view(':');
  237. if (split.size() != 4 + NCCS + 2) {
  238. warnln("Save string has an incorrect number of parameters");
  239. return 1;
  240. }
  241. auto parse_hex = [&](StringView v) {
  242. tcflag_t ret = 0;
  243. for (auto c : v) {
  244. c = tolower(c);
  245. ret *= 16;
  246. if (isdigit(c)) {
  247. ret += c - '0';
  248. } else {
  249. VERIFY(c >= 'a' && c <= 'f');
  250. ret += c - 'a';
  251. }
  252. }
  253. return ret;
  254. };
  255. t.c_iflag = parse_hex(split[0]);
  256. t.c_oflag = parse_hex(split[1]);
  257. t.c_cflag = parse_hex(split[2]);
  258. t.c_lflag = parse_hex(split[3]);
  259. for (size_t i = 0; i < NCCS; ++i) {
  260. t.c_cc[i] = (cc_t)parse_hex(split[4 + i]);
  261. }
  262. t.c_ispeed = parse_hex(split[4 + NCCS]);
  263. t.c_ospeed = parse_hex(split[4 + NCCS + 1]);
  264. return {};
  265. }
  266. Result<void, int> apply_modes(size_t parameter_count, char** raw_parameters, termios& t, winsize& w)
  267. {
  268. Vector<StringView> parameters;
  269. parameters.ensure_capacity(parameter_count);
  270. for (size_t i = 0; i < parameter_count; ++i)
  271. parameters.append(StringView { raw_parameters[i], strlen(raw_parameters[i]) });
  272. auto parse_baud = [&](size_t idx) -> Optional<speed_t> {
  273. auto maybe_numeric_value = parameters[idx].to_uint<uint32_t>();
  274. if (maybe_numeric_value.has_value())
  275. return numeric_value_to_speed(maybe_numeric_value.value());
  276. return {};
  277. };
  278. auto parse_number = [&](size_t idx) -> Optional<cc_t> {
  279. return parameters[idx].to_uint<cc_t>();
  280. };
  281. auto looks_like_stty_readable = [&](size_t idx) {
  282. bool contains_colon = false;
  283. for (auto c : parameters[idx]) {
  284. c = tolower(c);
  285. if (!isdigit(c) && !(c >= 'a' && c <= 'f') && c != ':')
  286. return false;
  287. if (c == ':')
  288. contains_colon = true;
  289. }
  290. return contains_colon;
  291. };
  292. auto parse_control_character = [&](size_t idx) -> Optional<cc_t> {
  293. VERIFY(!parameters[idx].is_empty());
  294. if (parameters[idx] == "^-" || parameters[idx] == "undef") {
  295. // FIXME: disabling characters is a bit wonky right now in TTY.
  296. // We should add the _POSIX_VDISABLE macro.
  297. return 0;
  298. } else if (parameters[idx][0] == '^' && parameters[idx].length() == 2) {
  299. return toupper(parameters[idx][1]) - 0x40;
  300. } else if (parameters[idx].starts_with("0x")) {
  301. cc_t value = 0;
  302. if (parameters[idx].length() == 2) {
  303. warnln("Invalid hexadecimal character code {}", parameters[idx]);
  304. return {};
  305. }
  306. for (size_t i = 2; i < parameters[idx].length(); ++i) {
  307. char ch = tolower(parameters[idx][i]);
  308. if (!isdigit(ch) && !(ch >= 'a' && ch <= 'f')) {
  309. warnln("Invalid hexadecimal character code {}", parameters[idx]);
  310. return {};
  311. }
  312. value = 16 * value + (isdigit(ch)) ? (ch - '0') : (ch - 'a');
  313. }
  314. return value;
  315. } else if (parameters[idx].starts_with("0")) {
  316. cc_t value = 0;
  317. for (size_t i = 1; i < parameters[idx].length(); ++i) {
  318. char ch = parameters[idx][i];
  319. if (!(ch >= '0' && ch <= '7')) {
  320. warnln("Invalid octal character code {}", parameters[idx]);
  321. return {};
  322. }
  323. value = 8 * value + (ch - '0');
  324. }
  325. return value;
  326. } else if (isdigit(parameters[idx][0])) {
  327. auto maybe_value = parameters[idx].to_uint<cc_t>();
  328. if (!maybe_value.has_value()) {
  329. warnln("Invalid decimal character code {}", parameters[idx]);
  330. return {};
  331. }
  332. return maybe_value.value();
  333. } else if (parameters[idx].length() == 1) {
  334. return parameters[idx][0];
  335. }
  336. warnln("Invalid control character {}", parameters[idx]);
  337. return {};
  338. };
  339. size_t parameter_idx = 0;
  340. auto parse_flag_or_char = [&]() -> Result<void, int> {
  341. if (parameters[parameter_idx][0] != '-') {
  342. if (parameters[parameter_idx] == "min") {
  343. auto maybe_number = parse_number(++parameter_idx);
  344. if (!maybe_number.has_value()) {
  345. warnln("Error parsing min: {} is not a number", parameters[parameter_idx]);
  346. return 1;
  347. }
  348. return {};
  349. } else if (parameters[parameter_idx] == "time") {
  350. auto maybe_number = parse_number(++parameter_idx);
  351. if (!maybe_number.has_value()) {
  352. warnln("Error parsing time: {} is not a number", parameters[parameter_idx]);
  353. return 1;
  354. }
  355. return {};
  356. } else {
  357. for (auto cc : control_characters) {
  358. if (cc.name == parameters[parameter_idx]) {
  359. if (parameter_idx == parameter_count - 1) {
  360. warnln("No control character specified for {}", cc.name);
  361. return 1;
  362. }
  363. auto maybe_control_character = parse_control_character(++parameter_idx);
  364. if (!maybe_control_character.has_value())
  365. return 1;
  366. t.c_cc[cc.index] = maybe_control_character.value();
  367. return {};
  368. }
  369. }
  370. }
  371. }
  372. // We fall through to here if what we are setting is not a control character.
  373. bool negate = false;
  374. if (parameters[parameter_idx][0] == '-') {
  375. negate = true;
  376. parameters[parameter_idx] = parameters[parameter_idx].substring_view(1);
  377. }
  378. auto perform_masking = [&](tcflag_t value, tcflag_t mask, tcflag_t& dest) {
  379. if (negate)
  380. dest &= ~mask;
  381. else
  382. dest = (dest & (~mask)) | value;
  383. };
  384. for (auto flag : all_iflags) {
  385. if (flag.name == parameters[parameter_idx]) {
  386. perform_masking(flag.value, flag.mask, t.c_iflag);
  387. return {};
  388. }
  389. }
  390. for (auto flag : all_oflags) {
  391. if (flag.name == parameters[parameter_idx]) {
  392. perform_masking(flag.value, flag.mask, t.c_oflag);
  393. return {};
  394. }
  395. }
  396. for (auto flag : all_cflags) {
  397. if (flag.name == parameters[parameter_idx]) {
  398. perform_masking(flag.value, flag.mask, t.c_cflag);
  399. return {};
  400. }
  401. }
  402. for (auto flag : all_lflags) {
  403. if (flag.name == parameters[parameter_idx]) {
  404. perform_masking(flag.value, flag.mask, t.c_lflag);
  405. return {};
  406. }
  407. }
  408. warnln("Invalid control flag or control character name {}", parameters[parameter_idx]);
  409. return 1;
  410. };
  411. while (parameter_idx < parameter_count) {
  412. if (looks_like_stty_readable(parameter_idx)) {
  413. auto maybe_error = apply_stty_readable_modes(parameters[parameter_idx], t);
  414. if (maybe_error.is_error())
  415. return maybe_error.error();
  416. } else if (isdigit(parameters[parameter_idx][0])) {
  417. auto new_baud = parse_baud(parameter_idx);
  418. if (!new_baud.has_value()) {
  419. warnln("Invalid baud rate {}", parameters[parameter_idx]);
  420. return 1;
  421. }
  422. t.c_ispeed = t.c_ospeed = new_baud.value();
  423. } else if (parameters[parameter_idx] == "ispeed") {
  424. if (parameter_idx == parameter_count - 1) {
  425. warnln("No baud rate specified for ispeed");
  426. return 1;
  427. }
  428. auto new_baud = parse_baud(++parameter_idx);
  429. if (!new_baud.has_value()) {
  430. warnln("Invalid input baud rate {}", parameters[parameter_idx]);
  431. return 1;
  432. }
  433. t.c_ispeed = new_baud.value();
  434. } else if (parameters[parameter_idx] == "ospeed") {
  435. if (parameter_idx == parameter_count - 1) {
  436. warnln("No baud rate specified for ospeed");
  437. return 1;
  438. }
  439. auto new_baud = parse_baud(++parameter_idx);
  440. if (!new_baud.has_value()) {
  441. warnln("Invalid output baud rate {}", parameters[parameter_idx]);
  442. return 1;
  443. }
  444. t.c_ospeed = new_baud.value();
  445. } else if (parameters[parameter_idx] == "columns" || parameters[parameter_idx] == "cols") {
  446. auto maybe_number = parse_number(++parameter_idx);
  447. if (!maybe_number.has_value()) {
  448. warnln("Invalid column count {}", parameters[parameter_idx]);
  449. return 1;
  450. }
  451. w.ws_col = maybe_number.value();
  452. } else if (parameters[parameter_idx] == "rows") {
  453. auto maybe_number = parse_number(++parameter_idx);
  454. if (!maybe_number.has_value()) {
  455. warnln("Invalid row count {}", parameters[parameter_idx]);
  456. return 1;
  457. }
  458. w.ws_row = maybe_number.value();
  459. } else if (parameters[parameter_idx] == "evenp" || parameters[parameter_idx] == "parity") {
  460. t.c_cflag &= ~(CSIZE | PARODD);
  461. t.c_cflag |= CS7 | PARENB;
  462. } else if (parameters[parameter_idx] == "oddp") {
  463. t.c_cflag &= ~(CSIZE);
  464. t.c_cflag |= CS7 | PARENB | PARODD;
  465. } else if (parameters[parameter_idx] == "-parity" || parameters[parameter_idx] == "-evenp" || parameters[parameter_idx] == "-oddp") {
  466. t.c_cflag &= ~(PARENB | CSIZE);
  467. t.c_cflag |= CS8;
  468. } else if (parameters[parameter_idx] == "raw") {
  469. cfmakeraw(&t);
  470. } else if (parameters[parameter_idx] == "nl") {
  471. t.c_iflag &= ~ICRNL;
  472. } else if (parameters[parameter_idx] == "-nl") {
  473. t.c_cflag &= ~(INLCR & IGNCR);
  474. t.c_iflag |= ICRNL;
  475. } else if (parameters[parameter_idx] == "ek") {
  476. t.c_cc[VERASE] = CERASE;
  477. t.c_cc[VKILL] = CKILL;
  478. } else if (parameters[parameter_idx] == "sane") {
  479. t.c_iflag = TTYDEF_IFLAG;
  480. t.c_oflag = TTYDEF_OFLAG;
  481. t.c_cflag = TTYDEF_CFLAG;
  482. t.c_lflag = TTYDEF_LFLAG;
  483. for (size_t i = 0; i < NCCS; ++i)
  484. t.c_cc[i] = ttydefchars[i];
  485. t.c_ispeed = t.c_ospeed = TTYDEF_SPEED;
  486. } else {
  487. auto maybe_error = parse_flag_or_char();
  488. if (maybe_error.is_error())
  489. return maybe_error.error();
  490. }
  491. ++parameter_idx;
  492. }
  493. return {};
  494. }
  495. ErrorOr<int> serenity_main(Main::Arguments arguments)
  496. {
  497. TRY(Core::System::pledge("stdio tty rpath"));
  498. TRY(Core::System::unveil("/dev", "r"));
  499. TRY(Core::System::unveil(nullptr, nullptr));
  500. String device_file;
  501. bool stty_readable = false;
  502. bool all_settings = false;
  503. // Core::ArgsParser can't handle the weird syntax of stty, so we use getopt_long instead.
  504. int argc = arguments.argc;
  505. char** argv = arguments.argv;
  506. opterr = 0; // We handle unknown flags gracefully by starting to parse the arguments in `apply_modes`.
  507. int optc;
  508. bool should_quit = false;
  509. while (!should_quit && ((optc = getopt_long(argc, argv, "-agF:", long_options, nullptr)) != -1)) {
  510. switch (optc) {
  511. case 'a':
  512. all_settings = true;
  513. break;
  514. case 'g':
  515. stty_readable = true;
  516. break;
  517. case 'F':
  518. if (!device_file.is_empty()) {
  519. warnln("Only one device may be specified");
  520. exit(1);
  521. }
  522. device_file = optarg;
  523. break;
  524. default:
  525. should_quit = true;
  526. break;
  527. }
  528. }
  529. if (stty_readable && all_settings) {
  530. warnln("Save mode and all-settings mode are mutually exclusive");
  531. exit(1);
  532. }
  533. int terminal_fd = STDIN_FILENO;
  534. if (!device_file.is_empty()) {
  535. if ((terminal_fd = open(device_file.characters(), O_RDONLY, 0)) < 0) {
  536. perror("open");
  537. exit(1);
  538. }
  539. }
  540. ScopeGuard file_close_guard = [&] { close(terminal_fd); };
  541. termios initial_termios = TRY(Core::System::tcgetattr(terminal_fd));
  542. winsize initial_winsize;
  543. TRY(Core::System::ioctl(terminal_fd, TIOCGWINSZ, &initial_winsize));
  544. if (optind < argc) {
  545. if (stty_readable || all_settings) {
  546. warnln("Modes cannot be set when printing settings");
  547. exit(1);
  548. }
  549. auto result = apply_modes(argc - optind, argv + optind, initial_termios, initial_winsize);
  550. if (result.is_error())
  551. return result.error();
  552. TRY(Core::System::tcsetattr(terminal_fd, TCSADRAIN, initial_termios));
  553. TRY(Core::System::ioctl(terminal_fd, TIOCSWINSZ, &initial_winsize));
  554. } else if (stty_readable) {
  555. print_stty_readable(initial_termios);
  556. } else {
  557. print_human_readable(initial_termios, initial_winsize, all_settings);
  558. }
  559. return 0;
  560. }