- 
                Notifications
    You must be signed in to change notification settings 
- Fork 8k
Optimize JSON string encoding #17734
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Conversation
| Nice! The benchmark is quite promising. I would prefer a bit clean up of the code so the ifdefs are not mixed up. It means creating some abstraction and having ifdefs just at the top or in different file. | 
| But it's not a big deal if you can't be bothered to do the abstraction - certainly not a blocker - just thought that it might be nicer. If you are too busy and all tests fine, feel free to merge it and I might refactore it later maybe. | 
| btw I think people more likely use default (including frameworks like Laravel) so I'm not sure if  | 
| I'll try to split it off / move some code more together soon-ish. 
 Should be easy to check, I can check this. | 
| 
 Indeed most uses within Laravel and Symfony don't use it, or let the user configure it, and only a handful of places use it. A bit unfortunate as they would miss out on a huge performance win (but necessary ofc if they want interop with different character encodings) | 
| pos = 0; | ||
| us = php_next_utf8_char_mb((unsigned char *)cur, us, len, &pos); | ||
| len -= pos; | ||
| pos += pos_old; | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
| pos = 0; | |
| us = php_next_utf8_char_mb((unsigned char *)cur, us, len, &pos); | |
| len -= pos; | |
| pos += pos_old; | |
| us = php_next_utf8_char_mb((unsigned char *)cur, us, len, &pos); | |
| len -= pos - pos_old; | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We need pos to be 0 when the function call happens because len is the remaining length instead of the total length.
| /* It's more important to keep this loop tight than to optimize this with | ||
| * a trailing zero count. */ | ||
| for (; mask; mask >>= 1, *s += 1) { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
abcd*400 takes 0.06 s (1600 bytes)
"longenclosedstring"*50 takes 0.22 s (1000 bytes)
thus when " is present, the encoding is 6x slower. This is actually not an uncommon case, like HTML/XML/JSON (or basically any code) encoded in JSON.
The worst unoptimized case would be mask 10..0..01. Is loop with zend_ulong_ntz really slower here?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The ulong_ntz approach was in my other branch. It was significantly slower for more dense bitset, perhaps there's some heuristic to switch between the 2. I also found that the smaller code size that this has was very important as well for the other inputs (even the ones without any escaping).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It suprises me, but I saw your experiments prior this PR and I trust them. Probably too many branching is simply too many even for the latest super-insanely-fast CPUs :)
I have nothing to add to this PR except one more huge thank you!
| int options | ||
| ) | ||
| { | ||
| while (*len >= sizeof(__m128i)) { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Encoding one "0123456789abcde*1000" string will have absolutely different performance then array of "0123456789abcde" 1000 strings.
How feasible is something like mb_fast_check_utf8 does? - https://github.com/php/php-src/blob/php-8.4.3/ext/mbstring/mbstring.c#L5447
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The blow up in code size may optimize that case but will likely deoptimize other cases.
0b110d9    to
    f4a2bc9      
    Compare
  
    | Can this PR be merged? | 
| I wanted to do some extra test comparing the thing referred to in this comment (#17734 (comment)) on both an older and newer machine as I now have access to some new hardware too. So I'll try that today. | 
f4a2bc9    to
    860b11f      
    Compare
  
    Decoding purely multibyte UTF-8 is common for example in the case of JSON. Furthermore, we want to avoid the switch on the character set in such hot code. Finally, we also add UNEXPECTED markers to move code to the cold section which reduces pressure on the µop and instruction caches.
There are a couple of optimizations that work together:
- We now use the specialized php_next_utf8_char_mb() helper function to
  avoid pressure on the µop and instruction cache.
- It no longer emits UTF-8 bytes under PHP_JSON_UNESCAPED_UNICODE until
  it actually has to. By emitting in bulk, this improves performance.
- Code layout tweaks
  * Use a specialized php_json_append() and assertions to avoid
    allocating the initial buffer, as this is already done upfront.
  * Factor out the call to smart_str_extend() to above the UTF-16 check
    to avoid code bloat.
- Use SIMD, either with SSE2 or SSE4.2. A resolver is used when SSE4.2
  is not configured at compile time.
    | I want to actually play with it and go properly through the logic before merging so please wait a bit. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I looked quickly again and not sure if I like bundling all those things together. It's then not clear what actually gives some improvements. I would prefer to split it to multiple PR's. I'm not sure if we should bother with that utf-8 validation as it's still not optimal. It would be better to eventually look to variant of https://github.com/lemire/fastvalidate-utf-8 . IIRC it's based on variant of Bjoern Hoehrman's validator that I used in jsond: https://github.com/bukka/php-jsond/blob/587a413ee44579b9422ba36eec63bfd1734a5d16/php_jsond_utf8_decoder.h . It wasn't actually giving me better results (mainly because I didn't use ascii check) so I didn't look further but simd variant should be better and might be better look into it. Unless this gives better results already which should be however measured independently.
Also that php_json_append seems a bit like churn but maybe I'm misunderstanding something.
| /* dest has a minimum size of the input length, | ||
| * this avoids generating initial allocation code */ | ||
| ZEND_ASSERT(dest->s); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't exactly how assert can avoid generating initial allocation if it's removed in production code..?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
if it's removed in production code..?
Because it's not removed: it's changed to a ZEND_ASSUME on compilers that support this, and it's removed on compilers that don't. So it can be used as an optimization hint.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah ok, it might make sense to integrate it to smart_str if it has some possitive impact as the same case applies on serialization and possibly other places.
| 
 You know what, if it's like that then I don't care anymore, someone else can pick this up. | 
| You probably confused comment with a change request - this was just a comment (purely initial thought after a quick review). Basically I just thought that it's better not to bundle unrelated changes together. But if it's the only preferred way, I wouldn't really try to block it as I can see it's still an improvement. | 
| Niels, please do not let this PR to stale. I was following your experiments in nielsdos#120 and I trust you you went with the fastest solution. Even if for some inputs the improvement can be maybe a little better, this PR is a huge improvent as it is now already. | 
| This PR can probably benefit from #18570 to support ARM as well. | 
Split out a specialized function to decode multibyte UTF-8 sequences
Decoding purely multibyte UTF-8 is common for example in the case of
JSON. Furthermore, we want to avoid the switch on the character set in
such hot code. Finally, we also add UNEXPECTED markers to move code to
the cold section which reduces pressure on the µop and instruction
caches.
Optimize JSON string encoding
There are a couple of optimizations that work together:
avoid pressure on the µop and instruction cache.
it actually has to. By emitting in bulk, this improves performance.
allocating the initial buffer, as this is already done upfront.
to avoid code bloat.
is not configured at compile time.
Here's a graph where I test different input strings.
The tests take an input string and repeat it using
str_repeat, and then perform200000calls tojson_encodewith no flags. If(no escape)is in the label, then we passJSON_UNESCAPED_UNICODEas a flag.The first four bars show the very short strings where performance improvements won't be able to do much and no SIMD is possible. The performance difference and overhead of this approach is IMO negligible in those cases.
Next we test some extreme cases. The best case happens when a string doesn't need escaping or anything special. We can see when we need to do escaping or UTF-8 validation/escaping that the performance win is still there, but not as pronounced.
In some extreme UTF-8 (
éa*200) the performance is slightly lower but when compared to the total execution time this shouldn't be noticeable. The last two bars show that the performance when usingJSON_UNESCAPED_UNICODEis improved as well, which is hopefully the common way people work with UTF-8 in JSON.Benchmarks done on an i7-4790 on Linux with the SSE4.2 resolver (so not the theoretical optimum, but a likely default).
The benchmark gist is here: https://gist.github.com/nielsdos/51f32ca9d4610b802d0ea65e7af6ae16