Page 1 of 2
					
				NXC: program hangs up when resizing arrays
				Posted: 14 Apr 2013, 08:43
				by HaWe
				hey,
maybe another NXC bug now when using ArrayInit and ArrayBuild for risizing arrays of string - 
the program blocks completely,
the BCC program window freezes,
the NXT screen shows "
testprogram running"
the NXT is clicking (like if in SAMBA mode):
Code: Select all
// Array Resize
#define ArrRze(&_array_[], _temp_[], _newsize_)  {  \
   int _oldlen_ = ArrayLen(_array_);                   \
   if(_newsize_ > _oldlen_) {                          \
     int _diffsize_ = _newsize_ - _oldlen_;            \
     ArrayInit (_temp_, 0, _diffsize_ );               \
     ArrayBuild(_array_, _array_, _temp_);             \
   }                                                   \
   else                                                \
   if(_newsize_ < _oldlen_) {                          \
     ArraySubset(_array_, _array_, 0, _newsize_);      \
   }                                                   \
}
#define ArrayResize(a,b)  ArrRze(a,a,b)
string array1[2];
task main()
{
   array1[0] = "ok1";
   array1[1] = "ok2";
   ArrayResize(array1, 4);  //
   TextOut(0, 48, array1[0]);     // ok1
   TextOut(0, 40, array1[1]);     // ok2
   TextOut(0, 32, array1[2]);     // ""
   TextOut(0, 24, array1[3]);     // ""
   Wait(10000);
}
only removing the batteries can reboot the brick.
 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 17 Apr 2013, 06:40
				by HaWe
				another issue now  when resizing arrays of int -
no program hang-up or block but it produces wrong results after resizing:
The array cells get muddled up and show values of fantasy and imagination instead of the (expected) initialized values:
The previously assigned values (101, 102) are overwritten to zero and the expanded cells show (big) negative values instead of zero:
Code: Select all
#define ArrRze(&_array_[], _temp_[], _newsize_)  {  \
   int _oldlen_ = ArrayLen(_array_);                   \
   if(_newsize_ > _oldlen_) {                          \
     int _diffsize_ = _newsize_ - _oldlen_;            \
     ArrayInit (_temp_, 0, _diffsize_ );               \
     ArrayBuild(_array_, _array_, _temp_);             \
   }                                                   \
   else                                                \
   if(_newsize_ < _oldlen_) {                          \
     ArraySubset(_array_, _array_, 0, _newsize_);      \
   }                                                   \
}
#define ArrayResize(a,b)  ArrRze(a,a,b)
int  array1[2];
#define CLREOL DRAW_OPT_CLEAR_EOL
task main()
{
   array1[0] = 101;
   array1[1] = 102;
   ArrayResize(array1,4);                // shows:
   NumOut(0, 48, array1[0], CLREOL);     // 0   instead of 101
   NumOut(0, 40, array1[1], CLREOL);     // 0   instead of 102
   NumOut(0, 32, array1[2], CLREOL);     //  -16657 instead of 0
   NumOut(0, 24, array1[3], CLREOL);     //  -8531  instead of 0
   Wait(10000);
}
Using a global temporary array as a buffer instead, the old values stay correct as assigned before (101, 102) but nevertheless the new expanded cells show the same irrational values instead of zero. 
Code: Select all
#define ArrRze(&_array_[], _temp_[], _newsize_)  {  \
   int _oldlen_ = ArrayLen(_array_);                   \
   if(_newsize_ > _oldlen_) {                          \
     int _diffsize_ = _newsize_ - _oldlen_;            \
     ArrayInit (_temp_, 0, _diffsize_ );               \
     ArrayBuild(_array_, _array_, _temp_);             \
   }                                                   \
   else                                                \
   if(_newsize_ < _oldlen_) {                          \
     ArraySubset(_array_, _array_, 0, _newsize_);      \
   }                                                   \
}
// #define ArrayResize(a,b)  ArrRze(a,a,b)
int  array1[2], temp[];
#define CLREOL DRAW_OPT_CLEAR_EOL
task main()
{
   array1[0] = 101;
   array1[1] = 102;
   ArrRze(array1, temp, 4);                // shows:
   NumOut(0, 48, array1[0], CLREOL);     //  101  ok
   NumOut(0, 40, array1[1], CLREOL);     //  102  ok
   NumOut(0, 32, array1[2], CLREOL);     //  -16657 instead of 0
   NumOut(0, 24, array1[3], CLREOL);     //  -8531  instead of 0
   Wait(10000);
}
supposed to be a NXC bug...  

 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 17 Apr 2013, 16:04
				by mattallen37
				Try stripping the program down to the point that it's only a couple lines, but still demonstrates the problem (put everything in the main task, rather than using a function and a macro wrapper to that function). Usually I can find the problem by doing so, and even if not, it makes it easier for other people to debug it.
I have a feeling it has to do with the way you are passing the array through the macro, into the function, back out to the macro, and then back into your program. I suggest you try an alternative way of doing this, and see if it improves performance.
			 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 17 Apr 2013, 16:57
				by HaWe
				thank you, Matt, 
originally it's meant for a library array function, 
but written top/down it's all the same:
Code: Select all
int  array1[2], temp[];
#define CLREOL DRAW_OPT_CLEAR_EOL
task main()
{
   array1[0] = 101;
   array1[1] = 102;
   int _oldlen_ = ArrayLen(array1);
   int _newsize_ = 4;
   
   if(_newsize_ > _oldlen_) {
     int _diffsize_ = _newsize_ - _oldlen_;
     ArrayInit (temp, 0, _diffsize_ );
     ArrayBuild(array1, array1, temp);
   }
   else
   if(_newsize_ < _oldlen_) {
     ArraySubset(array1, array1, 0, _newsize_);
   }
   // ArrRze(array1, temp, 4);                // shows:
   NumOut(0, 48, array1[0], CLREOL);     //  101  ok
   NumOut(0, 40, array1[1], CLREOL);     //  102  ok
   NumOut(0, 32, array1[2], CLREOL);     //  -16657 instead of 0
   NumOut(0, 24, array1[3], CLREOL);     //  -8531  instead of 0
   Wait(10000);
}
interesting side effect:
if I choose int _newsize_ = 2 then it ends with a file error -1  

 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 17 Apr 2013, 17:02
				by mattallen37
				What if you keep stripping lines out? Try adding some debugging (NumOut usually works well).
I'm pretty busy, but I might be able to look into it a bit. I'll let you know if I happen to find anything.
			 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 17 Apr 2013, 17:04
				by HaWe
				there are no lines to be stripped out.
it's simple as it could be:
- define an array[2]
- define a new size
- decide what to do either if new size is larger or smaller than old size
- do it!
- show it!
I guess there are API function issues concerning ArrayBuild and ArrayInit.
			 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 17 Apr 2013, 20:34
				by mattallen37
				ArrayBuild doesn't seem to like the destination to be the same as any of the sources (which is likely a bug).
This seems to work fine:
Code: Select all
#define ArrayResize(& Array[], NewSize){     \
  int OldSize = ArrayLen(Array);             \
  if(NewSize > OldSize){                     \
    int DiffSize = NewSize - OldSize;        \
    int Array_Temp[];                        \
    ArrayInit(Array_Temp, 0, DiffSize);      \
    int Array_In[];                          \
    ArraySubset(Array_In, Array, 0, NA);     \
    ArrayBuild(Array, Array_In, Array_Temp); \
  }                                          \
  else if(NewSize < OldSize){                \
    ArraySubset(Array, Array, 0, NewSize);   \
  }                                          \
}
When NewSize is greater than the length of the input array, it creates Array_Temp with the number of elements to make up the difference, it copies Array into Array_In, and it builds Array from Array_In and Array_Temp.
 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 18 Apr 2013, 03:02
				by afanofosc
				ArrayBuild cannot use the output as an input, which makes sense and is not a bug.  The output array is reallocated by the firmware before data is copied into it and that replaces the pointer to the array contents in memory with the newly allocated address so it can't read from the original source array anymore.
John Hansen
			 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 18 Apr 2013, 08:04
				by HaWe
				thanks to both of you!
I'll now have to try the following code -  to keep the array variable type universal, the original array will have to be passed 3 times to the macro.
(can't test it for the moment but will report):
Code: Select all
#define ArrRze(&_array_[], _arrold_[], _temp_[], _newsize_)  {  \
   int _oldlen_ = ArrayLen(_arrold_);                  \
   if(_newsize_ > _oldlen_) {                          \
     int _diffsize_ = _newsize_ - _oldlen_;            \
     ArrayInit (_temp_, 0, _diffsize_ );               \
     ArrayBuild(_array_, _arrold_, _temp_);            \
   }                                                   \
   else                                                \
   if(_newsize_ < _oldlen_) {                          \
     ArraySubset(_array_, _arrold_, 0, _newsize_);     \
   }                                                   \          
  ArrayInit (_temp_, 0, 0 );                           \
  ArrayInit (_arrold_, 0, 0 );                         \
}                                                        
#define ArrayResize(a,b)  ArrRze(a,a,a,b)
int array1[2];
task main()
{
   array1[0] = 101;
   array1[1] = 102;
   ArrayResize(array1, 5);  //
   NumOut(0, 48, array1[0]);     //
   NumOut(0, 40, array1[1]);     //
   NumOut(0, 32, array1[2]);     //
   NumOut(0, 24, array1[3]);     //
   Wait(10000);
}
 
			
					
				Re: NXC: program hangs up when resizing arrays
				Posted: 18 Apr 2013, 17:36
				by HaWe
				trying the code above I get file error -1
if I delete the last both ArrayInits I still get faulty results:
0
0
0
-8531
instead of
101
102
0
0
Code: Select all
#define ArrRze(&_array_[], _arrold_[], _temp_[], _newsize_)  {  \
   int _oldlen_ = ArrayLen(_arrold_);                  \
   if(_newsize_ > _oldlen_) {                          \
     int _diffsize_ = _newsize_ - _oldlen_;            \
     ArrayInit (_temp_, 0, _diffsize_ );               \
     ArrayBuild(_array_, _arrold_, _temp_);            \
   }                                                   \
   else                                                \
   if(_newsize_ < _oldlen_) {                          \
     ArraySubset(_array_, _arrold_, 0, _newsize_);     \
   }                                                   \
}
#define ArrayResize(a,b)  ArrRze(a,a,a,b)
int array1[2];
task main()
{
   array1[0] = 101;
   array1[1] = 102;
   ArrayResize(array1, 5);  //  shows
   NumOut(0, 48, array1[0]);     //   0
   NumOut(0, 40, array1[1]);     //   0
   NumOut(0, 32, array1[2]);     //   0
   NumOut(0, 24, array1[3]);     //   -8531
   Wait(10000);
}
There must be an unexpected NXC bug - or two...