eddyz87 added a comment.

Hi @dblaikie

Thank you for the detailed response!

> So you get some bunch of annotations from the BTFTagAttributedType, then you 
> build the underlying type (which might already be built/needed/fine because 
> it's used without attributes in other places/needs to exist independently) - 
> and then at the end you copy any of these types that are needed and put the 
> annotations on them?

Yes.

> Does the BTFTagAttributedType always have to apply to the immediate type 
> that's going to be modified/mutated to have the attributes applied to it 
> directly? Is the set of types that may have these attributes/annotations 
> added small/closed? (struct types, void, anything else? could you add tags to 
> int __tag *, for instance and get a DW_TAG_base_type for "int" with 
> annotations on it?

We decided against applying it to const/volatile/restrict, but it can be 
applied to struct types, void, "base" types ("int"), pointers and typedefs.

> If it's really generic/can apply to any type - but always the /immediate/ 
> type (I guess with the special handling you've got to skip applying it to the 
> DW_TAG_const_type, etc)...
>
> What if you skipped getOrCreateType - and called into the CreateType dispatch 
> below that? Since you never want a cached instance of a type, right? You want 
> to create a new copy of the type you could then apply annotations to.
>
> Except, I guess, in the instance you showed, where the type is being 
> completed - can't go and create another one, because we're part-way through 
> building this one... hmm, maybe you can, though? What happens if we start 
> making a totally distinct copy of that same type? I guess there's some map 
> that contains such partially completed types, and that map would get 
> confused/break if we tried to build two types for the same type without 
> adding in some extra key goo that contained the annotations... - maybe that 
> wouldn't be too invasive, though?

I made a prototype of such change, available here 
<https://github.com/llvm/llvm-project/commit/5ad452a913cd102d38deec130a09a9e5cc2d2b30#diff-ad303855624f31f09513ee5d6e5b435714d874b6d6c2afcb140b72a4b61a9166R1257>.
 The interesting function is `CGDebugInfo::CreateType(const 
BTFTagAttributedType *Ty, ...)`.
Pseudo code for old version (this revision):

  def CGDebugInfo::CreateType(const BTFTagAttributedType *Ty, ...):
    WrappedTy, Annotations = collect annotations are base type from Ty
    WrappedDI = getOrCreateType(WrappedTy, ...)
    Placeholder = create temporary placeholder node with
                  references to WrappedDI and Annotations
    AnnotationPlaceholders.push_back(Placeholder)
    return Placeholder
    
  def CGDebugInfo::finalize():
    ...
    for Placeholder in AnnotationPlaceholders:
      T = Placeholder.WrappedDI.Clone()
      T.replaceAnnotations(Placeholder.Annotations)
      replace Placeholder with T

Pseudo code for new version (link to my github from above):

  def CGDebugInfo::CreateType(const BTFTagAttributedType *Ty, ...):
    WrappedTy, Annotations = collect annotations are base type from Ty
    Placeholder = empty temporary node
    TypeCache[Ty] = Placeholder
    WrappedDI = CreateType(UnwrapTypeForDebugInfo(WrappedTy), ...)
    T = WrappedDI.Clone()
    T.replaceAnnotations(Annotations)
    replace Placeholder with T
    return T

Comparing the "old" and "new" versions "old" appears to be less hacky to me: it 
does not break `getOrCreateType()` abstraction and it does not process members 
two times for circular types.

Note that this 
<https://github.com/llvm/llvm-project/commit/5ad452a913cd102d38deec130a09a9e5cc2d2b30#diff-ad303855624f31f09513ee5d6e5b435714d874b6d6c2afcb140b72a4b61a9166R2805>
 diff is important for the new version, without it `CreateType` won't actually 
create a new object in the circular type case. I checked call-graph for 
`CreateType(RecordType *)` and it looks like it is only invoked from 
`getOrCreateType()`, so `getTypeOrNull()` within it is not actually needed. It 
was added in commit 3b1cc9b85846 back in 2013. Test cases for that commit are 
passing.

> given this code:
>
>   #define __tag1 __attribute__((btf_type_tag("tag1")))
>   
>   struct st {
>     struct st *self;
>   };
>   struct st __tag1 x;
>
> What's the expected DWARF here? x's type is a the attributed/annotated st 
> type, but then does the self pointer inside there point to the attributed 
> type or the unattributed type?

The `self` should have type w/o tag, the `x` should have type with tag.

> Maybe what you've got is the best of some bad options, but I'm not 
> sure/trying to think it through. (@aprantl wouldn't mind your perspective - 
> if it's too much to consume easily, let me know and I can make a best-effort 
> at summarizing, maybe even better over a video chat if you've got a few 
> minutes)

I appreciate the effort, if you and @aprantl decide that the call is necessary 
I can attend if you need me.


Repository:
  rG LLVM Github Monorepo

CHANGES SINCE LAST ACTION
  https://reviews.llvm.org/D143967/new/

https://reviews.llvm.org/D143967

_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to