io_utils.carbon 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. library "io_utils";
  5. import Core library "io";
  6. import Core library "range";
  7. class EOFType {}
  8. class CharOrEOF {
  9. adapt i32;
  10. fn EOF() -> Self;
  11. }
  12. impl CharOrEOF as Core.Copy {
  13. fn Op[self: Self]() -> Self {
  14. return (self as i32).(Core.Copy.Op)() as Self;
  15. }
  16. }
  17. fn CharOrEOF.EOF() -> Self {
  18. // Ordering between `CharOrEOF` and `char` treats EOF as less than all `char`
  19. // values.
  20. return (-1 as i32) as CharOrEOF;
  21. }
  22. impl forall [U:! Core.ImplicitAs(char)] U as Core.ImplicitAs(CharOrEOF) {
  23. fn Convert[self: char]() -> CharOrEOF {
  24. return ((self as u8) as i32) as CharOrEOF;
  25. }
  26. }
  27. impl forall [U:! Core.ImplicitAs(CharOrEOF)]
  28. CharOrEOF as Core.EqWith(U) {
  29. fn Equal[self: Self](other: CharOrEOF) -> bool {
  30. return (self as i32) == (other as i32);
  31. }
  32. fn NotEqual[self: Self](other: CharOrEOF) -> bool {
  33. return (self as i32) != (other as i32);
  34. }
  35. }
  36. impl forall [U:! Core.ImplicitAs(CharOrEOF)]
  37. CharOrEOF as Core.OrderedWith(U) {
  38. fn Less[self: Self](other: CharOrEOF) -> bool {
  39. return (self as i32) < (other as i32);
  40. }
  41. fn LessOrEquivalent[self: Self](other: CharOrEOF) -> bool {
  42. return (self as i32) <= (other as i32);
  43. }
  44. fn Greater[self: Self](other: CharOrEOF) -> bool {
  45. return (self as i32) > (other as i32);
  46. }
  47. fn GreaterOrEquivalent[self: Self](other: CharOrEOF) -> bool {
  48. return (self as i32) >= (other as i32);
  49. }
  50. }
  51. impl forall [U:! Core.ImplicitAs(char)]
  52. CharOrEOF as Core.SubWith(U) where .Result = i32 {
  53. fn Op[self: Self](other: char) -> i32 {
  54. return (self as i32) - ((other as u8) as i32);
  55. }
  56. }
  57. var unread_char: Core.Optional(CharOrEOF)
  58. = Core.Optional(CharOrEOF).None();
  59. fn ReadChar() -> CharOrEOF {
  60. if (unread_char.HasValue()) {
  61. var result: CharOrEOF = unread_char.Get();
  62. unread_char = Core.Optional(CharOrEOF).None();
  63. return result;
  64. }
  65. var value: i32 = Core.ReadChar();
  66. if (value == Core.EOF()) {
  67. return CharOrEOF.EOF();
  68. } else {
  69. return value as CharOrEOF;
  70. }
  71. }
  72. fn UnreadChar(c: CharOrEOF) {
  73. // TODO: assert(unread_char == 0);
  74. // TODO: unread_char = c;
  75. unread_char = Core.Optional(CharOrEOF).Some(c);
  76. }
  77. fn PeekChar() -> CharOrEOF {
  78. var next: CharOrEOF = ReadChar();
  79. UnreadChar(next);
  80. return next;
  81. }
  82. fn ConsumeChar(c: char) -> bool {
  83. var next: CharOrEOF = ReadChar();
  84. if (next != c) {
  85. UnreadChar(next);
  86. return false;
  87. }
  88. return true;
  89. }
  90. fn ReadInt[N:! Core.IntLiteral()](ref p: Core.Int(N)) -> bool {
  91. var read_any_digits: bool = false;
  92. let negative: bool = ConsumeChar('-');
  93. // TODO: `p = 0;` crashes the toolchain, see
  94. // https://github.com/carbon-language/carbon-lang/issues/6500.
  95. p = (0 as i32) as Core.Int(N);
  96. while (true) {
  97. var c: CharOrEOF = ReadChar();
  98. if (c < '0' or c > '9') {
  99. UnreadChar(c);
  100. break;
  101. }
  102. // TODO: Check for overflow.
  103. // TODO: p *= 10; crashes the toolchain.
  104. p *= (10 as i32) as Core.Int(N);
  105. p += (c - '0') as Core.Int(N);
  106. read_any_digits = true;
  107. }
  108. if (negative) {
  109. p = -p;
  110. if (not read_any_digits) {
  111. UnreadChar('-');
  112. }
  113. }
  114. return read_any_digits;
  115. }
  116. fn PrintIntNoNewline[N:! Core.IntLiteral()](n_val: Core.Int(N)) {
  117. // TODO: var pow10: Core.Int(N) = 1; crashes the toolchain.
  118. var pow10: Core.Int(N) = (1 as i32) as Core.Int(N);
  119. var n: Core.Int(N) = n_val;
  120. // TODO: let ten: Core.Int(N) = 10;
  121. let ten: Core.Int(N) = (10 as i32) as Core.Int(N);
  122. while (n / ten >= pow10) {
  123. pow10 = pow10 * ten;
  124. }
  125. // TODO: while (pow10 != 0) {
  126. while (pow10 != ((0 as i32) as Core.Int(N))) {
  127. let d: Core.Int(N) = n / pow10;
  128. // TODO: Core.PrintChar('0' + d);
  129. Core.PrintChar(((d as u8) + (('0' as char) as u8)) as char);
  130. n = n % pow10;
  131. pow10 = pow10 / ten;
  132. }
  133. }
  134. fn PrintInt[N:! Core.IntLiteral()](n_val: Core.Int(N)) {
  135. PrintIntNoNewline(n_val);
  136. Core.PrintChar('\n');
  137. }
  138. fn SkipSpaces() -> bool {
  139. var skipped_any_spaces: bool = false;
  140. while (ConsumeChar(' ')) {
  141. skipped_any_spaces = true;
  142. }
  143. return skipped_any_spaces;
  144. }
  145. fn SkipNewline() -> bool {
  146. // Optional carriage return.
  147. ConsumeChar('\r');
  148. // Newline.
  149. // TODO: Unread the CR if it was present?
  150. return ConsumeChar('\n');
  151. }
  152. fn SkipNChars(n: i32) -> bool {
  153. for (_: i32 in Core.Range(n)) {
  154. if (ReadChar() == CharOrEOF.EOF()) {
  155. return false;
  156. }
  157. }
  158. return true;
  159. }