day5_common.carbon 2.6 KB

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