day5_common.carbon 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  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. // https://adventofcode.com/2024/day/5
  5. library "day5_common";
  6. import Core library "range";
  7. import library "io_utils";
  8. fn PageMask(page: i32) -> Core.UInt(100) {
  9. // TODO: return (1 as Core.UInt(100)) << page;
  10. return (1 as Core.UInt(100)) << (page as Core.UInt(100));
  11. }
  12. class Rules {
  13. fn Read() -> Rules {
  14. returned var rules: Rules;
  15. for (i: i32 in Core.Range(100)) {
  16. rules.disallowed_before[i] = 0;
  17. }
  18. var a: i32;
  19. var b: i32;
  20. while (ReadInt(&a) and ConsumeChar(0x7C) and ReadInt(&b)) {
  21. rules.disallowed_before[a] |= PageMask(b);
  22. SkipNewline();
  23. }
  24. return var;
  25. }
  26. fn IsValidOrder[self: Self](a: i32, b: i32) -> bool {
  27. return self.disallowed_before[b] & PageMask(a) == 0;
  28. }
  29. var disallowed_before: array(Core.UInt(100), 100);
  30. };
  31. class PageList {
  32. fn Empty() -> PageList {
  33. returned var me: PageList;
  34. me.num_pages = 0;
  35. return var;
  36. }
  37. fn Read() -> PageList {
  38. returned var me: PageList = Empty();
  39. var page: i32;
  40. if (not ReadInt(&page)) {
  41. return var;
  42. }
  43. me.Add(page);
  44. while (ConsumeChar(0x2C)) {
  45. ReadInt(&page);
  46. me.Add(page);
  47. }
  48. SkipNewline();
  49. return var;
  50. }
  51. fn Add[addr self: Self*](page: i32) {
  52. self->pages[self->num_pages] = page;
  53. ++self->num_pages;
  54. }
  55. fn FollowsRules[self: Self](rules: Rules) -> bool {
  56. var seen: Core.UInt(100) = 0;
  57. for (i: i32 in Core.Range(self.num_pages)) {
  58. let page: i32 = self.pages[i];
  59. if (seen & rules.disallowed_before[page] != 0) {
  60. return false;
  61. }
  62. seen |= PageMask(page);
  63. }
  64. return true;
  65. }
  66. fn IsPossibleFirstPage[self: Self](rules: Rules, page: i32) -> bool {
  67. for (i: i32 in Core.Range(self.num_pages)) {
  68. if (not rules.IsValidOrder(page, self.pages[i])) {
  69. return false;
  70. }
  71. }
  72. return true;
  73. }
  74. fn ExtractPossibleFirstPage[addr self: Self*](rules: Rules) -> i32 {
  75. for (i: i32 in Core.Range(self->num_pages)) {
  76. var page: i32 = self->pages[i];
  77. if (self->IsPossibleFirstPage(rules, page)) {
  78. self->pages[i] = self->pages[self->num_pages - 1];
  79. --self->num_pages;
  80. return page;
  81. }
  82. }
  83. // TODO: Assert.
  84. return 0;
  85. }
  86. fn MiddlePage[self: Self]() -> i32 {
  87. return self.pages[self.num_pages / 2];
  88. }
  89. var pages: array(i32, 24);
  90. var num_pages: i32;
  91. };