Received: by 2002:a05:6a11:4021:0:0:0:0 with SMTP id ky33csp375302pxb; Wed, 22 Sep 2021 04:20:51 -0700 (PDT) X-Google-Smtp-Source: ABdhPJxOIejYwlfXCPwGDP60RTohC8gShnowbiHIETwahXQShNg0YqICwHcRMyO3f6HExxmB3Wg+ X-Received: by 2002:a05:6638:203:: with SMTP id e3mr4348631jaq.88.1632309651547; Wed, 22 Sep 2021 04:20:51 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1632309651; cv=none; d=google.com; s=arc-20160816; b=HfPtS8JgNr5kQ/IbZ+OkHk4W+nvYfdWg2MfH4CNHE4YYIf/RTtb+tT6ADGbAE3nErA 63Vgqs1F6NlSeWooplHiMhAxwpl1fVQWcKlOY8/j2oHCXODlCBz9VbpdJe2n3/luUYTe Bt46gQwIzZq66NQKqawUEhupNWaQcZc2V/5gNPLPlDNwJDlkwUVPgimwQg2/EfxwSDeA nMD+P9aSuAFoT9tUvcCDf4uwZQXv0J17yLKOvgv7hAPtYnj81z0NfnBFXBmnCrhRhu+C Tp5QAYtWV4VZUFmfwvU3aG36PKY5fNfzwFR2Aw9EgeXIuwCCeNxwas4AZbmEOkCvQ9Fe qkRw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :message-id:date:references:organization:in-reply-to:subject:cc:to :from; bh=UkqRGGa2PULQdRjMtnlLlXwaUa9sUoh8A7b0sWaZ61Y=; b=r9Z2RG4AwDCzj5IJhlmMJZ9ns+IcA/bExPPCVTVaiT8JZ2eLYdEq0bSlC6ZpyJvxp1 yDp3vlYAzNmMpm/5Kw061oq9L4vVEgpMxNBh7m41pS8VrdvNWDxmE8Il27DebvAmCp+J HyuykhTZRXMlVEREs3YX2Sx+xxjOmDQMibUq7kVOZ+voP95UAXHoV0cjFiKnJRK40Hzq t95S6YY6jN7ZBfXhPhSV1jBqM6T2w8+NB5fMxcM6RwIKUY/FDZhaMsXrput/m173jRej pjCOojK8rNWhMV51HtV5PSlJvoeJZS+FfIEl/uI9i9ngz+ouU9eCMjraXcbCCjZw1X1O Fn1g== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id x9si2262784jaf.48.2021.09.22.04.20.37; Wed, 22 Sep 2021 04:20:51 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S235431AbhIVLVR convert rfc822-to-8bit (ORCPT + 99 others); Wed, 22 Sep 2021 07:21:17 -0400 Received: from mga12.intel.com ([192.55.52.136]:55531 "EHLO mga12.intel.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230171AbhIVLVQ (ORCPT ); Wed, 22 Sep 2021 07:21:16 -0400 X-IronPort-AV: E=McAfee;i="6200,9189,10114"; a="203065012" X-IronPort-AV: E=Sophos;i="5.85,313,1624345200"; d="scan'208";a="203065012" Received: from fmsmga002.fm.intel.com ([10.253.24.26]) by fmsmga106.fm.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 22 Sep 2021 04:19:46 -0700 X-IronPort-AV: E=Sophos;i="5.85,313,1624345200"; d="scan'208";a="557422965" Received: from vidyaram-mobl1.gar.corp.intel.com (HELO localhost) ([10.251.218.73]) by fmsmga002-auth.fm.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 22 Sep 2021 04:19:39 -0700 From: Jani Nikula To: Alexey Dobriyan , linux-kernel@vger.kernel.org Cc: Linus Torvalds , Joe Perches , Andrew Morton , apw@canonical.com, Christoph Lameter , Daniel Micay , Dennis Zhou , dwaipayanray1@gmail.com, Joonsoo Kim , Linux-MM , Lukas Bulwahn , mm-commits@vger.kernel.org, Nathan Chancellor , Nick Desaulniers , Miguel Ojeda , Pekka Enberg , David Rientjes , Tejun Heo , Vlastimil Babka , linux-doc@vger.kernel.org Subject: Re: function prototype element ordering In-Reply-To: Organization: Intel Finland Oy - BIC 0357606-4 - Westendinkatu 7, 02160 Espoo References: <20210909200948.090d4e213ca34b5ad1325a7e@linux-foundation.org> <20210910031046.G76dQvPhV%akpm@linux-foundation.org> <202109211630.2D00627@keescook> <202109211757.F38DF644@keescook> Date: Wed, 22 Sep 2021 14:19:28 +0300 Message-ID: <874kacn9hb.fsf@intel.com> MIME-Version: 1.0 Content-Type: text/plain; charset=utf-8 Content-Transfer-Encoding: 8BIT Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Wed, 22 Sep 2021, Alexey Dobriyan wrote: > On Tue, Sep 21, 2021 at 07:25:53PM -0700, Kees Cook wrote: >> On Tue, Sep 21, 2021 at 04:45:44PM -0700, Joe Perches wrote: >> > On Tue, 2021-09-21 at 16:37 -0700, Kees Cook wrote: >> > > On Fri, Sep 10, 2021 at 10:23:48AM -0700, Linus Torvalds wrote: >> > > > On Thu, Sep 9, 2021 at 8:10 PM Andrew Morton wrote: >> > > > > >> > > > > +__alloc_size(1) >> > > > >  extern void *vmalloc(unsigned long size); >> > > > [...] >> > > > >> > > > All of these are added in the wrong place - inconsistent with the very >> > > > compiler documentation the patches add. >> > > > >> > > > The function attributes are generally added _after_ the function, >> > > > although admittedly we've been quite confused here before. >> > > > >> > > > But the very compiler documentation you point to in the patch that >> > > > adds these macros gives that as the examples both for gcc and clang: >> > > > >> > > > + * gcc: https://gcc.gnu.org/onlinedocs/gcc/Common-Function-Attributes.html#index-alloc_005fsize-function-attribute >> > > > + * clang: https://clang.llvm.org/docs/AttributeReference.html#alloc-size >> > > > >> > > > and honestly I think that is the preferred format because this is >> > > > about the *function*, not about the return type. >> > > > >> > > > Do both placements work? Yes. >> > > >> > > I'm cleaning this up now, and have discovered that the reason for the >> > > before-function placement is consistency with static inlines. If I do this: >> > > >> > > static __always_inline void * kmalloc(size_t size, gfp_t flags) __alloc_size(1) >> > > { >> > > ... >> > > } >> > > >> > > GCC is very angry: >> > > >> > > ./include/linux/slab.h:519:1: error: attributes should be specified before the declarator in a function definition >> > >   519 | static __always_inline void *kmalloc_large(size_t size, gfp_t flags) __alloc_size(1) >> > >       | ^~~~~~ >> > > >> > > It's happy if I treat it as a "return type attribute" in the ordering, >> > > though: >> > > >> > > static __always_inline void * __alloc_size(1) kmalloc(size_t size, gfp_t flags) >> > > >> > > I'll do that unless you have a preference for somewhere else... >> > >> > _please_ put it before the return type on a separate line. >> > >> > [__attributes] >> > [static inline const] function() >> >> Somehow Linus wasn't in CC. :P >> >> Linus, what do you want here? I keep getting conflicting (or >> uncompilable) advice. I'm also trying to prepare a patch for >> Documentation/process/coding-style.rst ... >> >> Looking through what was written before[1] and through examples in the >> source tree, I find the following categories: >> >> 1- storage class: static extern inline __always_inline >> 2- storage class attributes/hints/???: __init __cold >> 3- return type: void * >> 4- return type attributes: __must_check __noreturn __assume_aligned(n) >> 5- function attributes: __attribute_const__ __malloc >> 6- function argument attributes: __printf(n, m) __alloc_size(n) >> >> Everyone seems to basically agree on: >> >> [storage class] [return type] [return type attributes] [name]([arg1type] [arg1name], ...) >> >> There is a lot of disagreement over where 5 and 6 should fit in above. And >> there is a lot of confusion over 4 (mixed between before and after the >> function name) and 2 (see below). >> >> What's currently blocking me is that 6 cannot go after the function >> (for definitions) because it angers GCC (see quoted bit above), but 5 >> can (e.g. __attribute_const__). >> >> Another inconsistency seems to be 2 (mainly section markings like >> __init). Sometimes it's after the storage class and sometimes after the >> return type, but it certainly feels more like a storage class than a >> return type attribute: >> >> $ git grep 'static __init int' | wc -l >> 349 >> $ git grep 'static int __init' | wc -l >> 8402 >> >> But it's clearly positioned like a return type attribute in most of the >> tree. What's correct? >> >> Regardless, given the constraints above, it seems like what Linus may >> want is (on "one line", though it will get wrapped in pathological cases >> like kmem_cache_alloc_node_trace): >> >> [storage class] [storage class attributes] [return type] [return type attributes] [function argument attributes] [name]([arg1type] [arg1name], ...) [function attributes] >> >> Joe appears to want (on two lines): >> >> [storage class attributes] [function attributes] [function argument attributes] >> [storage class] [return type] [return type attributes] [name]([arg1type] [arg1name], ...) >> >> I would just like to have an arrangement that won't get NAKed by >> someone. ;) And I'm willing to document it. :) > > Attributes should be on their own line, they can be quite lengthy. > > __attribute__((...)) > [static] [inline] T f(A1 arg1, ...) > { > ... > } > > There will be even more attributes in the future, both added by > compilers and developers (const, pure, WUR), so let's make "prototype lane" > for them. > > Same for structures: > > __attribute__((packed)) > struct S { > }; > > Kernel practice of hiding attributes under defines (__ro_after_init) > breaks ctags which parses the last identifier before semicolon as object > name. Naturally, it is ctags bug, but placing attributes before > declaration will autmatically unbreak such cases. git grep seems to suggest __packed is preferred over __attribute__((packed)), and at the end of the struct declaration instead of at front: struct S { /* ... */ } __packed; And GNU Global handles this just fine. ;) BR, Jani. -- Jani Nikula, Intel Open Source Graphics Center