TLS, gcc optimizations, and PIC on x86

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



I have a question regarding cached values for the addresses of TLS
variables in PIC when gcc optimizations are turned on.

Specifically, I want to be able to change the value of my TLS
descriptor as well as access TLS variables from that new descriptor
within the body of a single function.  With gcc optimizations turned
on and when compiling for PIC, this doesn't always seem to work.  The
problem lies in the fact that, with optimizations turned on, the
addresses of any TLS variables accessed before the TLS descriptor is
changed, persist to accesses of these variables after the TLS has
changed.  Note, this is only really a problem with PIC, as addresses
are calculated via a function call to ___tls_get_addr() and gcc is
trying to optimize these calls away (yes, even with
tls-model=global-dynamic explicitly set).  For non PIC everything
appears to be fine, since a different method of calculating TLS
variable addresses is used and doesn't need to be optimized.

Consider the following pseudo code:

__thread int i = 0;
i = 5;  // Set variable in original tls region
set_tls_desc(new_tls);
i = 6;  // Set variable in new tls region

I want i=6 to be set for the i in my new tls region, not the original
one.  However, with optimizations turned on (i.e. -O2), the i from the
old tls region is usually the one whose value gets changed.  I mean,
this makes sense since there is no real way (that I am aware of) to
tell gcc that the TLS has changed out from under it -- all O2
optimizations for caching variables in registers, etc. are valid.

One thing that had occurred to me was to force a compiler memory
barrier after set_tls_desc() (i.e. asm volatile("" ::: "memory")), but
this doesn't work because there's not any memory changing for the
variables I have access to (i.e. i in this case) - but rather it's the
address of the variable i as calculated via a call to __tls_get_addr()
that has now changed....

Essentially, I want a way to inform the compiler that any TLS
variables accessed after my call to set_tls_desc() need to have their
addresses refreshed via a new call to __tls_get_addr().  I was hoping
I could accomplish this via something like asm volatile ("" ::: "%gs")
for i386 or asm volatile ("" ::: "%fs") on x86_64, but apparently
these registers aren't even allowed in the clobber list....

Therefore, my current solution to solve this problem is to define the
following macros:

#define safe_set_tls_var(name, val)
({
  void __attribute__((noinline, optimize("O0")))
  safe_set_tls_var_internal() {
    asm("");
    name = val;
  } safe_set_tls_var_internal();
})

#define safe_get_tls_var(name)
({
  typeof(name) __attribute__((noinline, optimize("O0")))
  safe_get_tls_var_internal() {
    return name;
  } safe_get_tls_var_internal();
})

These macros create a nested function at each call site where a TLS
variable is get/set with attributes of 'noinline' and
'optimize("O0")'.  Having these attributes makes sure that the
getting/setting operations actually load the tls addr via a call to
___tls_get_addr() instead of using a cached value.  Furthermore, by
making these macros that define nested functions instead of defining a
single pair of functions that take a set of parameters, I am able to
reference the specific TLS variable I want to set BY NAME, at the call
site, don't have to pass the address to it instead. This forces a
recalculation of the address of the specific TLS variable under
question within the body of the nested function.

The code then becomes:

__thread int i = 0;
i = 5;
set_tls_desc(new_tls);
safe_set_tls_var(i, 6);

This can get cumbersome though, if I keep accessing TLS variables over
and over again further down in the function.  I could always break up
the function into multiple pieces to force all TLS variables to be
accessed in a new function after the call to set_tls_desc(), but it
doesn't seem like something I should have to do.

As I said before, what I really want is something like: asm volatile
("" ::: "%gs") (or better yet, asm volatile ("" ::: "tls")) do hint to
the compiler that all the addresses of all TLS variables after this
point need to be recalculated.

For all I know though, maybe something like this already exists.
Does anyone know of something like this that's already out there?

-- 
~Kevin


[Index of Archives]     [Linux C Programming]     [Linux Kernel]     [eCos]     [Fedora Development]     [Fedora Announce]     [Autoconf]     [The DWARVES Debugging Tools]     [Yosemite Campsites]     [Yosemite News]     [Linux GCC]

  Powered by Linux