<div class="gmail_quote">On Thu, Sep 20, 2012 at 7:36 PM, Boudewijn Rempt <span dir="ltr"><<a href="mailto:boud@valdyas.org" target="_blank">boud@valdyas.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<div class="im">On Thursday 20 September 2012 Sep, Dmitry Kazakov wrote:<br>
<br>
> 1) Grayscale Selections<br>
><br>
> I think, we surely need to do it. We speak about this feature for a<br>
> really long time. Boud made some progress in it some time ago. We must<br>
> complete it finally. I think, optimistically, it may take about two<br>
> weeks.<br>
<br>
</div>Depending on the approach you take, it might well take 4 to 8 weeks, I think.<br>
<br>
The simple approach is the one I tried to implement: make selections have a grayscale + alpha channel and use the combined value as the selection value. This works with the current paint system because the current paint system cannot handle precomputed alpha. Disadvantages: twice the memory usage for selections, plus the calculation needed when determining the selection value. But because it works within the way krita was designed to work, it's not hacky, imo.<br>
</blockquote><div><br></div><div>It somewhat problematic as most of the code in Krita assumes that the selection/mask is just one channel.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
The "right" approach would be to make krita work (optionally) with premultiplied alpha internally. I cannot offer insight in how complex this would be. In 2003, Patrick Julien was already talking about it, but it never happened. I guess that has a reason...<br>
</blockquote><div><br></div><div>This would still require a completely different way the composite ops work.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
The really hacky approach would be to hack the paint system to special case the grayscale-no-alpha colorspace...<br></blockquote><div><br></div><div>What you could do is to temporarely add an alpha channel to the selection/mask e.g. in readBytes. Disadvantage is that this would cost some performance.</div>
<div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
Tangentially related: for vectorizing the composite ops, it would actually be useful to have the alpha channel out of band. That could be relevant for the selection mask issue as well.<br></blockquote><div><br></div><div>
That is an issue I ran into. The interleaved storage is a problem for vectorization were I'm not sure what's the best way to solve that. For example the alpha channel is computed completely seperate most of the time, also the SIMD registers might be a lot bigger than what we can use when we only load one pixel at once. I wonder if it might be better to use a planar storage for pixels.</div>
<div><br></div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
When starting on this job, it's essential to carefully review the earlier discussions on the topic since there were quite a few important discoveries about krita internals made.<br>
<div class="im"><br>
<br>
> 3) Optimization for speed<br>
><br>
> Recently I've done some tests to find what I can do in the data<br>
> manager or merging code to make Krita faster and got no surprising<br>
> results. The data manager's code takes less than 1% of time (master,<br>
> situation with enough of memory) and the merging takes 23% (6% of<br>
> it is composite op), which is not much as well. The greatest<br>
> portion of time is taken by the brush and compositeop code, which is<br>
> currently the area of Sven's work. If he wants, of course, I can do<br>
> something in these area as well, but I guess, I should first try to do<br>
> some other important tasks in Krita.<br>
<br>
</div>I agree.<br>
<div class="im"><br>
> 4 and 5) Porting the rest of Krita to the strokes<br>
><br>
> I tried to compose a list of classes which are left to be ported to<br>
> strokes. You can find the complete list here [2]. The tools, visitors<br>
> and commands should be definitely ported as part of this work. But the<br>
> situation with "actions" is a bit more complicated. I think this work<br>
> should be done together with some steps towards the recording system<br>
> which we discussed before [1]. Otherwise the refactoring of their code<br>
> will be quite useless, because currently they work using the legacy<br>
> interface to the strokes system quite well.<br>
><br>
> I guess, I can do this in the following way: first, I do the porting<br>
> of tools, commands and visitors. This may take 1 or 2 weeks. And after<br>
> that I can start doing the basis for the recording system refactoring<br>
> the actions on my way. I don't know how much time this work can take.<br>
<br>
</div>Yes, I agree with that plan. I hate having unfinished refactorings around, and it would be good to finish this one step by step.<br>
<div class="im"><br>
<br>
> 6) Memory footprint problem<br>
><br>
> Well, speaking truly, I have seen only few memory problems in<br>
> Krita.<br>
<br>
</div>The really core issue here is that we take more memory to store pixels than Gimp or Photoshop. I don't know how they do this -- I just know that comparing the memory footprint after loading another image shows a big difference. All the other things, undo history, brushes, canvas projection caches aren't the issue for me.</blockquote>
<div><br></div><div>Might be that they compress tiles in memory or just work on a smaller image and do the work in the background. RLE could be used to compress in memory, but that requires planar data as the experiments with file saving have shown (shuffleing was slow). </div>
<div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div class="im">
> According to the massif we spend the memory on the following:<br>
><br>
> 1. About constant 78MiB we spend on Gbr brushes, paintop presets and<br>
> pattrens.<br>
<br>
</div>This could be easily solved by doing lazy loading of all these resources. Just cache the icon used in the in the selector -- KDE has a class for that, I think. Then when the user actually uses the resource, load it from disk. This gets more important when people start using bigger brush collections. Our developer resource set is about 80M, but I bet that Deevad's set is much larger.<br>
<div class="im"><br>
><br>
> 2. 2% of your computer RAM is always spent on the pool of precloned<br>
> tiles. These tiles are cached to ease painting of next strokes.<br>
><br>
> 3. Let the image takes N MiB of memory, then (N + h*N) MiB we spend on<br>
> storing the image and the history information. Currently the<br>
> swapper is configured in the following way:<br>
><br>
> - normally, Krita stores all the tiles (both history and image<br>
> tiles) in RAM.<br>
><br>
> - when the amount of memory consumed reaches the level of 25% of<br>
> total RAM , Krita starts dumping the history tiles to the hard drive.<br>
> The process of dumping of history tiles stops either when Krita<br>
> occupies<br>
> less than 22% of memory or when there are no history tiles left in<br>
> RAM.<br>
> When all the history tiles are gone, Krita continues to grow above the<br>
> 25% level.<br>
><br>
> - when the amount of memory consumed reaches the level of 50%, the<br>
> swapper starts dumping real tiles and keeps the total memory on<br>
> this level.<br>
<br>
</div>I actually think that this is fine -- though we might want a config option so users can commit a bigger % of their memory to krita, maybe?<br>
<div class="im"><br>
><br>
> Speaking truly, I don't see a reason why we should dump all the<br>
> history information immediately, but if you think it can make Krita<br>
> faster we can change the default value for memorySoftLimitPercent<br>
> to zero and then no history information will be present in memory<br>
> ever. It will always be stored on a hard-drive.<br>
><br>
> 4. QPainter canvas. For the QPainter canvas we store the copy of the<br>
> image projection in memory (in a usual paint device). But (!) when<br>
> the color spaces of the image and the monitor coincide (I guess, it<br>
> is the common case now, but it will surely change) the two paint<br>
> devices share all the tiles, so this copy doesn't consume any extra<br>
> memory. If the color spaces are different, e.g. image is 48bit,<br>
> then the tiles can't be shared and this paint device stores the<br>
> full copy of the projection.<br>
><br>
> 5. OpenGL canvas. The case of the OpenGL canvas is even worse: it<br>
> always stores the full copy of the image in form of tiled<br>
> textures. These tiles can be stored either in RAM (in case of<br>
> on-chip video) or in memory of graphics controller (in case of<br>
> discrete graphics).<br>
><br>
><br>
> Ok, so I can see the following things we can do with our memory<br>
> consumption:<br>
><br>
> 1. Make GUI for the configuration of the swapper so everyone can<br>
> choose how much memory Krita's tiles take.<br>
><br>
<br>
</div>Ah, right :-)<br>
<div class="im"><br>
> 2. Try to set default value for history tiles in memory to 0 and see<br>
> what results it will have.<br>
><br>
> 3. Implement Region Of Interest (ROI) for the copy of the projection<br>
> in the UI. It means that this copy will store only the part of the<br>
> image that is actually shown on screen. The rest of the image will<br>
> be fetched from KisImage->projection() on-demand. This will give a<br>
> benefit on high zooms, but will not give any help when the image is<br>
> shown "fit-page". And of course, it'll make panning and zooming a<br>
> bit slower.<br>
><br>
> Such thing is not directly possible currently. The problem is that<br>
> all the data transfers are currently initiated by the KisImage. UI<br>
> cannot fetch any data from KisImage, because it can be in progress<br>
> of calculation. But this problem can be solved in an elegant way:<br>
> when a user will pan (or zoom) the UI will create empty tiles (in<br>
> case of OpenGL) for these areas and temporarily show them on<br>
> screen. Then it will sent KisImage a signal saying it needs these<br>
> areas and KisImage will deliver these parts of the image using<br>
> ususal mechanism sigImageUpdated(). This can be used for<br>
> krita-setch-rempt as well, btw.<br>
><br>
> 4. If we do ROI we can prescale the image (using SSE, probably?)<br>
> before transferring it to the UI. It could look like Mipmapping,<br>
> but we could store one level only. Say, the copy could store<br>
> the nearest power-of-two level of the image.<br>
<br>
<br>
</div>I don't think that this is really something urgent. As for both the qpainter and the opengl1 canvas, I still would like to finish the opengl2 es canvas we use in sketch and make it the default, maybe only canvas. For that, it needs to be have much better performance than it currently has, though. as well as the requisite hooks for shaders for opencolorio and so on.<br>
<br>
That reminds me... I need to fix the opengl1 canvas support for ocio.<br>
<div class="im"><br>
> 5. Implement a better compression for tiles going to the<br>
> disk. E.g. with this algorithm [3]. I'm not sure how much it'll<br>
> give us.<br>
><br>
> 6. We can play a bit with compression of tiles in memory. This is<br>
> possible right now: just increase swapWindowSize and swapSlabSize<br>
> proportionally and decrease memory{Hard,Soft}LimitPercent. But I<br>
> doubt it'll give us anything.<br>
<br>
</div>Well, it depends. Aren't tiles allocated from a pool right now? That only works if all tiles are the same size, which won't be true if we compress them. But I think that this is how photoshop works, actually. Keep the layer's pixel data compressed until the tile is accessed and either discard the decompressed data if accessed RO or recompress if changed after the access changed.<br>
<div class="im"><br>
> Conclusion:<br>
><br>
> Well, I think I layed out too many ideas ;) And we should choose which<br>
> one should be implemented and which not.<br>
><br>
> I believe that the first two points (grayscale masks and bugs) are the<br>
> most important topics now. Currently, we can gain speed only from the<br>
> optimization of the brushes and composite ops. So after doing first<br>
> two options I can either help Sven doing this (if needed) or start<br>
> working with strokes and recording. And juggling with memory<br>
> footprint... Speaking truly, I don't think that all the memory options<br>
> except the first two worth the effort. Currently, Krita eats as much<br>
> memory as we tell it to eat, and the limiting factor now is not<br>
> memory, but calculations.<br>
><br>
> So, all the comments are welcome! :)<br>
<br>
</div>I'd say:<br>
<br>
1) grayscale selection, because that gives artists immediate benefits<br>
<br>
217292 Ability to paint transparency masks )<br>
<br>
2) finish the strokes refactoring, so we can remove duplicate code paths in krita<br>
<br>
252071 The stroke of most of the the tools can't be cancelled<br>
284457 Remaining tools need to be ported to the strokes framework<br>
<br>
3) either work with Sven on optimizing composite ops or start checking a number of "interesting, possibly related bugs":<br>
<br>
302664 Brush leaving artifacts after changing cursor shape in settings, also affecting whole canvas<br>
286485 Moving layers is extremely slow and leaks memory (esp. Clone layers)<br>
289915 Krita's Canvas produces subtle artifacts on incremental updates<br>
295228 Fast undo-redo using history docker with clones freezes undo/redo actions<br>
302758 Zoom center is wrong (was: Canvas Interaction feedback)<br>
<br>
I think that we should postpone the memory optimizations for now.<br>
<div class="HOEnZb"><div class="h5"><br>
<br>
--<br>
Boudewijn Rempt<br>
<a href="http://www.valdyas.org" target="_blank">http://www.valdyas.org</a>, <a href="http://www.krita.org" target="_blank">http://www.krita.org</a>, <a href="http://www.boudewijnrempt.nl" target="_blank">http://www.boudewijnrempt.nl</a><br>
_______________________________________________<br>
kimageshop mailing list<br>
<a href="mailto:kimageshop@kde.org">kimageshop@kde.org</a><br>
<a href="https://mail.kde.org/mailman/listinfo/kimageshop" target="_blank">https://mail.kde.org/mailman/listinfo/kimageshop</a><br>
</div></div></blockquote></div><br>