-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsolutions_module.h
240 lines (220 loc) · 5.72 KB
/
solutions_module.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
#ifndef _AOC_Y2022_SOLUTIONS_H_INCLUDED
#define _AOC_Y2022_SOLUTIONS_H_INCLUDED
#include <stdio.h>
#include <time.h>
#include "common.h"
#include "d01.h"
#include "d02.h"
#include "d03.h"
#include "d04.h"
#include "d05.h"
#include "d06.h"
#include "d07.h"
#include "d08.h"
#include "d09.h"
#include "d10.h"
#include "d11.h"
#include "d12.h"
#include "d13.h" // NOTE: defines a custom PyTypeObject
#include "d14.h"
#include "d15.h"
#include "d16.h"
#include "d17.h"
#include "d18.h"
#include "d19.h"
#include "d20.h"
#include "d21.h"
#include "d22.h"
#include "d23.h"
#include "d24.h"
#include "d25.h"
static PyObject *_AoC_solve_y2022(int day, PyObject *unicode_input) {
switch (day) {
case 1:
return AoC_y2022_d01(unicode_input);
case 2:
return AoC_y2022_d02(unicode_input);
case 3:
return AoC_y2022_d03(unicode_input);
case 4:
return AoC_y2022_d04(unicode_input);
case 5:
return AoC_y2022_d05(unicode_input);
case 6:
return AoC_y2022_d06(unicode_input);
case 7:
return AoC_y2022_d07(unicode_input);
case 8:
return AoC_y2022_d08(unicode_input);
case 9:
return AoC_y2022_d09(unicode_input);
case 10:
return AoC_y2022_d10(unicode_input);
case 11:
return AoC_y2022_d11(unicode_input);
case 12:
return AoC_y2022_d12(unicode_input);
case 13:
return AoC_y2022_d13(unicode_input);
case 14:
return AoC_y2022_d14(unicode_input);
case 15:
return AoC_y2022_d15(unicode_input);
case 16:
return AoC_y2022_d16(unicode_input);
case 17:
return AoC_y2022_d17(unicode_input);
case 18:
return AoC_y2022_d18(unicode_input);
case 19:
return AoC_y2022_d19(unicode_input);
case 20:
return AoC_y2022_d20(unicode_input);
case 21:
return AoC_y2022_d21(unicode_input);
case 22:
return AoC_y2022_d22(unicode_input);
case 23:
return AoC_y2022_d23(unicode_input);
case 24:
return AoC_y2022_d24(unicode_input);
case 25:
return AoC_y2022_d25(unicode_input);
}
return PyErr_Format(PyExc_ValueError,
"no solution implemented for day %d",
day);
}
static PyObject *AoC_solve_y2022(PyObject *module, PyObject *args) {
int day;
if (!PyArg_ParseTuple(args, "i", &day)) {
PyErr_SetString(PyExc_RuntimeError,
"failed parsing positional arg 'day' as int");
goto error;
}
if (!(1 <= day && day <= 25)) {
PyErr_Format(PyExc_ValueError,
"positional arg 'day' must be in range [1, 25], not %d",
day);
goto error;
}
PyObject *input_path = PyUnicode_FromFormat("./txt/input/2022/%.2d", day);
if (!input_path) {
goto error;
}
PyObject *unicode_input = AoC_slurp_file(input_path);
Py_DECREF(input_path);
if (!unicode_input) {
goto error;
}
PyObject *solution = 0;
{
const clock_t start_time = clock();
solution = _AoC_solve_y2022(day, unicode_input);
const clock_t end_time = clock();
const double solve_msec =
1e3 * ((double)(end_time - start_time)) / CLOCKS_PER_SEC;
fprintf(stderr, "%.1f ms\n", solve_msec);
}
Py_DECREF(unicode_input);
if (!solution) {
goto error;
}
return solution;
error:
if (!PyErr_Occurred()) {
PyErr_Format(PyExc_RuntimeError,
"unknown failure when solving day %d",
day);
}
return 0;
}
// Methods available in module 'aoc_solve'
static PyMethodDef AoC_solve_methods[] = {
{
"y2022",
(PyCFunction)AoC_solve_y2022,
METH_VARARGS,
"Advent of Code 2022 solver",
},
// Custom modules end
{
0,
0,
0,
0,
},
};
// Definition of module 'aoc_solve'
static struct PyModuleDef AoC_solve_module = {
PyModuleDef_HEAD_INIT,
"aoc_solve",
"Advent of Code solvers",
-1,
AoC_solve_methods,
};
PyMODINIT_FUNC PyInit_AoC_solve(void) {
PyObject *module = 0;
if (PyType_Ready(&AoC_y2022_d13_PacketType) < 0) {
return 0;
}
module = PyModule_Create(&AoC_solve_module);
if (!module) {
return 0;
}
Py_INCREF(&AoC_y2022_d13_PacketType);
if (PyModule_AddObject(module,
AoC_y2022_d13_PacketType.tp_name,
(PyObject *)&AoC_y2022_d13_PacketType) < 0) {
Py_DECREF(&AoC_y2022_d13_PacketType);
Py_DECREF(module);
return 0;
}
return module;
}
PyStatus _AoC_init_python(const int argc, char *const *argv, const int debug) {
/*
* https://docs.python.org/3/c-api/init_config.html#initialization-with-pyconfig
*/
PyConfig config;
PyConfig_InitPythonConfig(&config);
if (debug) {
config.optimization_level = 0;
config.bytes_warning = 2;
config.tracemalloc = 1;
config.malloc_stats = 1;
config.show_ref_count = 1;
config.dump_refs = 1;
} else {
config.optimization_level = 2;
config.bytes_warning = 2;
}
PyStatus status = PyConfig_SetBytesArgv(&config, argc, argv);
if (PyStatus_Exception(status)) {
goto done;
}
status = PyConfig_Read(&config);
if (PyStatus_Exception(status)) {
goto done;
}
status = Py_InitializeFromConfig(&config);
done:
PyConfig_Clear(&config);
return status;
}
int main(int argc, char *const *argv) {
// Make 'aoc_solve' available as a builtin module
if (PyImport_AppendInittab(AoC_solve_module.m_name, PyInit_AoC_solve) == -1) {
PyErr_Print();
fprintf(stderr,
"Failed to extend builtin modules table with '%s'\n",
AoC_solve_module.m_name);
return EXIT_FAILURE;
}
PyStatus status = _AoC_init_python(argc, argv, _AOC_Y2022_DEBUG_MAIN);
if (PyStatus_Exception(status)) {
Py_ExitStatusException(status);
}
return Py_RunMain();
}
#endif // _AOC_Y2022_SOLUTIONS_H_INCLUDED