<div dir="ltr">On Tue, Feb 27, 2018 at 9:23 AM, Kurt Kartaltepe <span dir="ltr"><<a href="mailto:kkartaltepe@gmail.com" target="_blank">kkartaltepe@gmail.com</a>></span> wrote:<br><div class="gmail_extra"><div class="gmail_quote"><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">I'd be more than happy volunteer my time, as I think this is a<br>
worthwhile endeavor. However the fontconfig code base is quite<br>
sophisticated from my digging, so I'm not sure I would be able to make<br>
any changes without some serious study time. I'll probably be working<br>
on this in my spare time for my own usage and hopefully a PoC will<br>
make it way back to the list if I am successful.<br></blockquote><div><br></div><div>I wouldn't say it's sophisticated per se, more that it's showing its age (more below).<br></div><div>But yeah, hard to contribute without studying it seriously first.<br><br></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
Any information/documentation that might ease deconstructing the<br>
fontconfig code base would of course be appreciated. Notably any<br>
documentation on the structures and terminology used throughout the<br>
code base and cache. Or any thoughts you may have had on this subject<br>
to guide the implementation.<span class="gmail-m_4799675307152726462gmail-HOEnZb"><font color="#888888"><br></font></span></blockquote><div><br></div><div>Where do I start...<br><br></div><div>Initially, fontconfig stored cached patterns in text files and read them back. In 2006, Patrick Lam, then a PhD student at MIT, worked under a grant from Novell, to change fontconfig to use mmap'ed binary caches, to save on load time & memory. There was a significant hurdle however: some of the structs were exposed in the public API. This made it hard to change them structurally. Instead, a hack was used to store offsets in pointer struct members by setting the lower bit of the pointer member to 1, knowing that the structs those point to have an alignment > 1. That complicates the code base.<br><br></div><div>Another problem with caching subsystem is that, indeed, on some filesystems (FAT), the timestamp is very coars (seconds). That introduces race conditions that are hard to work around. So, locking was introduced. On other filesystems (NFS), any filesystem syscall is very expensive. So, for example, stat'ing every font is prohibitively expensive. These things all constraint the designs that can be used. For example, if stat was not expensive (as in on a regular Linux system with local filesystem), we could just ftw() all font dirs and for each font file check in an on-disk hashmap if it's already in the cache and add it if it's not. But we can't do that...<br><br></div><div>More recently, work was done to be able to reuse the cache even when a font directory is mounted at a different position in the filesystem, to support flatpak. This, again, given that we had to support existing API, complicated the codebase even more.<br><br></div><div>Back to what can be done.... If we could break API or if it wasn't exposed so much, I would have suggested building on top of GVariant [0] or Protocol Buffers [1]. The cache then will simply be a GVariant hash on the disk, same way that dconf [1] works. But we cannot plain use that, so next thing is to study it and reimplement to accommodate existing API.<br><br></div><div>I'll try to write more later.<br></div><div><br></div><div><br>[0] <a href="https://macsphere.mcmaster.ca/handle/11375/21321" target="_blank">https://macsphere.mcmaster.ca/<wbr>handle/11375/21321</a><br>[1] <a href="https://github.com/google/protobuf">https://github.com/google/protobuf</a><br>[2] <a href="https://wiki.gnome.org/action/show/Projects/dconf">https://wiki.gnome.org/action/show/Projects/dconf</a><br><br> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><span class="gmail-m_4799675307152726462gmail-HOEnZb"><font color="#888888">
--Kurt Kartaltepe<br>
</font></span><div class="gmail-m_4799675307152726462gmail-HOEnZb"><div class="gmail-m_4799675307152726462gmail-h5"><br>
On Mon, Feb 26, 2018 at 8:57 PM, Behdad Esfahbod <<a href="mailto:behdad@behdad.org" target="_blank">behdad@behdad.org</a>> wrote:<br>
> Thanks for clarification. No worries.<br>
><br>
> Rewriting the cache is an interesting challenge, but so far we don't have<br>
> any volunteers.<br>
><br>
> On Mon, Feb 26, 2018 at 6:28 PM, Kurt Kartaltepe <<a href="mailto:kkartaltepe@gmail.com" target="_blank">kkartaltepe@gmail.com</a>><br>
> wrote:<br>
>><br>
>> I have rebuilt 2.12.93 tonight and it appears I was mistaken. I had<br>
>> attempted to replace 2.12.6 in my build chain but that must have been<br>
>> reverted as 2.12.93 indeed provides ~100x improvement and builds the<br>
>> cache on my system in 600ms.<br>
>><br>
>> I still hold this is not a "blanket you should improve it" post.<br>
>> Indeed a hashmap (or any mapping between patterns and files that<br>
>> allows rapid validation of non-dirty files) on disk that reuses the<br>
>> patterns for files that didnt change is indeed what I suggesting from<br>
>> the start. I don't see why this needs to be lock-free as the entire<br>
>> structure can be atomically updated using the same mechanisms already<br>
>> in use for the cache. I defer to your experience if this cache is<br>
>> contended enough to warrant such a structure.<br>
>><br>
>> I understand this is would be a significant project which is why i<br>
>> brought it to the mailing list and now that font cache build times are<br>
>> in seconds for large font libraries it is indeed harder to justify.<br>
>> Thank you very much for your time and sorry this all started due to a<br>
>> mistake on my own part. (I hope I have not been using terms<br>
>> inappropriately. but I have been using font/file interchangeably and<br>
>> from your replies it appears this may have been a mistake).<br>
>><br>
>> --Kurt Kartaltepe<br>
>><br>
>> On Mon, Feb 26, 2018 at 5:26 PM, Behdad Esfahbod <<a href="mailto:behdad@behdad.org" target="_blank">behdad@behdad.org</a>><br>
>> wrote:<br>
>> > On my laptop, warm fc-cache -f of over 2000 fonts takes 3.5s. So maybe<br>
>> > worth<br>
>> > checking what's taking so much time on your setup.<br>
>> ><br>
>> > Believe me, if we knew how to make it faster easily, we would have done.<br>
>> > So,<br>
>> > any blanket "you should improve it" has no information content<br>
>> > whatsoever.<br>
>> ><br>
>> > Caching per font is not realistic.<br>
>> ><br>
>> > The best I can think of, requires a complete rewrite of the caching, and<br>
>> > would use a single cache file that implements a lock-free hashmap on the<br>
>> > disk and reuses pattern for files that didn't change. But that's a very<br>
>> > significant project to undertake.<br>
>> ><br>
>> > On Mon, Feb 26, 2018 at 3:38 AM, Kurt Kartaltepe <<a href="mailto:kkartaltepe@gmail.com" target="_blank">kkartaltepe@gmail.com</a>><br>
>> > wrote:<br>
>> >><br>
>> >> Sorry It appears I have not been replying to the list.<br>
>> >><br>
>> >> I would like to add testing on 2.12.6 before the much improved<br>
>> >> performance changes was ~40s cache build times with significant disk<br>
>> >> I/O. So the newly improved scanning is much appreciated but doesn't<br>
>> >> solve all the issues with cache build times.<br>
>> >><br>
>> >> On Mon, Feb 26, 2018 at 5:29 AM, Kurt Kartaltepe<br>
>> >> <<a href="mailto:kkartaltepe@gmail.com" target="_blank">kkartaltepe@gmail.com</a>><br>
>> >> wrote:<br>
>> >> > 2.12.93 as released on<br>
>> >> > <a href="https://www.freedesktop.org/software/fontconfig/release/" rel="noreferrer" target="_blank">https://www.freedesktop.org/so<wbr>ftware/fontconfig/release/</a><br>
>> >> ><br>
>> >> > On Mon, Feb 26, 2018 at 5:27 AM, Behdad Esfahbod <<a href="mailto:behdad@behdad.org" target="_blank">behdad@behdad.org</a>><br>
>> >> > wrote:<br>
>> >> >> Just to make sure we are on the same page, which fontconfig version<br>
>> >> >> are<br>
>> >> >> you<br>
>> >> >> testing with?<br>
>> >> >><br>
>> >> >> On Mon, Feb 26, 2018 at 3:21 AM, Kurt Kartaltepe<br>
>> >> >> <<a href="mailto:kkartaltepe@gmail.com" target="_blank">kkartaltepe@gmail.com</a>><br>
>> >> >> wrote:<br>
>> >> >>><br>
>> >> >>> For clarification, I have tested with ONLY the ttf fonts on my<br>
>> >> >>> system.<br>
>> >> >>> In this case the normal 18s cache build step takes 15s. This<br>
>> >> >>> suggests<br>
>> >> >>> to me there is no significant difference between FON and TTF, as<br>
>> >> >>> they<br>
>> >> >>> made up ~21% of my fonts and removing them resulted in a<br>
>> >> >>> proportionate<br>
>> >> >>> savings. Sorry If my OP was misleadingly suggesting that FON files<br>
>> >> >>> were exceptionally slow, I only meant that they may not have<br>
>> >> >>> received<br>
>> >> >>> the same improvement as TTF files which may just be my<br>
>> >> >>> misunderstanding of the changes you made and lack of testing.<br>
>> >> >>><br>
>> >> >>> I am concerned with why it seems acceptable to rebuild the entire<br>
>> >> >>> cache when only a tiny portion of it has actually changed. Users<br>
>> >> >>> for<br>
>> >> >>> which rebuilding the cache is a significant event are those with<br>
>> >> >>> large<br>
>> >> >>> font libraries. These users are are by their very nature more<br>
>> >> >>> likely<br>
>> >> >>> to add or remove fonts from their library. It seems that this is<br>
>> >> >>> the<br>
>> >> >>> worst possible case for the current caching strategy, and *this*<br>
>> >> >>> seems<br>
>> >> >>> like an issue worth fixing.<br>
>> >> >>><br>
>> >> >>> In this case if checksuming files is slower than scanning them the<br>
>> >> >>> issue still stands. Why checksum files that haven't changed? Does<br>
>> >> >>> fontconfig not trust filesystem metadata? It would appear directory<br>
>> >> >>> change times are used in detecting when to rescan so why can this<br>
>> >> >>> not<br>
>> >> >>> be extended to files instead of the expensive checksum?<br>
>> >> >>><br>
>> >> >>> FWIW an md5sum of my entire font library takes ~1s with hot caches<br>
>> >> >>> which I still find unacceptable as my library is possibly<br>
>> >> >>> significantly smaller and my system significantly more powerful<br>
>> >> >>> than a<br>
>> >> >>> potential user's.<br>
>> >> >>><br>
>> >> >>> --Kurt Kartaltepe<br>
>> >> >>><br>
>> >> >>> On Sun, Feb 25, 2018 at 9:10 PM, Behdad Esfahbod<br>
>> >> >>> <<a href="mailto:behdad@behdad.org" target="_blank">behdad@behdad.org</a>><br>
>> >> >>> wrote:<br>
>> >> >>> > What's with fon files being slow? Please report *that* and let's<br>
>> >> >>> > fix<br>
>> >> >>> > it.<br>
>> >> >>> ><br>
>> >> >>> > We've made scanning, like, 100x faster already. 2007 stats are<br>
>> >> >>> > irrelevant.<br>
>> >> >>> > Checksuming files is slower than scanning them now.<br>
>> >> >>> ><br>
>> >> >>> > On Sun, Feb 25, 2018 at 8:08 AM, Kurt Kartaltepe<br>
>> >> >>> > <<a href="mailto:kkartaltepe@gmail.com" target="_blank">kkartaltepe@gmail.com</a>><br>
>> >> >>> > wrote:<br>
>> >> >>> >><br>
>> >> >>> >> While trying to move a project to the pango stack I noticed the<br>
>> >> >>> >> native<br>
>> >> >>> >> font selection backends were bad/useless on some platforms (like<br>
>> >> >>> >> windows see [1]). So I opted to try and use fontconfig on all<br>
>> >> >>> >> platforms as it performs outstandingly and has wonderful<br>
>> >> >>> >> defaults<br>
>> >> >>> >> for<br>
>> >> >>> >> all platforms.<br>
>> >> >>> >><br>
>> >> >>> >> However during this transition I noticed that there are some<br>
>> >> >>> >> major<br>
>> >> >>> >> issues with cache build speed and during investigation I see<br>
>> >> >>> >> that<br>
>> >> >>> >> there has recently been effort to improve the situation[2]. From<br>
>> >> >>> >> what<br>
>> >> >>> >> I can tell the fontconfig team has maintained that these cache<br>
>> >> >>> >> issues<br>
>> >> >>> >> were irrelevent for the primary fontconfig platform (linux) [3].<br>
>> >> >>> >> On<br>
>> >> >>> >> linux of course the cache is global and maintained usually by<br>
>> >> >>> >> font<br>
>> >> >>> >> packages ensuring its up-to-date. However it was precisely this<br>
>> >> >>> >> the<br>
>> >> >>> >> slow cache build times that lead to package managers being<br>
>> >> >>> >> required<br>
>> >> >>> >> to<br>
>> >> >>> >> build in additional tooling to support not rebuilding cache for<br>
>> >> >>> >> every<br>
>> >> >>> >> font installed [4].<br>
>> >> >>> >><br>
>> >> >>> >> Anyway I hope that is enough reason to persuade you that there<br>
>> >> >>> >> are<br>
>> >> >>> >> substantial improvements to make to the caching strategy and<br>
>> >> >>> >> they<br>
>> >> >>> >> are<br>
>> >> >>> >> beneficial not only for the odd platforms (osx, windows) but<br>
>> >> >>> >> also<br>
>> >> >>> >> for<br>
>> >> >>> >> Linux.<br>
>> >> >>> >><br>
>> >> >>> >> My question is if fontconfig would be receptive to<br>
>> >> >>> >> building/accepting<br>
>> >> >>> >> a patch modifying the caching strategy to include checkums per<br>
>> >> >>> >> file<br>
>> >> >>> >> instead of/in addition to per directory. Currently any change to<br>
>> >> >>> >> directory (such as adding a new font) invalidates all fonts<br>
>> >> >>> >> within<br>
>> >> >>> >> that directory. This means for directories like the system<br>
>> >> >>> >> directory<br>
>> >> >>> >> it results in re scans of hundreds or more fonts. Thankfully<br>
>> >> >>> >> this<br>
>> >> >>> >> is<br>
>> >> >>> >> faster on platforms like linux where all fonts on freetype.<br>
>> >> >>> >> However<br>
>> >> >>> >> this improvement in scanning did not carry over to windows with<br>
>> >> >>> >> its<br>
>> >> >>> >> many FNT (150 on the average install) and even on my very robust<br>
>> >> >>> >> development machine building a cache for a mere 650 files takes<br>
>> >> >>> >> half a<br>
>> >> >>> >> minute. This might be acceptable on install of the application<br>
>> >> >>> >> where<br>
>> >> >>> >> we can take our time building the cache, but what happens when a<br>
>> >> >>> >> user<br>
>> >> >>> >> installs 1 more font? A change to cache individual file<br>
>> >> >>> >> checksums<br>
>> >> >>> >> would provide fontconfig a way to only require the expensive<br>
>> >> >>> >> coverage<br>
>> >> >>> >> check of a single font instead of the entirety of a users. I<br>
>> >> >>> >> dare<br>
>> >> >>> >> say<br>
>> >> >>> >> with this exact change the need to use a faster less robust<br>
>> >> >>> >> coverage<br>
>> >> >>> >> check that made scanning freetype fonts faster may be unneeded<br>
>> >> >>> >> as<br>
>> >> >>> >> the<br>
>> >> >>> >> number of scans required to rebuild a cache would reduced 100x<br>
>> >> >>> >> on<br>
>> >> >>> >> the<br>
>> >> >>> >> average system or more.<br>
>> >> >>> >><br>
>> >> >>> >> I'm certain such a change would be highly appreciated by all<br>
>> >> >>> >> fontconfig consumers who are hoping to use its powerful feature<br>
>> >> >>> >> set<br>
>> >> >>> >> in<br>
>> >> >>> >> a multiplatform context.<br>
>> >> >>> >><br>
>> >> >>> >> --Kurt Kartaltepe<br>
>> >> >>> >><br>
>> >> >>> >> [1] <a href="https://bugzilla.gnome.org/show_bug.cgi?id=162681" rel="noreferrer" target="_blank">https://bugzilla.gnome.org/sho<wbr>w_bug.cgi?id=162681</a><br>
>> >> >>> >> [2]<br>
>> >> >>> >><br>
>> >> >>> >><br>
>> >> >>> >><br>
>> >> >>> >> <a href="https://lists.freedesktop.org/archives/fontconfig/2017-August/005986.html" rel="noreferrer" target="_blank">https://lists.freedesktop.org/<wbr>archives/fontconfig/2017-Augus<wbr>t/005986.html</a><br>
>> >> >>> >> [3] <a href="https://bugs.freedesktop.org/show_bug.cgi?id=64766" rel="noreferrer" target="_blank">https://bugs.freedesktop.org/s<wbr>how_bug.cgi?id=64766</a><br>
>> >> >>> >> [4]<br>
>> >> >>> >><br>
>> >> >>> >><br>
>> >> >>> >><br>
>> >> >>> >> <a href="https://lists.freedesktop.org/archives/fontconfig/2007-October/002728.html" rel="noreferrer" target="_blank">https://lists.freedesktop.org/<wbr>archives/fontconfig/2007-Octob<wbr>er/002728.html</a><br>
>> >> >>> >> ______________________________<wbr>_________________<br>
>> >> >>> >> Fontconfig mailing list<br>
>> >> >>> >> <a href="mailto:Fontconfig@lists.freedesktop.org" target="_blank">Fontconfig@lists.freedesktop.o<wbr>rg</a><br>
>> >> >>> >> <a href="https://lists.freedesktop.org/mailman/listinfo/fontconfig" rel="noreferrer" target="_blank">https://lists.freedesktop.org/<wbr>mailman/listinfo/fontconfig</a><br>
>> >> >>> ><br>
>> >> >>> ><br>
>> >> >>> ><br>
>> >> >>> ><br>
>> >> >>> > --<br>
>> >> >>> > behdad<br>
>> >> >>> > <a href="http://behdad.org/" rel="noreferrer" target="_blank">http://behdad.org/</a><br>
>> >> >><br>
>> >> >><br>
>> >> >><br>
>> >> >><br>
>> >> >> --<br>
>> >> >> behdad<br>
>> >> >> <a href="http://behdad.org/" rel="noreferrer" target="_blank">http://behdad.org/</a><br>
>> ><br>
>> ><br>
>> ><br>
>> ><br>
>> > --<br>
>> > behdad<br>
>> > <a href="http://behdad.org/" rel="noreferrer" target="_blank">http://behdad.org/</a><br>
><br>
><br>
><br>
><br>
> --<br>
> behdad<br>
> <a href="http://behdad.org/" rel="noreferrer" target="_blank">http://behdad.org/</a><br>
</div></div></blockquote></div><br><br clear="all"><br>-- <br><div class="gmail-m_4799675307152726462gmail_signature">behdad<br><a href="http://behdad.org/" target="_blank">http://behdad.org/</a></div>
</div></div>