On Fri, 14 Feb 2020 19:16:34 +0100 Pablo Neira Ayuso <pablo@xxxxxxxxxxxxx> wrote: > On Mon, Feb 10, 2020 at 04:10:47PM +0100, Stefano Brivio wrote: > > On Fri, 7 Feb 2020 12:23:08 +0100 > > Pablo Neira Ayuso <pablo@xxxxxxxxxxxxx> wrote: > [...] > > > Did you tests with 8-bits grouping instead of 4-bits? > > > > Yes, at the very beginning, not with the final implementation. It was > > somewhat faster (on x86_64, I don't remember how much) for small > > numbers of rules, then I thought we would use too much memory, because: > > > > > Assuming a concatenation of 12 bytes (each field 4 bytes, hence 3 > > > fields): > > > > > > * Using 4-bits groups: the number of buckets is 2^4 = 16 multiplied > > > by the bucket word (assuming one long word, 8 bytes, 64 pipapo > > > rules) is 16 * 8 = 128 bytes per group-row in the looking table. Then, > > > the number of group-rows is 8 given that 32 bits, then 32 / 4 = 8 > > > group-rows. > > > > > > 8 * 128 bytes = 1024 bytes per lookup table. > > > > > > Assuming 3 fields, then this is 1024 * 3 = 3072 bytes. > > > > > > * Using 8-bits groups: 2^8 = 256, then 256 * 8 = 2048 bytes per > > > group-row. Then, 32 / 8 = 4 group-rows in total. > > > > > > 4 * 2048 bytes = 8192 bytes per lookup table. > > > > > > Therefore, 3 * 8192 = 24576 bytes. Still rather small. > > > > > > This is missing the mapping table that links the lookup tables in the > > > memory counting. And I'm assuming that the number of pipapo rules in > > > the lookup table fits into 64-bits bucket long word. > > > > ...the (reasonable?) worst case I wanted to cover was two IPv6 > > addresses, one port, one MAC address (in ipset terms > > "net,net,port,mac"), with 2 ^ 16 unique, non-overlapping entries each > > (or ranges expanding to that amount of rules), because that's what > > (single, non-concatenated) ipset "bitmap" types can do. > > I see, so you were considering the worst case. You're assuming each > element takes exactly one pipapo rule, so it's 2^16 elements, correct? Yes, right: the ranges I considered would be disjoint and of size one (single, non-overlapping addresses). I start thinking my worst case is nowhere close to being reasonable, actually. :) > You refer to a property that says that you can split a range into a > 2*n netmasks IIRC. Do you know what is the worst case when splitting > ranges? I'm not sure I got your question: that is exactly the worst case, i.e. we can have _up to_ 2 * n netmasks (hence rules) given a range of n bits. There's an additional upper bound on this, given by the address space, but single fields in a concatenation can overlap. For example, we can have up to 128 rules for an IPv6 range where at least 64 bits differ between the endpoints, and which would contain 2 ^ 64 addresses. Or, say, the IPv4 range 1.2.3.4 - 255.255.0.2 is expressed by 42 rules. By the way, 0.0.0.1 - 255.255.255.254 takes 62 rules, so we can *probably* say it's 2 * n - 2, but I don't have a formal proof for that. I have a couple of ways in mind to get that down to n / 2, but it's not straightforward and it will take me some time (assuming it makes sense). For the n bound, we can introduce negations (proof in literature), and I have some kind of ugly prototype. For the n / 2 bound, I'd need some auxiliary data structure to keep insertion invertible. In practice, the "average" case is much less, but to define it we would first need to agree on what are the actual components of the multivariate distribution... size and start? Is it a Poisson distribution then? After spending some time on this and disagreeing with myself I'd shyly recommend to skip the topic. :) > There is no ipset set like this, but I agree usecase might happen. Actually, for ipset, a "net,port,net,port" type was proposed (netfilter-devel <20181216213039.399-1-oliver@xxxxxxxxxxxxxx>), but when József enquired about the intended use case, none was given. So maybe this whole "net,net,port,mac" story makes even less sense. > > Also ignoring the mapping table (it's "small"), with 4-bit buckets: > > > > - for the IPv6 addresses, we have 16 buckets, each 2 ^ 16 > > bits wide, and 32 groups (128 bits / 4 bits), that is, 8MiB in > > total > > > > - for the MAC address, 16 buckets, each 2 ^ 16 bits wide, and 12 > > groups, 1.5MiB > > > > - for the port, 16 buckets, each 2 ^ 12 bits wide, 2 groups, 0.25MiB > > > > that is, 9.75MiB. > > > > With 8-bit buckets: we can just multiply everything by 8 (that is, > > 2 ^ 8 / 2 ^ 4 / 2, because we have 2 ^ (8 - 4) times the buckets, with > > half the groups), 78MiB. > > Yes, this is large. Compared to a hashtable with 2^16 entries, then > it's 2^17 hashtable buckets and using struct hlist_head, this is 2 > MBytes. Then, each hlist_node is 16 bytes, so 2^16 * 16 ~= 1 MByte. > That is 3 MBytes if my maths are fine. Sounds correct to me as well. > Just telling this to find what could be considered as reasonable > amount of memory to be consumed. ~10 MBytes is slightly more than, but > I agree you selected a reasonable worst through this "complex tuple". > > > And that started feeling like "a lot". However, I'm probably overdoing > > with the worst case -- this was just to explain what brought me to the > > 4-bit choice, now I start doubting about it. > > > > > Anyway, my understanding is that the more bits you use for grouping, > > > the larger the lookup table becomes. > > > > > > Still, both look very small in terms of memory consumption for these > > > days. > > > > > > I'm just telling this because the C implementation can probably get > > > better numbers at the cost of consuming more memory? Probably do this > > > at some point? > > > > Another topic is the additional amount of cachelines we would use. I > > don't expect that effect to be visible, but I might be wrong. > > > > So yes, I think it's definitely worth a try, thanks for the hint! I'll > > try to look into this soon and test it on a few archs (something with > > small cachelines, say MIPS r2k, would be worth checking, too). > > > > We could even consider to dynamically adjust group size depending on > > the set size, I don't know yet if that gets too convoluted. > > Yes, keeping this maintainable is a good idea. > > The per-cpu scratch index is only required if we cannot fit in the > "result bitmap" into the stack, right? Right. > Probably up to 256 bytes result bitmap in the stack is reasonable? > That makes 8192 pipapo rules. There will be no need to disable bh and > make use of the percpu scratchpad area in that case. Right -- the question is whether that would mean yet another implementation for the lookup function. > If adjusting the code to deal with variable length "pipapo word" size > is not too convoluted, then you could just deal with the variable word > size from the insert / delete / get (slow) path and register one > lookup function for the version that is optimized for this pipapo word > size. Yes, I like this a lot -- we would also need one function to rebuild tables when the word size changes, but that sounds almost trivial. Changes for the slow path are actually rather simple. Still, I start doubting quite heavily that my original worst case is reasonable. If we stick to the one you mentioned, or even something in between, it makes no sense to keep 4-bit buckets. By the way, I went ahead and tried the 8-bit bucket version of the C implementation only, on my usual x86_64 box (one thread, AMD Epyc 7351). I think it's worth it: 4-bit 8-bit net,port 1000 entries 2304165pps 2901299pps port,net 100 entries 4131471pps 4751247pps net6,port 1000 entries 1092557pps 1651037pps port,proto 30000 entries 284147pps 449665pps net6,port,mac 10 entries 2082880pps 2762291pps net6,port,mac,proto 1000 entries 783810pps 1195823pps net,mac 1000 entries 1279122pps 1934003pps I would now proceed extending this to the AVX2 implementation and (once I finish it) to the NEON one, I actually expect bigger gains there. > Probably adding helper function to deal with pipapo words would help > to prepare for such update in the future. There is the ->estimate > function that allows to calculate for the best word size depending on > all the information this gets from the set definition. Hm, I really think it should be kind of painless to make this dynamic on insertion/deletion. -- Stefano