public inbox for gdb@sourceware.org
 help / color / mirror / Atom feed
From: Simon Marchi <simon.marchi@polymtl.ca>
To: David Blaikie <dblaikie@gmail.com>
Cc: Tim Newsome <tim@sifive.com>, gdb <gdb@sourceware.org>
Subject: Re: Remote query for structure layout
Date: Wed, 31 Mar 2021 09:00:02 -0400	[thread overview]
Message-ID: <4f5e4de2-035d-a1eb-69ef-bb4144ce82de@polymtl.ca> (raw)
In-Reply-To: <CAENS6EuJwV4sxM6NwdffQ5bpPqkPrdk62qD_PNo0Jiy3pQqv=Q@mail.gmail.com>

On 2021-03-30 11:27 p.m., David Blaikie wrote:
> (let me know if I'm just being an unhelpful bystander here - I clearly
> don't have a lot of context)

I think it is interesting.  If people are not interested they can just
skip reading this thread :).

> On Tue, Mar 30, 2021 at 5:16 PM Simon Marchi <simon.marchi@polymtl.ca> wrote:
>>
>> On 2021-03-30 6:44 p.m., David Blaikie wrote:
>>> If it's "just" some user-code, is there a variable of the desired type being declared around the function call?
>>
>> AFAIK, this type wouldn't be used by the FreeRTOS code at all, so no.
>> Again, here's my understanding, hopefully it's close enough to the
>> reality.
>>
>> When a trap occurs, FreeRTOS saves the current task's register values on
>> the task's stack, using some arch-specific assembly code.  For example,
>> for RISC-V:
>>
>>     https://github.com/FreeRTOS/FreeRTOS-Kernel/blob/534eba66ce4a5bda45d5edeeb81ac5a3cf6d0df8/portable/GCC/RISC-V/portASM.S#L121
> 
> Ah, OK. So this is part of signal handling - so it's not part of the
> code that's compiled into the user's program, for instance... not even
> in some system library linked in, necessarily, I guess?

It is part of "user code".  Wih "big" OSes (e.g. Linux), the OS is
compiled on its own and the user programs are compiled separately.  With
embedded RTOSes (at least those I worked with), it's just one big
program.  Your user code includes <FreeRTOS.h> and calls FreeRTOS
functions to spawn to tasks.

>> The way these registers are pushed is an implementation detail of
>> FreeRTOS.  And we can imagine that it can vary depending on the
>> compile-time FreeRTOS configuration,
> 
> I guess in the worst case it could be totally dynamic - it could pick
> a different layout each time.

Yes, if it wanted to be really annoying :).

> (guess a side question: How's this different from other systems? I
> don't know how other/more common systems handle registers during
> signals)

I think Linux does the same, pushing the registers (and some information
about the signal) on the thread's stack before calling a signal handler.
The layout is defined by the ucontext structure, exposed in the Linux
ABI:

    https://github.com/torvalds/linux/blob/master/arch/riscv/include/uapi/asm/ucontext.h

GDB hardcodes that layout here:

    https://sourceware.org/git/?p=binutils-gdb.git;a=blob;f=gdb/riscv-linux-tdep.c;h=ca97a60128ffe681766e7e1600d3f24048bf1f68;hb=HEAD#l103

Being exposed by the Linux kernel and part of this ABI, this structure
is not likely to change.  That makes it reasonable to hardcode it.

>> which OpenOCD doesn't know about.
>>
>> To get register values of scheduled out tasks, OpenOCD needs to
>> interpret these register values from the tasks' stacks.
>>
>> So Tim's suggestion is: have FreeRTOS declare a structure that has the
>> exact layout as the saved registers on the stack:
>>
>> struct freertos_saved_regs {
>>   int x1;
>>   int x5;
>>   int x6;
>>   ...
>> };
>>
>> Consumers could read that structure's layout from the DWARF info, and
>> read the register values based on that.  That would be a lot more robust
>> than hard-coding in the consumers how FreeRTOS stores things.
> 
> If practical experience has shown the hard-coding is not
> stable/reliable (than FreeRTOS does change its strategy from time to
> time - but it's always constant for any given build of the FreeRTOS) I
> guess.

My guess is that FreeRTOS does not change it that often, there's not
reason to.  But it can vary from build to build due to compile-time
configuration of FreeRTOS (i.e. include or exclude support for using
some particular register).  An external observer such as OpenOCD does
not have access to the compile-time configuration that was used to build
FreeRTOS, so it's difficult to know what the layout used by this
particular build is.

> But I'm not sure where FreeRTOS would expose this structure to user
> code - it's not like there's a system library header that user code
> must include...

There is, any code using FreeRTOS will include FreeRTOS.h.  You can see
FreeRTOS more like a library than an OS.  FreeRTOS is not involved at
all until the user program calls functions like xTaskCreate to create
some tasks and calls vTaskStartScheduler to hand over control to
FreeRTOS.

So if the structure was declared somewhere in the port's include file,
it would necessarily be seen by some CUs (those that invoke FreeRTOS
functions).

>> However, since that struct would never actually be used by FreeRTOS'
>> code, the compiler won't emit it.  Hence the need to find a way to force
>> the compiler to include it in the DWARF.
>>
>> Does that clarify the situation?
> 
> Somewhat - any lack of understanding is just my ignorance in this
> field/area in general, to be clear. (I'm also not a core gdb
> developer, so I'm not the sort of person you have to convince of
> anything - just a curious bystander trying to understand/maybe offer
> some insightful suggestions (I predominantly work on LLVM's debug info
> emission, so that's my background/connection))

Well, since it's related to debug info emission, it could land on your
plate at some point, it's good that you have some context.

Simon

  reply	other threads:[~2021-03-31 13:00 UTC|newest]

Thread overview: 20+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
     [not found] <mailman.6.1616932808.1485743.gdb@sourceware.org>
2021-03-30 20:35 ` Tim Newsome
2021-03-30 21:33   ` Simon Marchi
2021-03-30 21:41     ` David Blaikie
2021-03-30 21:49       ` Simon Marchi
2021-03-30 22:03         ` David Blaikie
2021-03-30 22:38           ` Simon Marchi
2021-03-30 22:44             ` David Blaikie
2021-03-31  0:16               ` Simon Marchi
2021-03-31  3:27                 ` David Blaikie
2021-03-31 13:00                   ` Simon Marchi [this message]
2021-04-02 22:05                     ` David Blaikie
2021-04-03 19:39                       ` Simon Marchi
2021-03-31 14:06                   ` Tim Newsome
2021-03-28 11:06 Thomas Weißschuh
2021-03-29 16:05 ` Simon Marchi
2021-03-29 17:33   ` Thomas Weißschuh
2021-03-29 19:42     ` Simon Marchi
2021-03-29 20:02       ` Philippe Waroquiers
2021-03-29 20:10         ` Simon Marchi
2021-03-29 20:20           ` Philippe Waroquiers

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4f5e4de2-035d-a1eb-69ef-bb4144ce82de@polymtl.ca \
    --to=simon.marchi@polymtl.ca \
    --cc=dblaikie@gmail.com \
    --cc=gdb@sourceware.org \
    --cc=tim@sifive.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).