# Why?
In a work related context, I had to create a hash algorithm working on a finite set of values ([0, 0xFFFFFFFF]
) to output a non sequential serie from a sequential one (the output had to be rendered as a UUID.
Basically, I wanted to avoid generating UUID looking like 00000000-0000-0000-0000-000000000001
, that will definitely appear to be sequential and easily abused to find other IDs.
# The transformation
Previously I called it a hash, it’s more like a map
function (or transform
function for my C++ folks), operating on [0, 0xFFFFFFFF] -> [0, 0xFFFFFFFF]
where the sequential aspect is lost.
What I was looking for is something like:
- 0 -> 5
- 1 -> 1632
- 2 -> 9018235
- 3 -> 543
and so on.
Easy to do, just put a few bitflips, bitswaps and call it a day. That might work, but I needed to make sure the transformation doesn’t generate any collision, or that would be bad in our case.
If you’re curious, the transformation I choose for this article looks like
|
|
A bunch of incomprehensible symbols that could seem random, as if I typed on my keyboard without looking. However the domain is crafted so that the first side (~constant ^ x) | 0x0000ffffl
won’t interfere with the second side (~(constant ^ x) & 0xffff0000l)
, with a & 0xffffffffl
at the end because I put long
everywhere to avoid casting my int
to long
each time I use them due to potential overflow (that’s still controled with the masks).
# Checking that the transformation is collision-free
My first approach was in Scala, as it is the language we use at work. I just put a few lines of code with a mutable set and hoped for the best.
|
|
Time (1 -> 0xFF’FFFF): scala Hash.scala -J-Xmx2g 5.43s user 0.37s system 291% cpu 1.988 total
However, after 10 minutes on an M1 MacBook Pro it wasn’t finished (and I had to tweak the max heap size multiple times to avoid getting max heap size errors), so I stopped it and started looking for a better way to check if my transformation was collision-free.
What’s better than trying again without changing your code? Well I did just that, but using another language, much more low level: C++.
# The C++ way
I wrote nearly the same code, but using C++, and a “better set” (since std::set
is ordered and slower in access and write), thinking that would do it:
|
|
Time (1 -> 0xfff’ffff): ./compute 9.48s user 2.48s system 93% cpu 12.796 total
And compiled in release mode to get the best performances: clang++ compute.cpp -o compute -O3 --std=c++20
. To my surprise, on my M1 MacBook Pro, it was a faster than the Scala solution but moved by 1% every 20 seconds or so! I didn’t want to have to wait for so long to iterate on the transformation in case it generated a collision. Also, no more java out of memory errors.
The “slowness” lies in two things here:
- the way I check for collisions can not be parallelized (unless I compute N sets and then merge them, which will likely take a lot of time since we are working on 4'294'967'295 values)
- emplacing in a set is more costly than just looking in an array at index
r
, it is itself hashing the value, and doing much more math than we need.
# A better C++ version
So what’s stopping me from using a std::array<char, 0xffffffff>
array? The memory (that would eat about 4GB of RAM!). But luckily all we want to do is mark an element as “used” and check if it is already registered. We could use a std::vector<bool>
, but I’ve taken this user’s comment as true: “std::vector is of poor performance compared to bitset”, which reminded me that std::bitset existed!
Creating a std::bitset<0xffffffff>
will most likely not work, since it would try to allocate it on the stack and we don’t have 530+MB of stack available, so I just wrapped it in a std::unique_ptr
for good measure:
|
|
Time (1 -> 0xfff’ffff): ./compute 0.59s user 0.05s system 73% cpu 0.864 total
If you have a keen eye you will have noticed a weird size for the std::bitset
: 0xffffffff + 1. That’s because 4'294'967'295 should be a valid value, so I needed to have it available.
This new version (including the std::cout
printing the percentage periodically) only takes about 12 seconds to tell me if the transformation has a collision or not (less if the collision is found at the beginning of the sequence). I’d say that this is quite good for me!