<span style=" font-size:10pt;font-family:sans-serif">Hi RTEMS developers.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">What follows is
a detailed bug report - and a suggested fix - regarding the impact on calls
to "__gcov_init" from a patch made to sparc's linkscmds file
back in 2014.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">The executive
summary of the problem:  In the process of running some coverage tests
on a GR740 board for the qualified math library for space, it became clear
that even though the code was compiled with "<i>-ftest-coverage</i>"
and "<i>-fprofile-arcs</i>", there were no calls made upon startup
towards "<i>__gcov_init</i>". The coverage checks depended on
that, because these calls pass pointers to structures that hold key information
related to coverage.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">In similar experiments
done with N2X boards and older version of the tools (<i>gcc 4.8 and RTEMS4.11</i>),
these automated calls to "<i>__gcov_init</i>" were properly made
during startup, as expected. </span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">In attempting
to debug this failure, the current structure of the code was then followed.
 In our - admittedly - limited understanding, it appears that the
compiler-generated "stubs" that call "<i>__gcov_init</i>"
are in the end meant to exist between the __CTOR_LIST__ and __CTOR_END__
symbols in the final binary - in the form of function pointers (<i>i.e.
their addresses are meant to exist in this "list"</i>).  </span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">To give a more
specific example, on an N2X board that was running the binaries built with
the older RTEMS toolchain, the CTOR_LIST ended up with content like this:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">$ sparc-rtems4.11-objdump
-x -d -S ../bin.debug.leon3.OAR/fputest</span>
<br><span style=" font-size:10pt;font-family:Courier New">...</span>
<br><span style=" font-size:10pt;font-family:Courier New">4001d100 <__CTOR_LIST__>:</span>
<br><span style=" font-size:10pt;font-family:Courier New">4001d100:  
    ff ff ff ff 40 00 13 ac 40 00 14 54 40 00 18 54    
....@...@..T@..T</span>
<br><span style=" font-size:10pt;font-family:Courier New">4001d110:  
    40 00 1b 38 40 00 22 4c 40 00 25 c0      
          @..8@."L@.%.</span>
<br><span style=" font-size:10pt;font-family:Courier New">4001d11c <__CTOR_END__>:</span>
<br><span style=" font-size:10pt;font-family:Courier New">4001d11c:  
    00 00 00 00              
                     
    ....</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">Notice that the
list start with 0xffffffff (i.e. -1) and proceeds with 400013ac, 40001454,
etc. </span>
<br><span style=" font-size:10pt;font-family:sans-serif">And here's the
coverage-generated "stub" that lives at 400013ac - and will call
"__gcov_init":</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">...</span>
<br><span style=" font-size:10pt;font-family:Courier New">400013ac:  
    9d e3 bf a0     save  %sp, -96, %sp</span>
<br><span style=" font-size:10pt;font-family:Courier New">400013b0:  
    03 10 00 7a     sethi  %hi(0x4001e800), %g1</span>
<br><span style=" font-size:10pt;font-family:Courier New">400013b4:  
    90 10 60 50     or  %g1, 0x50, %o0  
   ! 4001e850 <__gcov_.Init+0x18></span>
<br><span style=" font-size:10pt;font-family:Courier New">400013b8:  
    40 00 03 ad     call  4000226c <__gcov_init></span>
<br><span style=" font-size:10pt;font-family:Courier New">400013bc:  
    01 00 00 00     nop</span>
<br><span style=" font-size:10pt;font-family:Courier New">400013c0:  
    81 e8 00 00     restore</span>
<br><span style=" font-size:10pt;font-family:Courier New">400013c4:  
    81 c3 e0 08     retl</span>
<br><span style=" font-size:10pt;font-family:Courier New">400013c8:  
    01 00 00 00     nop</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">When we built
the same code with either (a) a very recent toolchain from the mainline
(RTEMS 5, RSB 703532cb04c6990fb21e97cb7347a16e9df11108 ) or (b) using the
RCC1.3rc3 toolchain from Gaisler - in both cases targetting the GR740 -
the result was different:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">000242c0 <__CTOR_LIST__>:</span>
<br><span style=" font-size:10pt;font-family:Courier New">   242c0:
      ff ff ff ff            
                     
      ....</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">000242c4 <__CTOR_END__>:</span>
<br><span style=" font-size:10pt;font-family:Courier New">   242c4:
      00 00 00 00 00 00 14 60 00 00 15 24 00 00 19 cc  
  .......`...$....</span>
<br><span style=" font-size:10pt;font-family:Courier New">   242d4:
      00 00 1d 34 00 00 24 74 00 00 28 08    
            ...4..$t..(.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">It appears that
for some reason the list of stubs ends up somehow accumulated on the CTOR_END
instead of the CTOR_LIST - indeed, 00001460 is a coverage "stub"
that is meant to call __gcov_init:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">00001460 <_GLOBAL__sub_I_00100_0__Linker_set__Sysinit_rtems_filesystem_initialize>:</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
1460:       9d e3 bf a0     save  %sp, -96,
%sp</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
1464:       03 00 00 96     sethi  %hi(0x25800),
%g1</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
1468:       90 10 61 20     or  %g1, 0x120,
%o0     ! 25920 <__gcov_.Init+0x18></span>
<br><span style=" font-size:10pt;font-family:Courier New">   
146c:       40 00 04 10     call  24ac <__gcov_init></span>
<br><span style=" font-size:10pt;font-family:Courier New">   
1470:       01 00 00 00     nop</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
1474:       81 e8 00 00     restore</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
1478:       81 c3 e0 08     retl</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
147c:       01 00 00 00     nop</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">But unfortunately,
no-one calls this stub - because the "list processing code" remains
the same - it starts at CTOR_END-4, and goes back until it meets a -1:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">grmon2> bp
__do_global_ctors_aux</span>
<br><span style=" font-size:10pt;font-family:Courier New">  Software
breakpoint 1 at <__do_global_ctors_aux></span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">grmon2> run</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">  CPU 0:
 breakpoint 1 hit</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
      0x000241b4: 3b000090  sethi  %hi(0x24000),
%i5</span>
<br><span style=" font-size:10pt;font-family:Courier New">  CPU 1:
 Power down mode</span>
<br><span style=" font-size:10pt;font-family:Courier New">  CPU 2:
 Power down mode</span>
<br><span style=" font-size:10pt;font-family:Courier New">  CPU 3:
 Power down mode</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">grmon2> disassemble
0x000241b4</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241b4:
      3b 00 00 90     sethi  %hi(0x24000),
%i5</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241b8:
      ba 17 62 c4     or  %i5, 0x2c4, %i5
    ! 242c4 <__CTOR_END__></span>
<br><span style=" font-size:10pt;font-family:Courier New">   241bc:
      c2 07 7f fc     ld  [ %i5 + -4 ], %g1</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241c0:
      80 a0 7f ff     cmp  %g1, -1</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241c4:
      02 80 00 08     be  241e4 <__do_global_ctors_aux+0x34></span>
<br><span style=" font-size:10pt;font-family:Courier New">   241c8:
      ba 07 7f fc     add  %i5, -4, %i5</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241cc:
      9f c0 40 00     call  %g1</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241d0:
      ba 07 7f fc     add  %i5, -4, %i5</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241d4:
      c2 07 40 00     ld  [ %i5 ], %g1</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241d8:
      80 a0 7f ff     cmp  %g1, -1</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241dc:
      12 bf ff fc     bne  241cc <__do_global_ctors_aux+0x1c></span>
<br><span style=" font-size:10pt;font-family:Courier New">   241e0:
      01 00 00 00     nop</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241e4:
      81 c7 e0 08     ret</span>
<br><span style=" font-size:10pt;font-family:Courier New">   241e8:
      81 e8 00 00     restore</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">This code will
clearly fail to call our coverage stubs -  it will immediately "see"
the -1, and return.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">After quite a
lot of hunting, we traced this failure to a patch in the "linkcmds.base"
- done back in 2014 ( commit 95cb09ed746 ) :</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">This commit changed
"c/src/lib/libbsp/sparc/shared/startup/linkcmds.base" from this:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(SORT(.ctors.*)))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(.ctors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*crtbegin.o(.dtors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*crtbegin?.o(.dtors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .dtors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(SORT(.dtors.*)))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(.dtors))</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">...to this:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .ctors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(SORT(.ctors*)))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*crtbegin.o(.dtors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*crtbegin?.o(.dtors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .dtors))</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
KEEP (*(SORT(.dtors*)))</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">Reverting this
change, the gcov stubs revert back to their proper place, and <i>__gcov_init</i>s
are called properly.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">The reason I tried
this patch, was because I noticed that the coverage stubs are no longer
placed in ".ctor" sections - they are instead placed by the newer
compilers at ".ctor.NUMBER" sections (with NUMBER set to 65435
in my case - but I am guessing this will change from invocation to invocation).
</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">The comment provided
by Joel back in 2014, only indicated that this patch was done to address
C++ concerns:</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">commit 95cb09ed746e7daeca2158c7ecdf0249cfcbc5c8</span>
<br><span style=" font-size:10pt;font-family:Courier New">Author: Joel
Sherrill <joel.sherrill@oarcorp.com></span>
<br><span style=" font-size:10pt;font-family:Courier New">Date:  
Wed Apr 2 11:39:20 2014 -0500</span>
<br>
<br><span style=" font-size:10pt;font-family:Courier New">   
sparc/shared/.../linkcmds.base: Correct C++ support</span>
<br><span style=" font-size:10pt;font-family:Courier New">   
Add KEEP() for .eh_frame*, .ctor*, and .dtor*.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">...but it appears
that it also breaks the proper operation of coverage stubs.</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">Can you please
confirm whether this is indeed a bug - and/or indicate whether there's
some other workaround that can be used besides "hacking" the
linkcmds this way, to make coverage functionality operate properly?</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif">Thanks in advance,</span>
<br><span style=" font-size:10pt;font-family:sans-serif">Thanassis.</span>
<br>
<br><span style=" font-size:10pt;color:#000080;font-family:sans-serif"><b>Thanassis
Tsiodras</b></span>
<br><span style=" font-size:10pt;font-family:sans-serif">Real-time Embedded
Software Engineer </span>
<br><span style=" font-size:10pt;font-family:sans-serif">System, Software
and Technology Department</span>
<br>
<br><span style=" font-size:10pt;font-family:sans-serif"><b>ESTEC</b></span>
<br><span style=" font-size:10pt;font-family:sans-serif">Keplerlaan 1,
PO Box 299</span>
<br><span style=" font-size:10pt;font-family:sans-serif">NL-2200 AG Noordwijk,
The Netherlands</span>
<br><span style=" font-size:10pt;font-family:sans-serif">Thanassis.Tsiodras@esa.int
| </span><a href=www.esa.int><span style=" font-size:10pt;color:blue;font-family:sans-serif">www.esa.int</span></a>
<br><span style=" font-size:10pt;font-family:sans-serif">T +31 71 565 5332</span><PRE>This message and any attachments are intended for the use of the addressee or addressees only.
The unauthorised disclosure, use, dissemination or copying (either in whole or in part) of its
content is not permitted.
If you received this message in error, please notify the sender and delete it from your system.
Emails can be altered and their integrity cannot be guaranteed by the sender.

Please consider the environment before printing this email.

</PRE>