Crash from OSLock in C++ constructor...?

Discussion to talk about software related topics only.
Post Reply
Ridgeglider
Posts: 513
Joined: Sat Apr 26, 2008 7:14 am

Crash from OSLock in C++ constructor...?

Post by Ridgeglider »

I have a C++ class where one of the private objects is an OS_CRIT object. This is followed by a large struct for which the OS_CRIT object is intended to control access. The class constructor inits the OS_CRIT object. Once inited, (and still in the constructor) I try to lock access by calling an OSCriticalSection after, which I intended to (now safely) init the struct. Unfortunately this causes a hard crash.

If I avoid calling an OSCriticalSection in the constructor and just init the struct without locking access, things are OK. Furthermore, once the constructor is done, in other methods, I can successfully call OSCritical sections to protect access. But calling an OSCrit section from within the constructor produces an instant crash. I don't see why. I'd like to call it in order to safely init the gps struct. Am I missing something? I tried adding a short delay between Initing the OSCRIT object before calling the OSCritSection macro, but no luck. Partial code follows...

Thanks

Code: Select all

class GPS_C {
	private:
		bool bGPS_LockInited;
		OS_CRIT GPS_Lock;

        gps_t gps	// a long struct of stuff needing inits...

	public:
        GPS_C();						//The constructor

	//Methods for loading data into the gps struct object:============================
		void StuffDataAfterParsingNMEA_GPGGA( gpgga_t & rGPGGA);
		//etc...
	//Methods for getting data out of the gps struct object:=======================
		void GetNMEA_GPGGA(  gpgga_t & rGPGGA);
		//etc...
}

//The constructor for the GPS_ Class--------------------------------------
 GPS_C::GPS_C() {

	 // init the OSCrit section only once...redundant, I know.
	 bGPS_LockInited = FALSE;

 	if (bGPS_LockInited == FALSE) {
 		if (OSCriticalSectionObj oscs(GPS_Lock);(&GPS_Lock) == OS_NO_ERR) {
 			bGPS_LockInited = TRUE ;
 			iprintf("GPS_Lock Init SUCCESSFUL\r\n"   );
 		} else {
 			bGPS_LockInited = FALSE ;
 			iprintf("GPS_Lock Init FAILED\r\n"   );
 		}
 	}

	// now that GS_Lock object (of type OS_CRIT) is inited, lock the
    // upcoming section of code in order to safely init the gps struct..
    //
    // The crash happens here:
	OSCriticalSectionObj oscs(GPS_Lock);
    // If the OSCriticalSectionObj line above is commented, things are OK, although the
    // init below happens without any protection...
	init_gps_struct( &gps, knots, degrees);
 }
Ridgeglider
Posts: 513
Joined: Sat Apr 26, 2008 7:14 am

Re: Crash from OSLock in C++ constructor...?

Post by Ridgeglider »

Sorry...the code in the previous post got a little messed up...
here's a revision:

Code: Select all

class GPS_C {
   private:
      bool bGPS_LockInited;
      OS_CRIT GPS_Lock;

        gps_t gps   // a long struct of stuff needing inits...

   public:
        GPS_C();                  //The constructor

   //Methods for loading data into the gps struct object:============================
      void StuffDataAfterParsingNMEA_GPGGA( gpgga_t & rGPGGA);
      //etc...
   //Methods for getting data out of the gps struct object:=======================
      void GetNMEA_GPGGA(  gpgga_t & rGPGGA);
      //etc...
}

//The constructor for the GPS_ Class--------------------------------------
GPS_C::GPS_C() {

    // init the OSCrit section only once...redundant, I know.
 	bGPS_LockInited = FALSE;

 	if (bGPS_LockInited == FALSE) {
 		if (OSCritInit (&GPS_Lock) == OS_NO_ERR) {
 			bGPS_LockInited = TRUE ;
 			iprintf("GPS_Lock Init SUCCESSFUL\r\n"   );
 		} else {
 			bGPS_LockInited = FALSE ;
 			iprintf("GPS_Lock Init FAILED\r\n"   );
 		}
 	}

   // now that GS_Lock object (of type OS_CRIT) is inited, lock the
    // upcoming section of code in order to safely init the gps struct..
    //
    // The crash happens here:
   OSCriticalSectionObj oscs(GPS_Lock);
    // If the OSCriticalSectionObj line above is commented, things are OK, although the
    // init below happens without any protection...
   init_gps_struct( &gps, knots, degrees);
}

 
User avatar
lgitlitz
Posts: 331
Joined: Wed Apr 23, 2008 11:43 am
Location: San Diego, CA
Contact:

Re: Crash from OSLock in C++ constructor...?

Post by lgitlitz »

Are you using this class to make a global object? If you are then it will be constructed before the OS is started, so no OS calls can be made. What you should do instead is set the SR register to prevent interrupts, which will prevent any task switching. There are functions to read and write the SR in cfinter.h.

//To lock
WORD Current_Int_Level = ( ( GetSR_IntLevel & 0x0F00 ) >> 8 ); // Gets SR value and converts to interrupt level.
SetSR_IntLevel( 7 ); // Sets SR so only un-maskable interrupts can occur

//To Unlock
SetSR_IntLevel( Current_Int_Level );

This will provide you with higher protection since you are blocking task switching and interrupt switching. The SR should not normally be changed by user code but this is one exception where there is not much of a choice. If you ever wish to use task and interrupt blocking elsewhere to protect a variable you can use:
USER_ENTER_CRITICAL();
USER_EXIT_CRITICAL();
These functions will not work in your case since they rely on an OS variable that is not yet initialized but should work for all other cases after the OS is started.

-Larry
User avatar
lgitlitz
Posts: 331
Joined: Wed Apr 23, 2008 11:43 am
Location: San Diego, CA
Contact:

Re: Crash from OSLock in C++ constructor...?

Post by lgitlitz »

I am going to try a fix for this so OS objects can be used in global constructors. It will likely be in the next release.
Ridgeglider
Posts: 513
Joined: Sat Apr 26, 2008 7:14 am

Re: Crash from OSLock in C++ constructor...?

Post by Ridgeglider »

Larry:

As usual, you're right on it. Thanks. After sitting on this for a bit, but before your response I figured something like this was the issue. Of course you saved hours of hunting.

I modified your code very slightly to avoid errors:
WORD Current_Int_Level = ( ( GetSR_IntLevel() & 0x0F00 ) >> 8 );

You also mention USER_ENTER_CRITICAL...

You are also correct in thinking that the class is a global object. In additon to the actual data, the class includes the object

Code: Select all

OS_CRIT GPS_Lock;
OS_CRIT GPS_Lock;
I have a variety of methods to access data in the class. Each of these methods calls

Code: Select all

OSCriticalSectionObj oscs(GPS_Lock);
before accessing and returning the data. This set up relieves me from remebering the unlock and does it automatically upon exiting the method. Still fuzzy on how this is different from using the USER_ENTER_CRITICAL(); USER_EXIT_CRITICAL(); pairs....I've used this strategy quite a bit and it seems to work. A work around was to mark the class as uninited, and once it was up and runnig for a sec, to call an non-constructor-located data init routine.




Thanks again.
thomastaranowski
Posts: 82
Joined: Sun May 11, 2008 2:17 pm
Location: Los Angeles, CA
Contact:

Re: Crash from OSLock in C++ constructor...?

Post by thomastaranowski »

Larry, that would be a great add. I ran into that issue awhile back, and it took awhile to debug and workaround.
User avatar
lgitlitz
Posts: 331
Joined: Wed Apr 23, 2008 11:43 am
Location: San Diego, CA
Contact:

Re: Crash from OSLock in C++ constructor...?

Post by lgitlitz »

I think this would be a good add too. I need to experiment with a few things to see how much needs to be changed. The constructor call needs to be pulled out of the CRT and put in after the OS is initialized but probably before it is started. This will be a big enough change that it will probably require an upgrade instead of a simple code patch.

-Larry
bbracken
Posts: 54
Joined: Mon Jun 23, 2008 11:21 am
Location: Southern California
Contact:

Re: Crash from OSLock in C++ constructor...?

Post by bbracken »

I have had similar issues. To solve the issue I create all instances of objects at compile time. For an object that holds on to other classes, I create an init message that initializes the high level object as well as those objects that it references (i.e, the ones created at compile time). Typically the init message sets a pointer to point to objects that are internallly referenced.

I would like to see this corrected also.

bb
Post Reply