|
Help me! |
tremendoustoast
Member #15,522
February 2014
|
Hello, I have been working on a game for a few days now but keep coming across problems where it says unhandled exception. -"allegro-5.0.7-monolith-md-debug.pdb not loaded" I have included an image of the error it's throwing. Any help with this issue would be great because i just have no idea how to go about fixing it.:'( The program falls apart when this code is called for the second time; 1bool got_event = al_get_next_event(event_queue, &allegro_event);
2
3 if (got_event == false) return; // no event waiting
|
Edgar Reynaldo
Major Reynaldo
May 2007
|
tremendoustoast said:
I have been working on a game for a few days now but keep coming across problems where it says unhandled exception. Use your debugger and see what line it is crashing on. Your image says line 70. Show us your code with <code>code goes here...</code> tags. Quote:
-"allegro-5.0.7-monolith-md-debug.pdb not loaded" All it means is there is no associated pdb debugging file to go along with the dll. A pdb file contains debugging symbols so you can tell where it is crashing. Quote: The program falls apart when this code is called for the second time; If it is crashing on al_get_next_event, then allegro hasn't been initialized, or the event queue is invalid, or the address of the event is invalid. I don't think that is really what is wrong though. My Website! [members.allegro.cc] | EAGLE GUI Library Demos [members.allegro.cc] | My Deviant Art Gallery [nemonameless.deviantart.com] | Spiraloid Preview [members.allegro.cc] | A4 FontMaker | Skyline! (Missile Defense) [members.allegro.cc] Eagle and Allegro 5 binaries [github.com] | Older Allegro 4 and 5 binaries [bitbucket.org] | Allegro 5 compile guide [members.allegro.cc] |
tremendoustoast
Member #15,522
February 2014
|
Thanks for the reply, The code is for a class I've called AllegroStart. What i want the class to do is to handle the initialization of allegro as well as event handling, the problem i'm having has started occurring since trying to include a timer in the program. Here is the entire Class; 1#pragma once
2#include <allegro5/allegro.h>
3#include <allegro5/allegro_native_dialog.h>
4#include <string>
5#include<iostream>
6#include "conio.h"
7using namespace std;
8
9class AllegroStart
10{
11private:
12 enum KEYS_PRESSED_LIST
13 {
14 KEY_ESCAPE, KEY_W, KEY_A, KEY_S, KEY_D, KEY_F, SIZE_OF_KEYS_PRESSED_ENUM
15 };
16
17 bool keys_pressed[SIZE_OF_KEYS_PRESSED_ENUM];
18
19 ALLEGRO_DISPLAY* display = NULL;
20 ALLEGRO_EVENT_QUEUE* event_queue = NULL;
21 ALLEGRO_BITMAP* bitmap = NULL;
22 ALLEGRO_BITMAP* bitmap_with_transparency = NULL;
23 ALLEGRO_TIMER* timer = NULL;
24 int frames_per_second = 60;
25 bool update_logic = true;
26 bool render_frame = true;
27 int consecutive_logic_updates = 0;
28 int max_consecutive_logic_updates = 10;
29
30
31public:
32 // Method declarations
33 AllegroStart();
34 int initialise_allegro();
35 void process_events();
36 ALLEGRO_TIMER* StartTimer();
37 bool DoLogic();
38 bool DoRender();
39 bool PressedEscape();
40 bool PressedW();
41 bool PressedA();
42 bool PressedS();
43 bool PressedD();
44 bool PressedF();
45 ~AllegroStart();
46};
.cpp file 1#include "AllegroStart.h"
2
3AllegroStart::AllegroStart()
4{
5 frames_per_second = 60;
6 update_logic = true;
7 render_frame = true;
8 consecutive_logic_updates = 0;
9 max_consecutive_logic_updates = 10;
10}
11
12int AllegroStart::initialise_allegro()
13{
14 if (!al_init())
15 {
16 cout << "Allegro initialisation failed" << endl;
17 cout << "Press any key to exit" << endl;
18 _getch();
19 return -1;
20 }
21
22 // tell allegro program needs keyboard input
23 if (!al_install_keyboard())
24 {
25 cout << "Keyboard initialisation failed" << endl;
26 cout << "Press any key to exit" << endl;
27 _getch();
28 return -1;
29 }
30
31 // initialise keypress array
32 for (int i = 0; i < SIZE_OF_KEYS_PRESSED_ENUM; i++)
33 {
34 keys_pressed[i] = false;
35 }
36
37 display = al_create_display(800, 640); // create a display window of size 640 by 480
38
39 if (!display)
40 {
41 cout << "Display initialisation failed" << endl;
42 cout << "Press any key to exit" << endl;
43 _getch();
44 return -1;
45 }
46
47 al_set_window_position(display, 250, 50); // position display window on screen
48
49 // need to create an event queue so allegro can respond to events, such as keypresses
50 event_queue = al_create_event_queue();
51
52 if (!event_queue)
53 {
54 cout << "Event queue creation failed" << endl;
55 cout << "Press any key to exit" << endl;
56 _getch();
57 al_destroy_display(display);
58 return -1;
59 }
60
61 // register display with the event queue, so display change events can be processed
62 al_register_event_source(event_queue, al_get_display_event_source(display));
63
64 // register keyboard with the event queue, so keypress events can be processed
65 al_register_event_source(event_queue, al_get_keyboard_event_source());
66
67 // creates a timer
68 timer = al_create_timer(1.0 / frames_per_second);
69
70 //register Timer with the event queue
71 al_register_event_source(event_queue, al_get_timer_event_source(timer));
72
73 return 0;
74}
75
76void AllegroStart::process_events()
77{
78 ALLEGRO_EVENT allegro_event;
79
80 // get the next event from the event queue
81 bool got_event = al_get_next_event(event_queue, &allegro_event);
82
83 if (got_event == false) return; // no event waiting
84
85 // test if a key is pressed down, update global keypress array
86 if (allegro_event.type == ALLEGRO_EVENT_KEY_DOWN)
87 {
88 switch (allegro_event.keyboard.keycode)
89 {
90 case ALLEGRO_KEY_ESCAPE:
91 keys_pressed[KEY_ESCAPE] = true;
92 break;
93 case ALLEGRO_KEY_W:
94 keys_pressed[KEY_W] = true;
95 break;
96 case ALLEGRO_KEY_A:
97 keys_pressed[KEY_A] = true;
98 break;
99 case ALLEGRO_KEY_S:
100 keys_pressed[KEY_S] = true;
101 break;
102 case ALLEGRO_KEY_D:
103 keys_pressed[KEY_D] = true;
104 break;
105 case ALLEGRO_KEY_F:
106 keys_pressed[KEY_F]= true;
107 break;
108 }
109 }
110 // Test if a key is let go, update global keypress array
111 else if (allegro_event.type == ALLEGRO_EVENT_KEY_UP)
112 {
113 switch (allegro_event.keyboard.keycode)
114 {
115 case ALLEGRO_KEY_ESCAPE:
116 keys_pressed[KEY_ESCAPE] = false;
117 break;
118 case ALLEGRO_KEY_W:
119 keys_pressed[KEY_W] = false;
120 break;
121 case ALLEGRO_KEY_A:
122 keys_pressed[KEY_A] = false;
123 break;
124 case ALLEGRO_KEY_S:
125 keys_pressed[KEY_S] = false;
126 break;
127 case ALLEGRO_KEY_D:
128 keys_pressed[KEY_D] = false;
129 break;
130 case ALLEGRO_KEY_F:
131 keys_pressed[KEY_F] = false;
132 break;
133 }
134 }
135 else if(got_event && allegro_event.type == ALLEGRO_EVENT_TIMER)
136{
137
138// if logic taking too long ensures frame is rendered occasionally and input is responsive
139// this essentially ignores logic updates if there are a lot of consecutive ones
140
141 if(++consecutive_logic_updates < max_consecutive_logic_updates)
142 {
143 update_logic = true;
144 }
145
146 // only render frame if no events left to process
147 if(al_is_event_queue_empty(event_queue))
148 {
149 render_frame = true;
150 consecutive_logic_updates=0;
151 }
152}
153
154}
155
156ALLEGRO_TIMER* AllegroStart::StartTimer()
157{
158 return timer;
159}
160
161bool AllegroStart::DoLogic()
162{
163 if (update_logic)
164 {
165 return update_logic;
166 update_logic = false;
167 }
168}
169
170bool AllegroStart::DoRender()
171{
172 if(render_frame)
173 {
174
175 return render_frame;
176 render_frame = false;
177 }
178}
179bool AllegroStart::PressedEscape()
180{
181 return keys_pressed[KEY_ESCAPE];
182}
183
184bool AllegroStart::PressedW()
185{
186 return keys_pressed[KEY_W];
187}
188bool AllegroStart::PressedA()
189{
190 return keys_pressed[KEY_A];
191}
192bool AllegroStart::PressedS()
193{
194 return keys_pressed[KEY_S];
195}
196
197bool AllegroStart::PressedD()
198{
199 return keys_pressed[KEY_D];
200}
201
202bool AllegroStart::PressedF()
203{
204 return keys_pressed[KEY_F];
205}
206
207
208AllegroStart::~AllegroStart()
209{
210 al_destroy_display(display);
211 al_destroy_event_queue(event_queue);
212 al_destroy_timer(timer);
213}
|
Edgar Reynaldo
Major Reynaldo
May 2007
|
I haven't looked super closely but you may be making a copy of your AllegroStart object. If that happens, the destructor will be called which will shutdown allegro which will make any further calls to allegro functions fail. Show the code where you use your AllegroStart object. My Website! [members.allegro.cc] | EAGLE GUI Library Demos [members.allegro.cc] | My Deviant Art Gallery [nemonameless.deviantart.com] | Spiraloid Preview [members.allegro.cc] | A4 FontMaker | Skyline! (Missile Defense) [members.allegro.cc] Eagle and Allegro 5 binaries [github.com] | Older Allegro 4 and 5 binaries [bitbucket.org] | Allegro 5 compile guide [members.allegro.cc] |
tremendoustoast
Member #15,522
February 2014
|
AllegroStart is only used in my main.cpp source file, here is the code from that file (i know, it's messy, all a work it progress!!) 1#include <allegro5/allegro.h>
2#include <allegro5/allegro_native_dialog.h>
3#include<iostream>
4#include <vector>
5#include "conio.h"
6using namespace std;
7
8#include "AllegroStart.h"
9#include "Creature.h"
10#include "Projectile.h"
11#include "Hero.h"
12#include "Monster.h"
13#include "Arena.h"
14AllegroStart* alleg;
15Projectile* projectiles[4];
16Hero* Hero1;
17
18Monster* Monster1;
19Monster* Monster2;
20Monster* Monster3;
21Monster* Monster4;
22vector<Monster*> MonsterVector;
23Arena* Arena1;
24void Logic();
25void Render();
26
27int bum = 0;
28int man[4];
29int main()
30
31{
32 alleg = new AllegroStart();
33 Hero1 = new Hero(5, 100, 200, "assets/PlaceHolderDude.bmp", 200, 2, Arena1);
34 Arena1 = new Arena();
35 Monster1 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
36 Monster2 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
37 Monster3 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
38 Monster4 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
39
40 MonsterVector.push_back(Monster1);
41 MonsterVector.push_back(Monster2);
42 MonsterVector.push_back(Monster3);
43 MonsterVector.push_back(Monster4);
44
45 int x = 0;
46
47
48 while (x < 4)
49 {
50 projectiles[x] = new Projectile();
51 projectiles[x]->SetValues(10, 1, 1, 1, "assets/Bullet.bmp");
52 x++;
53 }
54
55 // initialise everything
56 bool init_ok = alleg->initialise_allegro();
57
58 if (init_ok != 0)
59 {
60 return init_ok; // exit if init failed
61 }
62
63 al_start_timer(alleg->StartTimer());
64
65 // loop until escape key is pressed
66 while (x<5) // changed this for debugging purposes!
67 {
68 alleg->process_events(); // deal with events, such as mouse and keyboard input
69
70 if(alleg->DoLogic())
71 {
72
73 Logic();
74 }
75 if(alleg->DoRender())
76 {
77 Render();
78 }
79
80 }
81
82 alleg->~AllegroStart();
83
84 delete alleg;
85
86 for (int count = 0; count < MonsterVector.size(); count++)
87 {
88
89 MonsterVector[count]->~Monster();
90 delete MonsterVector[count];
91 }
92
93 Hero1->~Hero();
94 delete Hero1;
95
96 int g = 0;
97 while (g < 4)
98 {
99 projectiles[g]->~Projectile();
100
101 g++;
102
103 }
104 return 0;
105}
106
107
108 void Logic()
109 {
110
111 // check keys pressed to update variables
112 if (alleg->PressedW())
113 {
114 Hero1->Up();
115 }
116
117 if (alleg->PressedA())
118 {
119 Hero1->Left();
120 }
121
122 if (alleg->PressedS())
123 {
124 Hero1->Down();
125 }
126
127 if (alleg->PressedD())
128 {
129 Hero1->Right();
130 }
131
132 if (alleg->PressedF())
133 {
134 if (bum == 0)
135 {
136
137 int y = 0;
138 int count = 1;
139
140 while (y<4)
141 {
142 if (count>0)
143 {
144 if (!projectiles[y]->isFiring())
145 {
146 projectiles[y]->Fire(1, Hero1->Getx(), Hero1->Gety());
147 bum = 5;
148 man[y] = 50;
149 count--;
150 }
151 }
152
153 y++;
154 }
155 }
156 }
157
158 if (bum > 0)
159 {
160 bum--;
161 }
162
163 int p = 0;
164 while (p < 4)
165 {
166 if (projectiles[p]->isFiring())
167 {
168 projectiles[p]->Firing();
169 if (man[p] == 1)
170 {
171 projectiles[p]->Stop(1);
172 }
173 }
174 p++;
175 man[p]--;
176 }
177
178 }
179
180void Render()
181 {
182 al_clear_to_color(al_map_rgb(0, 0, 0)); // colour entire display with rgb colour
183
184 Arena1->Render();
185 Hero1->Draw();
186 Monster1->Draw();
187
188 al_flip_display(); // show what has just been drawn
189 }
|
Edgar Reynaldo
Major Reynaldo
May 2007
|
Don't call destructors directly. delete takes care of calling the destructor for you. I don't have time to look over the rest of it now, but I will look again later. My Website! [members.allegro.cc] | EAGLE GUI Library Demos [members.allegro.cc] | My Deviant Art Gallery [nemonameless.deviantart.com] | Spiraloid Preview [members.allegro.cc] | A4 FontMaker | Skyline! (Missile Defense) [members.allegro.cc] Eagle and Allegro 5 binaries [github.com] | Older Allegro 4 and 5 binaries [bitbucket.org] | Allegro 5 compile guide [members.allegro.cc] |
tremendoustoast
Member #15,522
February 2014
|
That's great, thank you for the help so far! |
Edgar Reynaldo
Major Reynaldo
May 2007
|
Things are bound to go wrong because you it looks like you are loading assets before allegro is initialized. You create Hero and Monster objects before calling bool init_ok = alleg->initialise_allegro();. { alleg = new AllegroStart(); Hero1 = new Hero(5 , 100 , 200 , "assets/PlaceHolderDude.bmp" , 200 , 2 , Arena1); Arena1 = new Arena(); Monster1 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1); Monster2 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1); Monster3 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1); Monster4 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1); You need to insert your initialization code here : 1 alleg = new AllegroStart(); 2
3 // initialise everything
4 bool init_ok = alleg->initialise_allegro(); 5 if (init_ok != 0) {
6 return init_ok; // exit if init failed
7 }
8
9 Hero1 = new Hero(5 , 100 , 200 , "assets/PlaceHolderDude.bmp" , 200 , 2 , Arena1);
10 Arena1 = new Arena();
11 Monster1 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
12 Monster2 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
13 Monster3 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
14 Monster4 = new Monster(100, 200, "assets/Monster.bmp", 100, 2, Arena1);
My Website! [members.allegro.cc] | EAGLE GUI Library Demos [members.allegro.cc] | My Deviant Art Gallery [nemonameless.deviantart.com] | Spiraloid Preview [members.allegro.cc] | A4 FontMaker | Skyline! (Missile Defense) [members.allegro.cc] Eagle and Allegro 5 binaries [github.com] | Older Allegro 4 and 5 binaries [bitbucket.org] | Allegro 5 compile guide [members.allegro.cc] |
tremendoustoast
Member #15,522
February 2014
|
Hey, |
pkrcel
Member #14,001
February 2012
|
I found very difficult to go through your codeflow, but would you like to explain what this: bool AllegroStart::DoLogic() { if (update_logic) { return update_logic; update_logic = false; } } should accomplish? It is unlikely that Google shares your distaste for capitalism. - Derezo |
tremendoustoast
Member #15,522
February 2014
|
Ok so basically, i have broken tasks in main into logic tasks and render task and placed them in separate functions. That method will return true if logic tasks need to be performed and then the boolean attribute used to say whether logic tasks need to be performed will be set to false again. The reason i am using these functions is to allow the program extra time to process tasks if it is needed, this way you won't get a back log of render/logic tasks creating a more steady frame rate |
Thomas Fjellstrom
Member #476
June 2000
|
I think he's trying to hint at the fact that your logic in DoLogic is a bit flawed. -- |
tremendoustoast
Member #15,522
February 2014
|
Oh shit... man if i had a nickel for every time i did something like this. |
pkrcel
Member #14,001
February 2012
|
Well the fact is thay is only an example of the flawed points in your code. Which apart for some naive things like using a vector to store elements you...singularly allocated (I'm looking at those monsters....) it's not THAT bas as you state in the OP...there's just a bit of WAY TOO MUCH BOILERPLATE, but that's fine. The problem is I can't really follow along your code, how it "flows". You said that you get the SEGFAULT only the SECOND time you run over the 'got_event' bit...is this still the case after Edgar's suggestion? EDIT: tremendoustoast said: The reason i am using these functions is to allow the program extra time to process tasks if it is needed, this way you won't get a back log of render/logic tasks creating a more steady frame rate Barring my difficulties in understanding the code (which are there!) I don't think this would be the case since ouy're counting frames but not checking how much time you spend in logic. Anyway I would suggest you drop this kind of optimization for now, it's quite an advanced topic...unlikely you will need that now. On the other hand, I've (mentally) simulated two consecutive runs of the innner loop but can't find a good reason for al_get_next_event throw a fit without reason....it is very very very likely that the second time that is called there is no event (maybe also the first), but in that case that is clearly guarded but the bool check right after. I now strongly suspect there's an unmatched closing curly brace somewhere... It is unlikely that Google shares your distaste for capitalism. - Derezo |
tremendoustoast
Member #15,522
February 2014
|
Yes, -I have changed the code so that my initialise_allgro function is being called before any other class objects are created. 1int main()
2
3{
4 alleg = new AllegroStart();
5
6 // initialise everything
7 bool init_ok = alleg->initialise_allegro();
8
9 if (init_ok != 0)
10 {
11 return init_ok; // exit if init failed
12 }
13
14 Hero1 = new Hero(5, 100, 200, "assets/PlaceHolderDude.bmp", 200, 2, Arena1);
15 Arena1 = new Arena();
-I have changed the code so that update_logic and update_render are being set to false after the methods have been called(by creating 2 methods which just set the values to false when called) Game loop in main 1 alleg->process_events(); // deal with events, such as mouse and keyboard input
2
3 if(alleg->DoLogic())
4 {
5 Logic();
6 alleg->EndLogic();
7 }
8 if(alleg->DoRender())
9 {
10 Render();
11 alleg->EndRender();
12 }
Methods in AllegroStart 1bool AllegroStart::DoLogic()
2{
3 if (update_logic)
4 {
5 return update_logic;
6
7 }
8}
9
10bool AllegroStart::DoRender()
11{
12 if(render_frame)
13 {
14
15 return render_frame;
16 }
17}
18
19void AllegroStart::EndLogic()
20{
21 update_logic = false;
22}
23void AllegroStart::EndRender()
24{
25 render_frame = false;
26}
After changing this the program still crashes when it reaches bool got_event = al_get_next_event(event_queue, &allegro_event) |
pkrcel
Member #14,001
February 2012
|
Just in case, I was editing my previous post before you posted again. You may want to check. EDIT: well braces match it seems... It is unlikely that Google shares your distaste for capitalism. - Derezo |
tremendoustoast
Member #15,522
February 2014
|
ah, i'm following a tutorial and my understanding of how that piece of code works is clearly not right. Despite this I just can't see any reason why it would stop on that line of code... |
pkrcel
Member #14,001
February 2012
|
You could share you Visual Studio Solution, maybe someone (even me if I get back to my VS machine) can follow along the code. Maybe It is unlikely that Google shares your distaste for capitalism. - Derezo |
Dizzy Egg
Member #10,824
March 2009
|
I'm just guessing here, but would it make any difference if you started the timer before registering it for events? Probably not but you never know...
---------------------------------------------------- |
Edgar Reynaldo
Major Reynaldo
May 2007
|
tremendoustoast said:
1bool AllegroStart::DoLogic()
2 {
3 if (update_logic) {
4 return update_logic;
5 }
6 } 7 bool AllegroStart::DoRender() {
8 if(render_frame) {
9 return render_frame;
10 }
11 } 12 void AllegroStart::EndLogic() {
13 update_logic = false;
14 }
15 void AllegroStart::EndRender() {
16 render_frame = false;
17 }
You should turn on your compiler warnings. On both of the highlighted lines if update_logic or render_frame is false then the program execution will fall through to whatever is after it in the binary code. The execution pointer could end up anywhere, and it is likely to crash indeterminately. Your compiler should have said something like 'missing return statement for non-void function'. I think gcc would have warned about it with -Wall. I don't have a C++ compiler handy to test though. If that's still not it, post all of your current main again and maybe attach a zip of your sources. My Website! [members.allegro.cc] | EAGLE GUI Library Demos [members.allegro.cc] | My Deviant Art Gallery [nemonameless.deviantart.com] | Spiraloid Preview [members.allegro.cc] | A4 FontMaker | Skyline! (Missile Defense) [members.allegro.cc] Eagle and Allegro 5 binaries [github.com] | Older Allegro 4 and 5 binaries [bitbucket.org] | Allegro 5 compile guide [members.allegro.cc] |
pkrcel
Member #14,001
February 2012
|
I strongly suspect that compiler optimizations would make the thing inlined and thus MAYBE leave things unchanged with no harm. But I completely missed that due to cross-posting...whatever the crash, those two functions won't do any good cause there's nowhere the properties are being set again (to true). I second Edgar's suggestion, maybewe are able to run the sources into a debugger... It is unlikely that Google shares your distaste for capitalism. - Derezo |
tremendoustoast
Member #15,522
February 2014
|
{"name":"GoodNewsEveryone.jpg","src":"\/\/djungxnpq2nug.cloudfront.net\/image\/cache\/c\/6\/c6698fa12292b680f739bfd436f0731e.jpg","w":990,"h":749,"tn":"\/\/djungxnpq2nug.cloudfront.net\/image\/cache\/c\/6\/c6698fa12292b680f739bfd436f0731e"} EDIT |
|