SFML community forums

General => Feature requests => Topic started by: Halsys on June 09, 2013, 09:13:40 am

Title: Bézier curves
Post by: Halsys on June 09, 2013, 09:13:40 am
Wiki:http://en.wikipedia.org/wiki/B%C3%A9zier_curve (http://en.wikipedia.org/wiki/B%C3%A9zier_curve)

Not quite sure how hard it would be to set this up in something like SFML but I could definitely use/need it! I want to use it for in-game wires and some minor details like wind or Stink lines(like in cartoons). Maybe even tough geometry where if I did it in something like gimp it would seem too pixelated or pregenerated.

I was also thinking it could also be adapted to improve the text code, so the lines on it would look less blurred and more smooth and sharp! Now there is nothing wrong with straight lines... I just really need those awesome curvy lines!
Who's with me for this?
Title: Re: Bézier curves
Post by: Laurent on June 09, 2013, 10:21:19 am
Quote
Who's with me for this?
Not me ;D
Title: Re: Bézier curves
Post by: Halsys on June 09, 2013, 10:45:07 am
(http://www.myfacewhen.net/uploads/3645-shock.jpg)
I thought that would work.... Well what if I made a commit on github? Maybe then?
Title: Re: Bézier curves
Post by: Nexus on June 09, 2013, 10:55:19 am
Well what if I made a commit on github? Maybe then?
If Laurent says no, it's rarely because he needs help with the implementation ;)

Bezier curves are outside the scope of SFML. But Stroke (http://en.sfml-dev.org/forums/index.php?topic=2193.0) or Zoost and Zoom (http://en.sfml-dev.org/forums/index.php?topic=8026.0) may help you.
Title: Re: Bézier curves
Post by: Laurent on June 09, 2013, 11:02:45 am
Quote
If Laurent says no, it's rarely because he needs help with the implementation
Sometimes it is. Such a feature (wich could really be "a SVG module for SFML") is too big and I can't start it. If you come up with a good implementation, and most important a very good design, it might work ;)
Title: Re: Bézier curves
Post by: Halsys on June 09, 2013, 11:16:25 am
Quote
Bezier curves are outside the scope of SFML. But Stroke or Zoost and Zoom may help you.
Both Stroke,Zoost, and zoom are either out dated or totally evaporated due to the jump to SFML 2.0 and/or MegaUpload was taken down a long time ago. :|

Quote
Sometimes it is. Such a feature (wich could really be "a SVG module for SFML") is too big and I can't start it. If you come up with a good implementation, and most important a very good design, it might work ;)
Time to whip out my math skills and a good open source graphics editor(Gimp most likely) for reference!
It might take awhile so I will probably make a prototype and and build on it... then work my way up to making a commit towards SFML.Wish me luck!  ;D
Title: AW: Bézier curves
Post by: eXpl0it3r on June 09, 2013, 11:40:45 am
I once made a quick and dirty port of Stroke for SFML 2 and hoped the original author would port it properly (as promissed), but I guesd, it won't happen. I think I've posted my port on the offical forum thread for Stroke.
Title: Re: Bézier curves
Post by: OniLinkPlus on June 09, 2013, 10:13:01 pm
If it's going to be implemented, look into the recursive definition. It can be very useful for finding the different points. Also consider using a finite number of "steps", like the circle shapes do. Circles use 40 points by default, so the angle betwen two consecutive points is 2pi/40. Bezier curves use a parameter t instead of angle, so perhaps split that into 40 pieces and creating a 40 point line?
Title: Re: Bézier curves
Post by: Mario on June 10, 2013, 10:38:56 am
Of course it can be done similar to circles. Essentially, all you have to do is determining the point coordinates for a given number of points, then connect them using lines. It's rather trivial once you got the calculation right.

Might give it a try once I've got some free time to burn. :) But only if I get to see some funny games with lots of stink lines. :P
Title: Re: Bézier curves
Post by: Mario on June 10, 2013, 02:43:35 pm
Made a quick function to calculate the points for a cubic bezier curve. To use it, you'll need:

- A start point.
- A end point.
- Two control points determining the initial slope at the start/end.
- The number of segments to draw (1 = straight line).

It will then return a list of points on the actual bezier curve to draw.

There's probably room for improvements here, but you can use it as a starting point. This is no sf::Drawable, so you'll have to use your own drawing (e.g. drawing lines or using a sf::VertexArray).

Code on the Wiki (https://github.com/SFML/SFML/wiki/Source%3A-cubic-bezier-curve)

Might be interesting to have some premade class similar to sf::Line I guess.

Example drawing (20 bezier curves with 50 segments each):
(http://i.imgur.com/vFqi4C1.png)
Title: Re: Bézier curves
Post by: Laurent on June 10, 2013, 03:02:13 pm
You should put it on the wiki, with an example of drawing that uses a vertex array.

Some minor improvements:
- I don't think that the template parameter is needed, I can't imagine a use case for another type than float
- p * p * p should be more efficient than std::pow(p, 3)
Title: Re: Bézier curves
Post by: Mario on June 10, 2013, 03:10:52 pm
I thought about replacing the pow() as well, although I always thought it's better to assume "the compiler knows what it's doing".

I added the template as this might be something to be used outside drawing. Playing around with the wiki...

Edit: Done - https://github.com/SFML/SFML/wiki/Source%3A-cubic-bezier-curve
Title: Re: Bézier curves
Post by: MorleyDev on June 10, 2013, 03:56:45 pm
The way to test would be to compile std::pow and x*x*x with maximum levels of optimisation and compare the assembly.

I do know that g++ with -O3 can optimise std::pow(x, 2) down to the same assembly. I'd assume it can do the same for power of 3.

Now whether this is actually more readable than x*x*x is a whole other issue, and much more down to personal preference...
Title: Re: Bézier curves
Post by: Nexus on June 10, 2013, 04:05:02 pm
Why std::list and not std::vector?

Also, I think you can work directly with sf::Vector2f for the computation, there's no need of separating the components...
Title: Re: Bézier curves
Post by: Mario on June 10, 2013, 04:12:53 pm
Just personal preference. Use whatever you want to implement it (should be easy to replace/update) and what you think fits your specific use case.

As for working with **sf::Vector2** only, this won't work, it's lacking operators taking simple numbers (you'd have to remove all multiplications).
Title: Re: Bézier curves
Post by: Nexus on June 10, 2013, 04:18:58 pm
As for working with **sf::Vector2** only, this won't work, it's lacking operators taking simple numbers (you'd have to remove all multiplications).
It doesn't lack the operators, you just have to use the correct types.

The literal 3 is not a float, take 3.f instead. Something you should do anyway when you work with specific types, don't rely on implicit conversions :)

And concerning STL containers, std::vector is usually the default choice because it's the most efficient in many situations. Only in special cases, it can be outperformed by std::list. Also, it provides random access.
Title: Re: Bézier curves
Post by: MorleyDev on June 10, 2013, 04:23:56 pm
Just personal preference. Use whatever you want to implement it (should be easy to replace/update) and what you think fits your specific use case.

Unfortunately it's incorrect. The way to view according to  Bjarne Stroustrup himself is that vector is the default list storage, whilst list is an optimisation and should not be used prematurely.

Source: The Bjarne C++11 Keynote (http://channel9.msdn.com/Events/GoingNative/GoingNative-2012/Keynote-Bjarne-Stroustrup-Cpp11-Style)
Title: Re: Bézier curves
Post by: Mario on June 10, 2013, 04:28:27 pm
The literal 3 is not a float, take 3.f instead. Something you should do anyway when you work with specific types, don't rely on implicit conversions :)

You're right, didn't scroll below addition/subtraction (silly me :P).

Updated the wiki, so it uses the Vector2 operators as well as vectors rather than lists.
Title: Re: Bézier curves
Post by: Jove on June 16, 2013, 02:42:38 am
I only skim-read the post so forgive me if this is off the mark, but the Claw Library features support Bézier curves.

http://libclaw.sourceforge.net/index.html

Good library, I use it principally for the Tweener.
Title: Re: Bézier curves
Post by: Mario on June 16, 2013, 11:18:52 am
Perfectly fine to mention it, really depends on whether people want to include another dependency or not.

Edit: Just saw it's using Boost. Nothing wrong in that, but IMO it's a horrible dependency, that can turn compiling a small program into compiling a whole afternoon, if you don't have precompiled binaries. :(

Although I really like that tweening library. Think I'll adapt it for my own projects. :)
Title: Re: Bézier curves
Post by: Halsys on June 21, 2013, 03:55:13 am
Nice Mario! The code is extremely usable and oddly enough I was too using a AS3 Example. I will try making a class for it and then add few modifications to give it more functionality. Because yours is more kind to what I wanted. Here is basically what I'm aiming for...
~Can draw sprites in place of points and you may have random rotation for it if you want.(You give the sprite)
~Can draw dots or boxes on the calculation points.(You give the shape)
~Color and opacity morphing from starting point to end point.(With different modes)
~Updates the curve geometry only when a value was modified.
~base rotation and offset.(So you can move and rotate it without doing another calculation)

Any suggestions?
Title: Re: Bézier curves
Post by: Klaim on June 22, 2013, 10:47:05 pm
Just a shameless plug to this thread: http://en.sfml-dev.org/forums/index.php?topic=8010.0

Because the more I try to use SFML the more I really think that today a multi-media basic components library without anything for tweening is problematic. (or at least it makes development slower than it could make - and that's only one feature missing in my opinion)

So if Bezier curves implementation manage to convince Laurent that it's worth putting it in the core (and now I am sure it is, but that's my opinion), then tweening should be the next feature. It's simple, there are tons of code already existing in other frameworks that shows how to do it. So basically the problem would mainly be to convince Laurent and having a surprisingly simple, efficient and elegant interface.

Also, as libclaw has been mentioned, see the comment on lilbclaw author working on Boost.Tweener which is a generalization of libclaw.


@Laurent I'm not sure I agree with the float type only argument, in particular when both bezier curves are often used in non-graphic works too (mostly doubles for example in wide space simulations involving multi-scale positionning, or even non-plannar geometry prediction stuffs). But I guess float only would be a good start.
Title: Re: Bézier curves
Post by: Mario on June 22, 2013, 11:02:57 pm
I already have a simple tweening class, but I'm still not 100% happy with it, plus I want more complex tweening (like VertexArrays or Colors) before releasing or suggesting anything.

It can essentially be used for creating bezier curves as well, so not that much off-topic, but it's only time dependent so far.

Usage would be something like this:

sfx::Tweener<float> x(0, 10, sf::seconds(5), sfx::TweenRepeat | sfx::TweenSine);

x can then be used just like any float variable, but it will change its values "on its own" (when retrieved; from 0 to 10 using a sine curve). Same can be done with pretty much any data type like sf::Vector2f and others.
Title: Re: Bézier curves
Post by: eXpl0it3r on June 23, 2013, 12:19:33 am
The goal of SFML's graphics module is to provide a very simple API to get some 2D graphics going. The fact that we have classes such as sf::Sprite and essentially sf::Transformable and sf::Drawable, is only for convenience, because all we essentially would need now to draw 2D stuff is some kind of render target, a transformation and vertices. How much convenience will get added to SFML is up for discussion, but the only person that can make a final decision is Laurent.

Personally I see the benefits of a Tweening class, but I don't think, that it would fit into SFML's "philosophy", because if we would add it, why not add feature x or feature y or feature z? And which point will SFML be blown up so much, that you essentially have to drop the Simple from it's name? ;)
Title: Re: Bézier curves
Post by: FRex on June 23, 2013, 12:44:41 am
Quote
And which point will SFML be blown up so much, that you essentially have to drop the Simple from it's name? ;)
The acronym will still hold!
(http://oi44.tinypic.com/snfmg6.jpg)
Title: Re: Bézier curves
Post by: Laurent on June 23, 2013, 08:37:06 am
Quote
Super Fat Multimedia Library
;D

I'm currently busy with other very important tasks (bug fixing, iOS port, more important features, ...) for a while, so if I ever consider adding such features, it won't happen before a few months.

So feel free to continue the discussion (that will help me decide), but don't expect immediate feedback from me.
Title: Re: Bézier curves
Post by: Klaim on June 23, 2013, 03:44:09 pm
The goal of SFML's graphics module is to provide a very simple API to get some 2D graphics going. The fact that we have classes such as sf::Sprite and essentially sf::Transformable and sf::Drawable, is only for convenience, because all we essentially would need now to draw 2D stuff is some kind of render target, a transformation and vertices. How much convenience will get added to SFML is up for discussion, but the only person that can make a final decision is Laurent.

Tweening is not supposed to be a graphic part, it's value-animation. I wouldn't add tweening to SFML graphics library, that would be dumb. Value-animation is a basic brick which is important for tons of different kind of stuffs, the most visible being moving graphics.

Quote
Personally I see the benefits of a Tweening class, but I don't think, that it would fit into SFML's "philosophy", because if we would add it, why not add feature x or feature y or feature z? And which point will SFML be blown up so much, that you essentially have to drop the Simple from it's name? ;)

You can dump this argument at each request too, it's not a valid argument. What would be an argument against adding value-animation in a separate library which is part of SFML because it uses it's basic types?
Title: Re: Bézier curves
Post by: eXpl0it3r on June 23, 2013, 03:52:43 pm
You can dump this argument at each request too, it's not a valid argument. What would be an argument against adding value-animation in a separate library which is part of SFML because it uses it's basic types?
It's my personal opinion and yes it will work for mostly all feature request, which only shows, that I personally think, SFML's current state is enough and doesn't need any more core features. :)
Title: Re: Bézier curves
Post by: Klaim on June 23, 2013, 05:11:27 pm
Ok but that's not an argument either. I personally think it would be too much to add another feature than value animation tools, so it's just one feature more I think is really missing. I also think that it's surprising that there are threading features to SFML if I follow the logic of not adding anything other than basic building blocks of multi media library (and threading is anything but "simple").

My argument is: for a multimedia library, it quite lack features helping implementing dynamic changes of the basic bricks it provides. A generic value animation feature would be, in my opinion, a generic tool to quickly plug together all these bricks to make something that change, like moving a sprite through a curve or changing dynamically the music volume, even sprite animation can be implemented using it - all that without defining in SFML how to do these, just providing the tool to animate values (for examples floats or better: aggregates).

EDIT> Just to complete my thinking: I believe it would be a minor addition technically, a set of simple tools, but it would help a lot combine other features with anything that "moves" or is animated.
Title: Re: Bézier curves
Post by: Nexus on June 23, 2013, 06:12:54 pm
I also think that it's surprising that there are threading features to SFML if I follow the logic of not adding anything other than basic building blocks of multi media library (and threading is anything but "simple").
This has already been discussed multiple times, and threads are only provided because they are needed in the SFML implementation. That's also the reason why requests for more elaborated multithreading features are constantly rejected.

When SFML ports its code to C++11 one day, sf::Thread will probably be removed.
Title: Re: Bézier curves
Post by: Klaim on June 25, 2013, 08:41:36 pm
I also think that it's surprising that there are threading features to SFML if I follow the logic of not adding anything other than basic building blocks of multi media library (and threading is anything but "simple").
This has already been discussed multiple times, and threads are only provided because they are needed in the SFML implementation. That's also the reason why requests for more elaborated multithreading features are constantly rejected.

When SFML ports its code to C++11 one day, sf::Thread will probably be removed.

Ok then, but was networking necessary too for SFML implementation?

Anyway my point was that adding value animation would not be a big addition nor would make SFML fat and is in my opinion the only really needed feature missing. However, it should not be automatic like in other frameworks which provide a main loop and everything is updated automatically. For SFML, just having the algorithms as wrapped values (or something similar) with an easy way to update them when needed would be enough.
Title: Re: Bézier curves
Post by: Nexus on June 25, 2013, 08:52:03 pm
Do you have a concrete idea how such an API might look?
Title: Re: Bézier curves
Post by: Klaim on June 27, 2013, 05:00:05 pm
I was thinking about it but got caught on something else. Give me some time to propose some example. Meanwhile, just think it would be just functions or callable objects that would update a value (either via ref or internal, which should be easier). I'll get back to this soon.
Title: Re: Bézier curves
Post by: Nexus on June 27, 2013, 09:19:58 pm
Tweeners were once suggested for Thor, maybe the corresponding discussion  (http://en.sfml-dev.org/forums/index.php?topic=4646.msg34365#msg34365) might inspire you.
Title: Re: Bézier curves
Post by: Klaim on June 28, 2013, 02:46:53 pm
Tweeners were once suggested for Thor, maybe the corresponding discussion  (http://en.sfml-dev.org/forums/index.php?topic=4646.msg34365#msg34365) might inspire you.

I believe a non-time-based solution would be far more generic (but a little less quick to use) for SFML itself than what is proposed there. Then in Thor you could easily plug things together to propose time-based tweeners.
Title: Re: Bézier curves
Post by: Nexus on June 28, 2013, 04:01:52 pm
Thor's animations (http://www.bromeon.ch/libraries/thor/v2.0/doc/group___animation.html) are not time based, only the front-end thor::Animator is. If you read a bit further (http://en.sfml-dev.org/forums/index.php?topic=4646.msg34963#msg34963), you see I proposed a std::function<float(float)> which is very generic. Maybe one could build something on top of that.

In Thor, the concept of an animation is as follows:
void animation(Animated& object, float progress);

This is quite generic, as it allows to animate an object using an arbitrary function that takes a float in [0,1]. You can use functions, functors, std::bind, or lambda expressions. Depending on what you need, you can implement an intermediate layer that transforms a progress range to another. For example, playing an animation in reverse can be done in a single line:
auto anim = ...; // animation function satisfying the concept
auto reverse = [anim] (sf::Sprite& s, float pr) { return anim(s, 1.f - pr); };

You might also want to read my design considerations (http://en.sfml-dev.org/forums/index.php?topic=7329.msg53237#msg53237). Maybe this can inspire you for an API...
Title: Re: Bézier curves
Post by: cpolymeris on July 12, 2013, 09:11:44 am
I was also thinking it could also be adapted to improve the text code, so the lines on it would look less blurred and more smooth and sharp!

Agree this would be A Good ThingTM, and an excuse to say this code belongs into the SFML core. Would also potentially make the Font size (points) independent from the target's pixel size.

- p * p * p should be more efficient than std::pow(p, 3)

I'd find that surprising. I tend to assume people who write compilers are smarter than me.
Title: Re: Bézier curves
Post by: Mario on July 12, 2013, 09:46:33 am
- p * p * p should be more efficient than std::pow(p, 3)

I'd find that surprising. I tend to assume people who write compilers are smarter than me.
In the end, both should result in the same code, if optimized, so it shouldn't matter. However, if you don't specifically optimize your code (usually the case for debug builds), the explicit multiplication should be faster I'd guess.
Title: Re: Bézier curves
Post by: Laurent on July 12, 2013, 09:53:37 am
Quote
I'd find that surprising. I tend to assume people who write compilers are smarter than me.
I don't know if it can be optimized. The second argument is a double, so the function has to apply a generic algorithm which is much more complex than p * p * p. So unless pow maps directly to a built-in processor instruction, I don't know how your compiler would be able to make it as optimized as the explicit version.

But that's interesting, generate the ASM listing and compare ;)
Title: Re: Bézier curves
Post by: Mario on July 12, 2013, 10:16:47 am
Despite it being defined as double, the compiler should be able to determine it being a constant integer value, so I've got hope there. Anyway, let someone... try it. *sneaks away*
Title: Re: Bézier curves
Post by: Laurent on July 12, 2013, 10:39:02 am
Quote
Despite it being defined as double, the compiler should be able to determine it being a constant integer value
And what? He "knows" that this is a function that computes a power, and apply its own optimized algorithm instead? This is just a function call, the compiler knows nothing about it.
Title: Re: Bézier curves
Post by: Mario on July 12, 2013, 10:57:08 am
I imagined the compiler could just consider the whole call for optimization, similar to how memset() can be optimized out, if the memory isn't accessed anymore later. But anyway, getting quite offtopic and out of scope here. :)
Title: Re: Bézier curves
Post by: binary1248 on July 12, 2013, 12:49:02 pm
Quote
I'd find that surprising. I tend to assume people who write compilers are smarter than me.
I don't know if it can be optimized. The second argument is a double, so the function has to apply a generic algorithm which is much more complex than p * p * p. So unless pow maps directly to a built-in processor instruction, I don't know how your compiler would be able to make it as optimized as the explicit version.

But that's interesting, generate the ASM listing and compare ;)
double foo( double p )
{
        return std::pow( p, 10 );
}

double bar( double p )
{
    return p * p * p * p * p * p * p * p * p * p;
}
Code: [Select]
foo(double):
movapd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm1, %xmm0
mulsd %xmm1, %xmm0
mulsd %xmm0, %xmm0
ret
bar(double):
movapd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
mulsd %xmm0, %xmm1
movapd %xmm1, %xmm0
ret
This was generated by GCC with -O2 -march=native. It seems the compiler is more intelligent than you might expect ;). The explicit multiplication generates as many machine operations as multiplications you perform. The std::pow() variant tells the compiler the base stays the same hence it can reuse the results of previous multiplications to reduce the number of machine operations required. Explicit multiplication is O(k) whereas std::pow() is O(log k) for p ^ k.
It groups
p * p * p * p * p * p * p * p * p * p to
( ( ( p * p ) * p ) * ( p * p ) ) * ( ( ( p * p ) * p ) * ( p * p ) ) which is
( ( ( p ^ 2 ) * p ) * p ^ 2 ) ^ 2

Maybe the word "intelligent" isn't as correct as "less conservative" as explained here (http://stackoverflow.com/questions/6430448/why-doesnt-gcc-optimize-aaaaaa-to-aaaaaa). With -ffast-math both functions produce identical asm output (the shorter variant).
Title: Re: Bézier curves
Post by: Mario on July 12, 2013, 01:15:32 pm
Very interesting (and very tricky). So it essentially groups the multiplications for pow() into repetitive patterns. But yeah, for short ones it wouldn't matter as well.

Still interesting to know. :) So it definitely parses the parameter rather than just passing it to something hardcoded.
Title: Re: Bézier curves
Post by: Laurent on July 12, 2013, 01:30:23 pm
That's what I thought. binary1248 is faster than the compiler to produce ASM listings and prove me wrong ;D

This is very interesting but I still wonder how the compiler does. So it's probably not an effect of generic optimization rules, but rather a hard-coded one (std::pow + integer arg = this specific code).
Title: Re: Bézier curves
Post by: Silvah on July 12, 2013, 03:31:00 pm
It's not even the compiler, it's the standard library implementation. libstdc++ defines std::pow(double, int) as a call to __builtin_powi (http://gcc.gnu.org/onlinedocs/gcc-4.8.1/gcc/Other-Builtins.html#index-g_t_005f_005fbuiltin_005fpowi-3399). Since 10 is a compile-time constant, the compiler is then able to emit the operation inline instead of making a library call to pow.

And what? He "knows" that this is a function that computes a power, and apply its own optimized algorithm instead? This is just a function call, the compiler knows nothing about it.
The standard library is special, the implementation is free to assume all kinds of things, as long as they don't violate the spec. The standard library may not even exist except in the form of rules hard-coded in the compiler.
Title: Re: Bézier curves
Post by: binary1248 on July 12, 2013, 03:32:26 pm
When using the headers that are provided by the compiler, std::pow calls the built-in pow function. In GCC it's __builtin_pow which the compiler knows it should optimize. Compilers basically have a table of functions that they know how to optimize, so when calling those, they get immediately translated to asm.

GCC Builtins: http://gcc.gnu.org/onlinedocs/gcc-4.0.4/gcc/Other-Builtins.html
VS12 Intrinsics: http://msdn.microsoft.com/en-us/library/26td21ds%28v=vs.110%29.aspx
Title: Re: Bézier curves
Post by: Silvah on July 12, 2013, 03:39:17 pm
In GCC it's __builtin_pow which the compiler knows it should optimize.
Uhm, no. GCC doesn't optimize __builtin_pow nearly as much as it optimizes __builtin_powi. In particular
double foo(double p)
{
  return __builtin_pow(p, 10);
}

double bar(double p)
{
  return __builtin_powi(p, 10);
}
compiles down on my machine to the following assembly:
        .text
        .p2align 4,,15
        .globl  _Z3food
        .def    _Z3food;        .scl    2;      .type   32;     .endef
        .seh_proc       _Z3food
_Z3food:
.LFB0:
        .seh_endprologue
        movsd   .LC0(%rip), %xmm1
        jmp     pow
        .seh_endproc
        .p2align 4,,15
        .globl  _Z3bard
        .def    _Z3bard;        .scl    2;      .type   32;     .endef
        .seh_proc       _Z3bard
_Z3bard:
.LFB1:
        .seh_endprologue
        movapd  %xmm0, %xmm1
        mulsd   %xmm0, %xmm1
        mulsd   %xmm1, %xmm0
        mulsd   %xmm1, %xmm0
        mulsd   %xmm0, %xmm0
        ret
        .seh_endproc



What may be interesting is that this optimization takes place only when C++11 mode is disabled. With -std=c++11, or even -std=gnu++11, for
double qux(double p)
{
  return std::pow(p, 10);
}
GCC 4.8.2 20130701 generates a library call:
        .p2align 4,,15
        .globl  _Z3quxd
        .def    _Z3quxd;        .scl    2;      .type   32;     .endef
        .seh_proc       _Z3quxd
_Z3quxd:
.LFB258:
        .seh_endprologue
        movsd   .LC0(%rip), %xmm1
        jmp     pow
        .seh_endproc
Title: Re: Bézier curves
Post by: binary1248 on July 12, 2013, 04:06:38 pm
GCC doesn't even have a __builtin_pow. I was referring to the family of builtin power functions such as __builtin_powi, __builtin_powif and __builtin_powil.
Title: Re: Bézier curves
Post by: Silvah on July 12, 2013, 04:10:15 pm
GCC doesn't even have a __builtin_pow.
It does. See the documentation (http://gcc.gnu.org/onlinedocs/gcc-4.8.1/gcc/Other-Builtins.html):
Quote
The ISO C90 functions abort, abs, acos, asin, atan2, atan, calloc, ceil, cosh, cos, exit, exp, fabs, floor, fmod, fprintf, fputs, frexp, fscanf, isalnum, isalpha, iscntrl, isdigit, isgraph, islower, isprint, ispunct, isspace, isupper, isxdigit, tolower, toupper, labs, ldexp, log10, log, malloc, memchr, memcmp, memcpy, memset, modf, pow, printf, putchar, puts, scanf, sinh, sin, snprintf, sprintf, sqrt, sscanf, strcat, strchr, strcmp, strcpy, strcspn, strlen, strncat, strncmp, strncpy, strpbrk, strrchr, strspn, strstr, tanh, tan, vfprintf, vprintf and vsprintf are all recognized as built-in functions unless -fno-builtin is specified (or -fno-builtin-function is specified for an individual function). All of these functions have corresponding versions prefixed with __builtin_.
Title: Re: Bézier curves
Post by: binary1248 on July 12, 2013, 04:38:59 pm
Hmm yeah I just saw myself. If you look at the description for __builtin_powi it states:
Quote
Returns the first argument raised to the power of the second. Unlike the pow function no guarantees about precision and rounding are made.
no guarantees about precision and rounding = enable optimization
So I guess __builtin_pow is really equivalent to the explicit multiplication.
Title: Re: Bézier curves
Post by: cpolymeris on July 12, 2013, 09:55:28 pm
This is very interesting but I still wonder how the compiler does. So it's probably not an effect of generic optimization rules, but rather a hard-coded one (std::pow + integer arg = this specific code).

It's been a while since I used C++, but couldn't something like that be templated?
float template <0> float _pow(float)
{
   return 1;
}

float template <int n> float _pow(float x)
{
   return x * _pow<n - 1>;
}

float std::pow(float x, const int n)
{
  return _pow<n>(x);
}
 

If that works (or something similar, don't remember the syntax), maybe it's not a compiler optimization at all, but rather a std:: optimization.
Title: Re: Bézier curves
Post by: binary1248 on July 12, 2013, 11:28:36 pm
Boost actually employs such template meta programming black magic to optimize some things to compute stuff at compile time. Nice side effect: your compiler blows up half the time with some obscure error because the template recursion depth was set to something like 100.