[llvm-dev] [cfe-dev] _Float16 support

John McCall via llvm-dev llvm-dev at lists.llvm.org
Thu Jan 24 10:57:54 PST 2019



On 24 Jan 2019, at 4:46, Sjoerd Meijer wrote:

> Hello,
>
> I added _Float16 support to Clang and codegen support in the AArch64 
> and ARM backends, but have not looked into x86. Ahmed is right: 
> AArch64 is fine, only a few ACLE intrinsics are missing. ARM has rough 
> edges: scalar codegen should be mostly fine, vector codegen needs some 
> more work.
>
> Implementation for AArch64 was mostly straightforward (it only has 
> hard float ABI, and has half register/type support), but for ARM it 
> was a huge pain to plumb f16 support because of different ABIs 
> (hard/soft), different architecture extensions of FP and FP16 support, 
> and the existence of another half-precision type with different 
> semantics. Sounds like you're doing a similar exercise, and yes, 
> argument passing was one of the trickiest parts.
>
>
>> IR and SelectionDAG representational choices aside, it seems to me 
>> that,
>
>> like GCC, Clang should not be permitting _Float16 on any target  that 
>> doesn't
>
>> specify an ABI for it, because otherwise we're just creating future 
>> compatibility
>
>> problems for that target.  I'm surprised and  disappointed that it 
>> wasn't implemented
>
>> this way.
>
> Apologies, I missed that.

It's alright, oversights happen (in both patch-writing and review).  Can 
we get a volunteer to do the work to restrict this now?  I'm a little 
crushed.

John.

>
> Sjoerd.
>
> ________________________________
> From: llvm-dev <llvm-dev-bounces at lists.llvm.org> on behalf of Kaylor, 
> Andrew via llvm-dev <llvm-dev at lists.llvm.org>
> Sent: 24 January 2019 00:23
> To: Ahmed Bougacha; Lu, Hongjiu
> Cc: llvm-dev; cfe-dev at lists.llvm.org
> Subject: Re: [llvm-dev] [cfe-dev] _Float16 support
>
> It seems that there are several issues here:
>
> 1. Should the front end be concerned with whether or not the IR that 
> it is emitting can be translated into a well-defined IR?
> 2. How should the selection DAG handle data types whose representation 
> isn't defined by the ABI we're targeting?
> 3. What should the ABI do with half-precision floats?
>
> Working backward...
>
> The third question here is obviously target specific. I've talked to 
> HJ Lu about this, and he's working on an update to the x86 psABI. I 
> believe that his eventual proposal will follow the lines of what you 
> (Ahmed) suggested below, but I'm not completely proficient at 
> comprehending ABI definitions so there may be some subtlety that I am 
> misunderstanding in what he told me. I also talked to Craig about 
> would be involved in making the LLVM x86 backend handle 'half' values 
> this way. That involves a good bit of work, but it can be done.
>
> The second question above probably involves a mix of 
> target-independent and target-specific code. Right now the selection 
> DAG code is operating on the assumption that it needs to do 
> *something* with any IR it is given. It tries to make a reasonable 
> choice, and the choice is consistent and predictable but not 
> necessarily what the user expects. It seems like we should at the very 
> least be producing a diagnostic so the user knows what we did (or even 
> just that we did something). Then there are the specific problems 
> Craig has brought up with the way we're currently handling 'half' 
> values. Would defining a legal f16 type take care of those problems?
>
> The first question exposes my lack of understanding of the proper role 
> of the front end. It isn't clear to me what responsibility the front 
> end has for enforcing conformance to the ABI. As a user of the 
> compiler, I would like the compiler to tell me when code I've written 
> can't be represented using the ABI I am targeting. Whether the front 
> end should detect this or the backend, I don't know. I suppose it's 
> also an open question how strictly this should be enforced. Is it a 
> warning that can be elevated to an error at the users' discretion? Is 
> it something that should be blocked by default but enabled by a 
> user-specified option? Should it always be rejected?
>
> -Andy
>
> -----Original Message-----
> From: Ahmed Bougacha <ahmed.bougacha at gmail.com>
> Sent: Wednesday, January 23, 2019 3:30 PM
> To: Kaylor, Andrew <andrew.kaylor at intel.com>
> Cc: cfe-dev at lists.llvm.org; llvm-dev <llvm-dev at lists.llvm.org>; Craig 
> Topper <craig.topper at gmail.com>; Richard Smith <richard at metafoo.co.uk>
> Subject: Re: [cfe-dev] _Float16 support
>
> Hey Andy,
>
> On Tue, Jan 22, 2019 at 10:38 AM Kaylor, Andrew via cfe-dev 
> <cfe-dev at lists.llvm.org> wrote:
>> I'd like to start a discussion about how clang supports _Float16 for 
>> target architectures that don't have direct support for 16-bit 
>> floating point arithmetic.
>
> Thanks for bringing this up;  we'd also like to get better support, 
> for sysv x86-64 specifically - AArch64 is mostly fine, and ARM is 
> usable with +fp16.
>
> I'm not sure much of this discussion generalizes across platforms 
> though (beyond Craig's potential bug fix?).  I guess the 
> "target-independent" question is: should we allow this kind of 
> "legalization" in the vreg assignment code at all? (I think that's 
> where it all comes from: RegsForValue, TLI::get*Register*) It's 
> convenient for experimental frontends: you can use weird types (half, 
> i3, ...) without worrying too much about it, and you usually get 
> something self-consistent out of the backend.  But you eventually need 
> to worry about it and need to make the calling convention explicit.  
> But I guess that's a discussion for the other thread ;)
>
>> The current clang language extensions documentation says, "If 
>> half-precision instructions are unavailable, values will be promoted 
>> to single-precision, similar to the semantics of __fp16 except that 
>> the results will be stored in single-precision." This is somewhat 
>> vague (to me) as to what is meant by promotion of values, and the 
>> part about results being stored in single-precision isn't what 
>> actually happens.
>>
>> Consider this example:
>>
>> _Float16 x;
>> _Float16 f(_Float16 y, _Float16 z) {
>>   x = y * z;
>>   return x;
>> }
>>
>> When compiling with “-march=core-avx2” that results (after some 
>> trivial cleanup) in this IR:
>>
>> @x = global half 0xH0000, align 2
>> define half @f(half, half) {
>>   %3 = fmul half %0, %1
>>   store half %3, half* @x
>>   ret half %3
>> }
>>
>> That’s not too unreasonable I suppose, except for the fact that it 
>> hasn’t taken the lack of target support for half-precision 
>> arithmetic into account yet. That will happen in the selection DAG. 
>> The assembly code generated looks like this (with my annotations):
>>
>> f:                                      # @f
>> # %bb.0:
>>        vcvtps2ph       xmm1, xmm1, 4             # Convert argument 1 
>> from single to half
>>         vcvtph2ps       xmm1, xmm1                # Convert argument 
>> 1 back to single
>>         vcvtps2ph       xmm0, xmm0, 4            # Convert argument 0 
>> from single to half
>>         vcvtph2ps       xmm0, xmm0                # Convert argument 
>> 0 back to single
>>         vmulss             xmm0, xmm0, xmm1   # xmm0 = xmm0*xmm1 
>> (single precision)
>>         vcvtps2ph       xmm1, xmm0, 4            # Convert the single 
>> precision result to half
>>         vmovd             eax, xmm1                      # Move the 
>> half precision result to eax
>>         mov                 word ptr [rip + x], ax     # Store the 
>> half precision result in the global, x
>>         ret                                                           
>>   # Return the single precision result still in xmm0
>> .Lfunc_end0:
>>                                         # -- End function
>>
>> Something odd has happened here, and it may not be obvious what it 
>> is. This code begins by converting xmm0 and xmm1 from single to half 
>> and then back to single. The first conversion is happening because 
>> the back end decided that it needed to change the types of the 
>> parameters to single precision but the function body is expecting 
>> half precision values. However, since the target can’t perform the 
>> required computation with half precision values they must be 
>> converted back to single for the multiplication. The single precision 
>> result of the multiplication is converted to half precision to be 
>> stored in the global value, x, but the result is returned as single 
>> precision (via xmm0).
>>
>> I’m not primarily worried about the extra conversions here. We 
>> can’t get rid of them because we can’t prove they aren’t 
>> rounding, but that’s a secondary issue. What I’m worried about is 
>> that we allowed/required the back end to improvise an ABI to satisfy 
>> the incoming IR, and the choice it made is questionable.
>
> As Richard said, an ABI rule emerged from the implementation, and I 
> believe we should solidify it, so here's a simple strawman proposal:
> pass scalars in the low 16 bits of SSE registers, don't change the 
> memory layout, and pack them in vectors of 16-bit elements.  That 
> matches the only ISA extension so far (ph<>ps conversions), and fits 
> well with that (as opposed to i16 coercion) as well as vectors (as 
> opposed to f32 promotion).  To my knowledge, there hasn't been any 
> alternative ABI proposal (but I haven't looked in 1 or 2 years).  It's 
> interesting because we technically have no way of accessing scalars 
> (so we have the same problems as i8/i16 vector elements, but without 
> the saving grace of having matching GPRs - x86, or direct copies - 
> aarch64), and there are not even any scalar operations.
>
> Any thoughts?  We can suggest this to x86-psABI if folks think this is 
> a good idea. (I don't know about other ABIs or other architectures 
> though).
>
> Concretely, this means no/little change in IRGen.  As for the SDAG 
> implementation, this is an unusual situation.  I've done some 
> experimentation a long time ago.  We can make the types legal, even
> though no operations are.   It's relatively straightforward to promote
> all operations (and we made sure that worked years ago for AArch64, 
> for the pre-v8.2 mode), but vectors are fun, because of build_vector 
> (where it helps to have the truncating behavior we have for integers, 
> but for fp), extract_vector_elt (where you need the matching extend), 
> and insert_vector_elt (which you have to lower using some movd and/or 
> pinsrw trickery, if you want to avoid the generic slow via-memory 
> fallback).
> Alternatively, we can immediately, in call lowering/register 
> assignment logic (this covers the SDAG cross-BB vreg assignments Craig
> mentions) promote to f32 "via" i16.  I'm afraid I don't remember the 
> arguments one way or the other, I can dust off my old patches and put 
> them up on phabricator.
>
>
> -Ahmed
>
>>
>> For a point of comparison, I looked at what gcc does. Currently, gcc 
>> only allows _Float16 in C, not C++, and if you try to use it with a 
>> target that doesn’t have native support for half-precision 
>> arithmetic, it tells you “’_Float16’ is not supported on this 
>> target.” That seems preferable to making up an ABI on the fly.
>>
>> I haven’t looked at what happens with clang when compiling for 
>> other targets that don’t have native support for half-precision 
>> arithmetic, but I would imagine that similar problems exist.
>>
>> Thoughts?
>>
>> Thanks,
>> Andy
>> _______________________________________________
>> cfe-dev mailing list
>> cfe-dev at lists.llvm.org
>> http://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-dev
> _______________________________________________
> LLVM Developers mailing list
> llvm-dev at lists.llvm.org
> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev
> IMPORTANT NOTICE: The contents of this email and any attachments are 
> confidential and may also be privileged. If you are not the intended 
> recipient, please notify the sender immediately and do not disclose 
> the contents to any other person, use it for any purpose, or store or 
> copy the information in any medium. Thank you.




More information about the llvm-dev mailing list