-
Notifications
You must be signed in to change notification settings - Fork 10
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
Speed up terrain drawing with glDrawArrays from OpenGL 2.0. #120
base: master
Are you sure you want to change the base?
Conversation
Nice to see that there is optimization work in progress! |
Sorry, I forgot to add the glad files to the PR. I downloaded the zip from Glad generator but forgot to extract and commit it. |
On second thought, maybe Epoxy or OpenGL Extension Wrangler would be better options. The glad header and source file the website generated is over 4,000 lines and the license for the generated code is unclear. Glad is MIT licensed, the XML file it uses to generate the code is Apache licensed, and I don't know what license the generated code is under. I found this though: License of generated code. I got it working using glad, but then decided to switch to epoxy. |
Rebased onto |
@stephanemagnenat My latest commit changes Please let me know if you need me to squash the several "Fix foo" commits into one or remove the old commits referencing PS: FPS measured with this patch. |
Fixed flags, zones, fog of war, and bullets not being drawn. |
@Quipyowert2 is this PR completed? If so, yes, please squash commits. I have some questions though:
|
The PR is just about done, but I had to add some missing finishDrawingSprite calls because some of the right panel items weren't being drawn. Regarding the first bullet point, Bullet point 2, yeah optional is probably a good idea here. Re commented out code: The atlas code in initTextureSize is commented out because I was testing something and forgot to remove it. The line setting I'm not sure about what happens if you call drawPixel on a sprite-sheet. I might have to test that tomorrow. |
Ok, thank you for your quick answer. Please ping me when everything is done and you want this merged! |
65a2988
to
bfee7fe
Compare
I have pushed a commit to prevent drawing to the Sprite::atlas surface after it has been generated. It is a little difficult to work on graphics programming right now as my laptop screen has started turning purple and flickering after every reboot for 15 minutes to several hours |
bfee7fe
to
0dfc8bd
Compare
libgag/src/Sprite.cpp
Outdated
} | ||
int tileWidth = images[0]->getW(); | ||
int tileHeight = images[0]->getH(); | ||
int sheetWidth = tileWidth * (static_cast<int>(sqrt(numImages)) + 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.
I should probably use a loop that tries all the possible width and height combinations here where width and height are <= the maximum texture size (glGetIntegerv(GL_MAX_TEXTURE_SIZE)
). May need multiple sprite sheets if the maximum texture size is too small.
I extended createAtlas to be able to handle a limited texture size, but I haven't pushed my changes to this branch yet. I have it on another branch (sprite-sheet-after-rebase). I thought I squashed the Fix stuff not being drawn in map editor commit into the other similar "Fix" commits, but apparently I didn't. I tried using std::optional, but it wouldn't compile until I changed Visual Studio's C++ standard to C++17. Then it complained about mem_fun, which was available in C++14 (the default standard in VS, I think), but has been removed in 17. Things we use that have been removed in C++17:
|
Buildings are drawn in three places now for some odd reason. Fix building showing up three times bug.
Doesn't compile yet because of a purportedly missing > when compiling native.h.
Creating and deleting two std::vectors for every sprite draw during every frame took a total of 7s over a profiling run of about 3 minutes. Fix null pointer deref if no sprite or atlas. Fix terrain drawing in-game. Fix nothing to draw causing crash in finishDrawingSprite.
Fix water not being drawn. Fix flags, zones, and building rendering. Fix fog of war; missing icons when placing building. Fix brush selection not showing up in flag view. Draw zone animations. Fix bullets not showing up. Fix for some sprites not being drawn. Fix some sprites not being drawn in map editor.
Don't create a sprite sheet for hue-shifted sprites. Check that we have at least one non-NULL image. If all the images in the array are NULL, we don't need to create a texture atlas/sprite sheet.
Add Epoxy to Debian build dependencies. Look for epoxy in SConstruct. Link against Epoxy.
Changes Sprite::atlas variable to be a unique_ptr and adds a bunch of preprocessor macros to find the right make_unique for the C++ and Boost version. Prevent drawing on atlas once generated by making it readonly.
This also makes this particular function follow Law of Demeter.
opt: use min_element instead of sort. We don't use any of the calculated dimensions except for the first one.
0dfc8bd
to
fae47c2
Compare
I used boost::optional instead of std::optional to avoid requiring C++17. I ended up moving What do you think, should I keep finishDrawingSprite in the GC class or move it to the Sprite class? The reason I am using two-dimensional vectors in the drawing code instead of 1D is that I was thinking that the graphics card might have a small max texture size, so would need multiple sprite-sheets to hold the entire sprite. Although OpenGL 2.0 supports at least 64x64 texture size (comment), my laptop's GTX 1060 supports textures up to a size of 32768, and the onboard Intel 630 graphics support textures with a size of 16384. What is the optimal texture size says to use 2k textures. Maybe YAGNI? Probably should make it a one-dimensional vector, I guess. |
Generates a sprite sheet on the fly, queues UV and vertex coordinates in GC::drawSurface and finally in finishDrawingSprite, many different sprites in the texture atlas can be drawn at the same time with glDrawArrays.
With VSync off and after commenting out the
SDL_Delay
s in GUIBase.cpp and Engine.cpp, I get a frame rate of about 500 on the title screen and around 100 or so in tutorial 4.With the new sprite sheet code, the frame rate is roughly ~60 FPS without the two SDL_Delay calls. It was only about ~40 FPS before I optimized it, so about a 50% increase in FPS.
Note: This PR adds a new dependency,
gladepoxy. The functions createTextureAtlas and finishDrawingSprite require OpenGL 2.0.TODO: