On Mon, Sep 25, 2017 at 11:32 PM, Arnd Bergmann <a...@arndb.de> wrote: > On Mon, Sep 25, 2017 at 7:41 AM, David Laight <david.lai...@aculab.com> wrote: >> From: Arnd Bergmann >>> Sent: 22 September 2017 22:29 >> ... >>> It seems that this is triggered in part by using strlcpy(), which the >>> compiler doesn't recognize as copying at most 'len' bytes, since strlcpy >>> is not part of the C standard. >> >> Neither is strncpy(). >> >> It'll almost certainly be a marker in a header file somewhere, >> so it should be possibly to teach it about other functions. > > I'm currently travelling and haven't investigated in detail, but from > taking a closer look here, I found that the hardened 'strlcpy()' > in include/linux/string.h triggers it. There is also a hardened > (much shorted) 'strncpy()' that doesn't trigger it in the same file, > and having only the extern declaration of strncpy also doesn't.
And a little more experimenting leads to this simple patch that fixes the problem: --- a/include/linux/string.h +++ b/include/linux/string.h @@ -254,7 +254,7 @@ __FORTIFY_INLINE size_t strlcpy(char *p, const char *q, size_t size) size_t q_size = __builtin_object_size(q, 0); if (p_size == (size_t)-1 && q_size == (size_t)-1) return __real_strlcpy(p, q, size); - ret = strlen(q); + ret = __builtin_strlen(q); if (size) { size_t len = (ret >= size) ? size - 1 : ret; if (__builtin_constant_p(len) && len >= p_size) The problem is apparently that the fortified strlcpy calls the fortified strlen, which in turn calls strnlen and that ends up calling the extern '__real_strnlen' that gcc cannot reduce to a constant expression for a constant input. Not sure if that change is the best fix, but it seems to address the problem in this driver and probably leads to better code in other places as well. Arnd