Derive-C
Loading...
Searching...
No Matches
employees.c
Go to the documentation of this file.
1
6
7#include <stdio.h>
8#include <stdlib.h>
9#include <string.h>
10
11#include <derive-c/alloc/std.h>
12#include <derive-c/prelude.h>
13#include <derive-c/utils/for.h>
14
15typedef struct {
16 char const* forename;
17 char const* surname;
18} name;
19
20static bool name_eq(const name* name_1, const name* name_2) {
21 if (!name_1 || !name_2)
22 return false;
23 if (!name_1->forename || !name_2->forename)
24 return false;
25 if (!name_1->surname || !name_2->surname)
26 return false;
27
28 return strcmp(name_1->forename, name_2->forename) == 0 &&
29 strcmp(name_1->surname, name_2->surname) == 0;
30}
31
32static void name_debug(const name* self, dc_debug_fmt fmt, FILE* stream) {
33 (void)fmt;
34 fprintf(stream, "name@%p { forename: \"%s\", surname: \"%s\" }", (void*)self, self->forename,
35 self->surname);
36}
37
38typedef struct {
39 size_t value;
40} age;
41
42static bool age_eq(age const* age_1, age const* age_2) { return age_1->value == age_2->value; }
43static size_t age_hash(age const* age) { return age->value; }
44static void age_debug(age const* self, dc_debug_fmt fmt, FILE* stream) {
45 (void)fmt;
46 fprintf(stream, "%zu years", self->value);
47}
48
49typedef struct {
51 char const* email;
53} employee;
54
55static void employee_debug(employee const* self, dc_debug_fmt fmt, FILE* stream) {
56 fprintf(stream, "employee@%p {\n", (void*)self);
57 fmt = dc_debug_fmt_scope_begin(fmt);
58
59 dc_debug_fmt_print(fmt, stream, "name: ");
60 name_debug(&self->name, fmt, stream);
61 fprintf(stream, ",\n");
62
63 dc_debug_fmt_print(fmt, stream, "email: \"%s\",\n", self->email);
64
65 dc_debug_fmt_print(fmt, stream, "age: ");
66 age_debug(&self->age, fmt, stream);
67 fprintf(stream, ",\n");
68
69 fmt = dc_debug_fmt_scope_end(fmt);
70 dc_debug_fmt_print(fmt, stream, "}");
71}
72
73#define INDEX_BITS 16
74#define VALUE employee
75#define VALUE_DEBUG employee_debug
76#define NAME employees
78
79#define ITEM employees_index_t
80#define ITEM_DEBUG employees_index_t_debug
81#define NAME same_age_employees
83
84#define KEY age
85#define KEY_EQ age_eq
86#define KEY_HASH age_hash
87#define KEY_DEBUG age_debug
88#define VALUE same_age_employees
89#define VALUE_DEBUG same_age_employees_debug
90#define NAME employees_by_age
92
93typedef struct {
94 employees data;
95 employees_by_age by_age;
96} hr_system;
97
99 return (hr_system){
100 .data = employees_new_with_capacity_for(1000, stdalloc_get_ref()),
101 .by_age = employees_by_age_new(stdalloc_get_ref()),
102 };
103}
104
106 printf("Adding employee %s %s\n", emp.name.forename, emp.name.surname);
107 employees_index_t idx = employees_insert(&self->data, emp);
108 same_age_employees* idxes = employees_by_age_try_write(&self->by_age, emp.age);
109 if (!idxes) {
110 idxes = employees_by_age_insert(&self->by_age, emp.age,
111 same_age_employees_new(stdalloc_get_ref()));
112 }
113 same_age_employees_push(idxes, idx);
114}
115
116static employee const* hr_system_newest_of_age(hr_system const* self, age age) {
117 same_age_employees const* idxes = employees_by_age_try_read(&self->by_age, age);
118 if (!idxes) {
119 return NULL;
120 }
121 if (same_age_employees_size(idxes) == 0) {
122 return NULL;
123 }
124 employees_index_t const* idx =
125 same_age_employees_read(idxes, same_age_employees_size(idxes) - 1);
126 return employees_read(&self->data, *idx);
127}
128
129static void hr_system_debug(hr_system const* self, dc_debug_fmt fmt, FILE* stream) {
130 fprintf(stream, "hr_system@%p {\n", (void*)self);
131 fmt = dc_debug_fmt_scope_begin(fmt);
132
133 dc_debug_fmt_print(fmt, stream, "data: ");
134 employees_debug(&self->data, fmt, stream);
135 fprintf(stream, ",\n");
136
137 dc_debug_fmt_print(fmt, stream, "by_age: ");
138 employees_by_age_debug(&self->by_age, fmt, stream);
139 fprintf(stream, ",\n");
140
141 fmt = dc_debug_fmt_scope_end(fmt);
142 dc_debug_fmt_print(fmt, stream, "}");
143}
144
145static void hr_system_delete(hr_system* self) {
146 employees_delete(&self->data);
147
148 DC_FOR(employees_by_age, &self->by_age, iter, entry) { same_age_employees_delete(entry.value); }
149
150 employees_by_age_delete(&self->by_age);
151}
152
153int main() {
155
156 employee frank = {
157 .age = (age){.value = 22},
158 .email = "veryverylongemail@someprovider.net",
159 .name =
160 (name){
161 .forename = "Frank",
162 .surname = "Lee",
163 },
164 };
165 hr_system_new_employee(&hr, frank);
166
167 name bob_name = {
168 .forename = "Bob",
169 .surname = "Mike",
170 };
171 employee bob = {
172 .age = (age){.value = 22},
173 .email = "bib@cool.org",
174 .name = bob_name,
175 };
176 hr_system_new_employee(&hr, bob);
177
178 employee const* newest_22 = hr_system_newest_of_age(&hr, (age){.value = 22});
179 DC_ASSERT(newest_22);
180 DC_ASSERT(name_eq(&newest_22->name, &bob_name));
181
182 hr_system_debug(&hr, dc_debug_fmt_new(), stdout);
183
184 hr_system_delete(&hr);
185}
static void name_debug(const name *self, dc_debug_fmt fmt, FILE *stream)
Definition employees.c:32
static void age_debug(age const *self, dc_debug_fmt fmt, FILE *stream)
Definition employees.c:44
static void hr_system_new_employee(hr_system *self, employee emp)
Definition employees.c:105
static bool age_eq(age const *age_1, age const *age_2)
Definition employees.c:42
static bool name_eq(const name *name_1, const name *name_2)
Definition employees.c:20
static void hr_system_delete(hr_system *self)
Definition employees.c:145
static employee const * hr_system_newest_of_age(hr_system const *self, age age)
Definition employees.c:116
static void hr_system_debug(hr_system const *self, dc_debug_fmt fmt, FILE *stream)
Definition employees.c:129
static size_t age_hash(age const *age)
Definition employees.c:43
static hr_system hr_system_new()
Definition employees.c:98
int main()
Definition employees.c:153
static void employee_debug(employee const *self, dc_debug_fmt fmt, FILE *stream)
Definition employees.c:55
static DC_PUBLIC void dc_debug_fmt_print(dc_debug_fmt fmt, FILE *stream, const char *format,...)
Definition fmt.h:32
static DC_PUBLIC dc_debug_fmt dc_debug_fmt_new()
Definition fmt.h:15
static DC_PUBLIC dc_debug_fmt dc_debug_fmt_scope_end(dc_debug_fmt fmt)
Definition fmt.h:57
static DC_PUBLIC dc_debug_fmt dc_debug_fmt_scope_begin(dc_debug_fmt fmt)
Definition fmt.h:50
#define DC_FOR(TYPE, INSTANCE, ITER, ITEM)
Definition for.h:13
#define DC_ASSERT(expr,...)
Definition panic.h:37
size_t value
Definition employees.c:39
Debug format helpers for debug printin data structures.
Definition fmt.h:11
char const * email
Definition employees.c:51
age age
Definition employees.c:52
name name
Definition employees.c:50
employees data
Definition employees.c:94
employees_by_age by_age
Definition employees.c:95
char const * forename
Definition employees.c:16
char const * surname
Definition employees.c:17
static DC_PUBLIC FILE * stream(SELF *self)
Definition template.h:108