We’ll still have to be a bit careful of C++11 only features. But for now, friesencr is right, PODVector should be used for ints for best performance. This can also be abused for non-POD types if the constructor/destructor don’t do anything vital.
Enhex
Why the need to be careful with C++11 ?
cadaver
On VS side that means upgrading to post-VS2010 versions, which can be argued to be more bloated than the previous, and don’t support compiling for old OS versions (there may be some niche use of Urho like using an old VS version such as 2008 to compile for Win2000 / XP)
Though I’ve been using VS2013 quite exclusively for some time and personally am happy with it.
Enhex
[quote=“cadaver”]On VS side that means upgrading to post-VS2010 versions, which can be argued to be more bloated than the previous, and don’t support compiling for old OS versions (there may be some niche use of Urho like using an old VS version such as 2008 to compile for Win2000 / XP)
Though I’ve been using VS2013 quite exclusively for some time and personally am happy with it.[/quote]
To me it seems like a huge sacrifice to hold back to be compatible with outdated software.
AFAIK VS2015 support at least down to win XP.
cadaver
You can also go the other way around and ask what are the absolute vital reasons/features (not just convenience) for the engine/library itself to go for c++11? Bear in mind, you can always choose to use c++11 in your application code.
I very much like that Urho3D is clean of C++11 constructs. Nothing is keeping me from using VS2015. We do have C++11 enabled for some ThirdParty libraries that use it very minimally. There are also compilers on various platforms that do not support C++11.
Enhex
[quote=“cadaver”]You can also go the other way around and ask what are the absolute vital reasons/features (not just convenience) for the engine/library itself to go for c++11? Bear in mind, you can always choose to use c++11 in your application code.
Looks like their reasoning is mainly based on ignorance and social proof. I also posted a reply there about the “auto is evil” nonsense.
Also the argument that C++'s stdlib somehow de-matured also doesn’t make sense.
While looking for reasons is important, “absolute vital” sounds like an attempt to invalidate any reason.
What are the “absolute vital” reasons for C++ over C? What are the ones of C over ASM?
The “absolutely vital” condition doesn’t make sense! Nothing is “absolutely vital” if you can get the job done. With that line of reasoning we’ll be still using punch cards.
C++11 have features that can make code shorter, easier to read and understand, easier to maintain, and faster. What else is there to ask for?
The discussion about C++ was started by how it can boost the performance of the most commonly used container by nearly 50%. (saying “use PODVector” is like saying “Use array” or something else with non-matching functionality)
Also VS2015 have great debugging tools:
https://www.youtube.com/watch?v=NVCSuzFPzEM
The only reason so far is backward compatibility with outdated compilers. Even then, how do you draw the line for how backward compatible it is? C++98? C++85?
Of course that full backward compatibility doesn’t make any sense. A good heuristic is to be as advanced as possible while supporting the majority of the users.
Both
vs2013
and
vs2015
are backward compatible down to win XP. And I don’t think there are that many people who still use some 15+ years old version of windows.
Sorry about the rant, it’s just irrational to me and I had to let it out.
cadaver
I realize it’s a rant, but if we’re being serious, the difference between C & C++ is quite a large feature cutoff for large software such as Urho3D that could be considered “absolutely vital” for sanity, if not anything else.
Projects have to prioritize themselves differently, I can well understand if for some being modern in their language usage takes high priority. I believe the worst part of Urho not using C++11 is that project contributors don’t get to exercise their skills in the new language features.
friesencr
I hear the Magnum game engine author trip over new c++ feature bugs all the time. Often on exotic platforms. They are hard to detect sometimes.
I am a light contributor. I do not know c/c++ well. I believe also that the style that Urho is written in semantically represents the problem it is trying to solve well. As someone who does not know c++ well the more modest use of c++ features actually allows me to learn how to code better, faster. Through implementing my current project I have learned a great deal about pointer arithmetic and data.
sabotage3d
I think a game engine should be generic as possible and let the user handle anything more specific. SDL2 is one of Urho’s third-party libraries and it is written in C, it is used everywhere and it is extremely portable. But I am thumbs up for faster containers
cadaver
The perf difference between a Vector and std vector is something worth investigating. Probably is related to the reallocation of strings when the vector’s storage expands.
globus
winXP - the best that has been created in Microsoft.
It is the working environment created by the developers.
win7-8-9-10 -… created by traders to make money.
Enhex
winXP - the best that has been created in Microsoft.
It is the working environment created by the developers.
win7-8-9-10 -… created by traders to make money.
[/quote]
MS always was for profit business.
Win 7 is the best IMO.
Win 9 never existed, you missed the news
globus
But Microsoft has completed the official support Win7.
The next time Microsoft will say: DirextX 14 only for Windows 11.
Nothing personal. Only business.
1vanK
I like XP too, but new hardware is not working on it.
So I had to go to Win7.
globus
For this reason, I collected a computer
with the latest hardware resources that support WinXP.
Motherboard: ASRock z77 extreme4-M
CPU: Intel Core i7 - 3770T
DDR3 memory
Video: GeForce GTX 750Ti 2Gb
It good but hard to find motherboard with winXP support.
I use WinXP as a working environment.
For other platforms, have plans to use the remote method compilation
(using FTP connection) or duble bootable HDD.
boberfly
Hi,
I’ve learned a lot of C++ through Urho3D and prior to that Horde3D, as well as libraries often found in the VFX industry (OpenEXR/IlmBase). There’s so much to learn, but it’s fun to read up on this stuff.
I have on/off plans to make a plugin for Maya, and usually you need to match the compiler to the Maya version. So I’m a bit concerned that is C++11 really needed?
http://help.autodesk.com/view/MAYAUL/2016/ENU/?guid=__files_Setting_up_your_build_environment_htm
Visual Studio 2012 update 4, and this is the latest Maya! Older ones like Maya 2015 uses GCC 4.1.x for instance, it doesn’t say what it uses for Linux in this documentation. Recently the library OpenSubdiv reverted some C++11-isms to keep compatibility, for instance.
Not that it’s a problem for Urho3D to limit itself to a 3D DCC and I don’t want to force this requirement, but it’s kind of nice to not go with the latest and greatest. Sure there are ways around this like make a C99 wrapper and only interface to that to keep ABI compatibility. Actually does anyone know of a convenient library to assist in making one, and not by hand? Something akin to the GL/Horde3D API would be cool and would let you wrap it to any language too.
About containers, lots of hardcore game engine developers usually are huge fans of C99 for good reasons and just use constrained/minimal C++ features (templates/operator overloads/classes and minimal use of virtual functions in hot paths which is important to not cause pointer indirection which slows up in-order CPUs) and stay clear of the STD. For performance and fine-grained control reasons usually, like making explicit allocators for what you want to do without fragmentation, and for efficient cache use.
https://www.youtube.com/watch?v=rX0ItVEVjHc
#include <Urho3D/Urho3DAll.h>
#include <iostream>
using namespace std;
int main()
{
Context context;
SharedPtr<Component> a, b;
shared_ptr<Component> c, d;
const long long count = 1000000LL;
for (int k = 0; k < 10; k++)
{
HiresTimer timer;
timer.Reset();
for (long long i = 0; i < count; i++)
{
a = new Component(&context);;
b = a;
}
long long elapsed = timer.GetUSec(false);
cout << "Urho3D::SharedPtr " << elapsed << "\n";
timer.Reset();
for (long long i = 0; i < count; i++)
{
c = make_shared<Component>(&context);
d = c;
}
elapsed = timer.GetUSec(false);
cout << "std:shared_ptr " << elapsed << "\n";
}
// Prevent throw out pieces of code by compiler optimizer.
clog << "Ignore it: " << b.Null() << d.use_count();
// Prevent crash on exit when context deleted before objects.
a.Detach();
b.Detach();
}
maybe it’s because Urho uses intrusive reference counting, with everything inheriting from
RefCounted
?
So we’re getting 2 ref counts with
std::shared_ptr
instead of one.
To fix the comparison you’ll need to create a class that inherits from
RefCounted
for Urho, and the same class without inheriting from
RefCounted
for
std::shared_ptr
.
This is how I corrected the code:
#include <Urho3D/Urho3DAll.h>
#include <iostream>
using namespace std;
struct TestUrho : RefCounted
{
int x = 0;
};
struct Test
{
int x = 0;
};
int main()
{
SharedPtr<TestUrho> a, b;
shared_ptr<Test> c, d;
const long long count = 1000000LL;
for (int k = 0; k < 10; k++)
{
HiresTimer timer;
timer.Reset();
for (long long i = 0; i < count; i++)
{
a = MakeShared<TestUrho>();
b = a;
}
long long elapsed = timer.GetUSec(false);
cout << "Urho3D::SharedPtr " << elapsed << "\n";
timer.Reset();
for (long long i = 0; i < count; i++)
{
c = make_shared<Test>();
d = c;
}
elapsed = timer.GetUSec(false);
cout << "std:shared_ptr " << elapsed << "\n";
}
// Prevent throw out pieces of code by compiler optimizer.
clog << "Ignore it: " << b.Null() << d.use_count();
}
it seems that in this test
std::shared_ptr
is about 50% to 80% faster than Urho’s
SharedPtr
.
Note: with the old code
SharedPtr
was indeed faster.
1vanK
Refcounting preccessd only by SharedPtr class. In your example sizes of structs is different, so you have faster memory allocation for shared_ptr
Enhex
std::shared_ptr
adds ref count externally, and
Urho3D::SharedPtr
can only be used with classes that inherit from
Urho3D::RefCounted
, so my benchmark is correct.
void CrowdAgent::OnCrowdUpdate(dtCrowdAgent* ag, float dt)
{
...
// Use pointer to self to check for destruction after sending events
WeakPtr<CrowdAgent> self(this);
...
crowdManager_->SendEvent(E_CROWD_AGENT_REPOSITION, map);
if (self.Expired())
return;
...
}
Sinoid
No it isn’t. The STL way is more verbose but can do the same thing using enable_shared_from_this and dynamic_pointer_cast.
Eugene
TBH,
enable_shared_from_this
isn’t the silver bullet. No way to do things in ctor, while intrusive ptrs have no problem there.
1vanK
If I understand correctly, enable_shared_from_this stores shared_ptr inside function and to allow deleting shared_ptr outside we need crazy constructions inside function to get weak_ptr from shared_ptr and delete shared_ptr
Sinoid
Backwards. It only stores a weak_ptr internally, which means there’s no craziness involved.
Clarify? You can’t do a host of things in a constructor either way, virtual calls being the big one.
1vanK
enable_shared_from_this requires to store weak_ptr in object, so… we get same overhead for object size, as when inheriting from Urho3D: RefCounted
You can not recieve this weak_ptr directly. You can recieve only share_ptr, convert to weak_ptr and then delete shared_ptr to allow self-destruct object outside function and get correct num refs in weak_ptr inside function
I’ve been using STL-only in my local fork for quite a while, I haven’t had a single issue or encountered anything that could not be done that the Urho containers were doing.
The switch also made it a lot easier to work with 3rd-party libraries that are almost always going to be using STL as well as concurrency primitives that I had to have for multithreaded rendering.
Eugene
Cannot spawn weak/shared ptr from ctor.
And yes, it was a problem in my day job projects.
E.g. I was working with async server, and there’s no way to spawn tasks from ctor.
Enhex
it sounds a bit dangerous to use an object that wasn’t fully constructed yet, it’s kinda like as if it’s uninitialized.
You can always use some helper function that construct the object, then passes it to some other function.
BTW not all Urho objects need to use ref counting (ex: a subsystem with a known lifetime, usually until the program shuts down), and using std::shared_ptr gives the flexibility to choose not to.
Eugene
Meh, it’s
sometimes
dangerous. It wasn’t my case tho. Async task manager needs weak ptr just to automatically cancel tasks if owner is destroyed. And there’s no way to use it from ctor, despite it’s completely safe.
1vanK
As an experiment and optimization for cache locality, I moved refCounter from external structure into Object. WeakRefCounter still stay as external structure, so external counter created only when WeakPointers is used. In my tests, I did not see any difference of FPS. So I believe that pointers do not affect overall performance compared to other things — physics, rendering, and so on.
It’s a link to the modified source code? I had hoped to see some benchmark result or some kind of comparison result instead.
1vanK
I just launch original and modified samples and recieve same FPS.
S.L.C
Although you can’t really expect to see a noticeable difference in a hello-world type of application. And a synthetic benchmark doesn’t show much either. And since Urho doesn’t have a complex game/application built with it. You can’t do much but assume things.
As for the claim in the first post that:
Well,
std::string
is likely implemented with a small buffer optimization. And this here
auto s = to_string(i);
falls into that kind of optimization.
If you would’ve created a test that goes away from that optimization. And the only way i can think of doing it in order to preserve the same amount and order of operations would be to use an array of random strings, like:
You probably would’ve had a different outcome now. Because
std::string
is now a fatter structure because of that local buffer and also contains twice the validation, also because of that local buffer.
Which brings me back to my original point. The fact that you cannot look at a simple test like that and hope that result to reflect on a complex real world project.
And which brings me to my issue with using the standard library instead of built in structures. With which I have no problem. Just so I’m clear about that. But this is a project that targets many platforms and many compilers. Which brings the main issue with using STL. The fact that it’s unpredictable. Results from one implementation might not align with the results of another. Each change must take into consideration each implementation that could be used to compile this engine with. Whoever maintains the engine must have a clear view of multiple implementations instead of just one.
Some implementations could have a smaller buffer used for the small buffer optimization. Some could use a bigger one. Different alignment on one implementation can introduce padding into your structures. Some could use reference counting and copy on write instead. Imagine finding the cause of a bottleneck through all these different implementations (
don’t just assume there cannot be one, just because it’s a string. you don’t know what you need until you need it
). Not to mention that you’ve now introduced exceptions into the engine. Like i said, very unpredictable and time consuming.
Sure, you have tests and what not. But you broaden the scope of what developers must be familiar with.
And not to mention that you can kiss goodbye to dynamic libraries. If anyone relies on that.
I’m not saying STL is bad. I’m just saying that for a game engine with such a broad scope. It might prove to be a bit more headache than necessary.
That’s just my opinion. Moving into this direction wouldn’t affect me that much. And I’d love to see this happen and what outcome would there be. Pretty sure it’ll be a valuable lesson for everyone involved (
including me
).
rku
What do you mean? STL works just fine when used in dynamic libraries.
S.L.C
Maybe. I won’t argue that. But if built with the same compiler/options/STL implementation etc. Which kinda takes away from their usefulness.
That being said. I (personally) wouldn’t consider doing it. The conditions for it to work the way it should are quite narrow. At least that’s my opinion.