From 7d4cb5bb96b061ed8708889b75e4d50757d9b3f2 Mon Sep 17 00:00:00 2001 From: lemon Date: Sat, 13 Aug 2022 10:38:27 +0200 Subject: set template --- src/set.hff | 46 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 46 insertions(+) create mode 100644 src/set.hff (limited to 'src/set.hff') diff --git a/src/set.hff b/src/set.hff new file mode 100644 index 0000000..acff95b --- /dev/null +++ b/src/set.hff @@ -0,0 +1,46 @@ +import "vec.hff"; +import "all.hff"; + +struct Set { + buf Vec, + set **const T, + N int, + count int, + + fn intern(self *Set, x T) *const T { + if self.set == #null { + self.set = xcalloc(self.N = 16, sizeof *T); + } + + if self.count == self.N / 2 { + free(self.set); + self.set = xcalloc(self.N *= 2, sizeof *T); + vec_each(p, i, self.buf, + let i = Traits:hash(p) & (self.N - 1); + for ;; { + if self.set[i] == #null { + self.set[i] = &self.buf.dat[i]; + break; + } + i = (i + 1) & (self.N - 1); + } + ) + } + + let i0 = Traits:hash(x) & (self.N - 1); + let i int = i0; + do { + if self.set[i] == #null { + ++self.count; + self.buf->push(Traits:dup(x)); + return self.set[i] = &self.buf.dat[self.buf.len - 1]; + } else if Traits:eq(*self.set[i], x) { + fprintf(stderr, "found %s\n", x); + return self.set[i]; + } + i = (i + 1) & (self.N - 1); + } while i != i0; + assert(#f, "unreachable"); + } + +} -- cgit v1.2.3