diff options
author | Nicolas Dichtel <nicolas.dichtel@6wind.com> | 2016-04-22 11:31:24 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2016-04-23 20:13:25 -0400 |
commit | 80df554275c21edca22ece02448bdb378c2ee9f1 (patch) | |
tree | ea4a3f342c07929025c193b2b67e206e1008afa9 /kernel/taskstats.c | |
parent | de95c4a46a6e608444ccaf541087594553c7df11 (diff) |
taskstats: use the libnl API to align nlattr on 64-bit
Goal of this patch is to use the new libnl API to align netlink attribute
when needed.
The layout of the netlink message will be a bit different after the patch,
because the padattr (TASKSTATS_TYPE_STATS) will be inside the nested
attribute instead of before it.
Signed-off-by: Nicolas Dichtel <nicolas.dichtel@6wind.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'kernel/taskstats.c')
-rw-r--r-- | kernel/taskstats.c | 37 |
1 files changed, 5 insertions, 32 deletions
diff --git a/kernel/taskstats.c b/kernel/taskstats.c index 21f82c29c914..b3f05ee20d18 100644 --- a/kernel/taskstats.c +++ b/kernel/taskstats.c | |||
@@ -357,10 +357,6 @@ static int parse(struct nlattr *na, struct cpumask *mask) | |||
357 | return ret; | 357 | return ret; |
358 | } | 358 | } |
359 | 359 | ||
360 | #if defined(CONFIG_64BIT) && !defined(CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS) | ||
361 | #define TASKSTATS_NEEDS_PADDING 1 | ||
362 | #endif | ||
363 | |||
364 | static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid) | 360 | static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid) |
365 | { | 361 | { |
366 | struct nlattr *na, *ret; | 362 | struct nlattr *na, *ret; |
@@ -370,29 +366,6 @@ static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid) | |||
370 | ? TASKSTATS_TYPE_AGGR_PID | 366 | ? TASKSTATS_TYPE_AGGR_PID |
371 | : TASKSTATS_TYPE_AGGR_TGID; | 367 | : TASKSTATS_TYPE_AGGR_TGID; |
372 | 368 | ||
373 | /* | ||
374 | * The taskstats structure is internally aligned on 8 byte | ||
375 | * boundaries but the layout of the aggregrate reply, with | ||
376 | * two NLA headers and the pid (each 4 bytes), actually | ||
377 | * force the entire structure to be unaligned. This causes | ||
378 | * the kernel to issue unaligned access warnings on some | ||
379 | * architectures like ia64. Unfortunately, some software out there | ||
380 | * doesn't properly unroll the NLA packet and assumes that the start | ||
381 | * of the taskstats structure will always be 20 bytes from the start | ||
382 | * of the netlink payload. Aligning the start of the taskstats | ||
383 | * structure breaks this software, which we don't want. So, for now | ||
384 | * the alignment only happens on architectures that require it | ||
385 | * and those users will have to update to fixed versions of those | ||
386 | * packages. Space is reserved in the packet only when needed. | ||
387 | * This ifdef should be removed in several years e.g. 2012 once | ||
388 | * we can be confident that fixed versions are installed on most | ||
389 | * systems. We add the padding before the aggregate since the | ||
390 | * aggregate is already a defined type. | ||
391 | */ | ||
392 | #ifdef TASKSTATS_NEEDS_PADDING | ||
393 | if (nla_put(skb, TASKSTATS_TYPE_NULL, 0, NULL) < 0) | ||
394 | goto err; | ||
395 | #endif | ||
396 | na = nla_nest_start(skb, aggr); | 369 | na = nla_nest_start(skb, aggr); |
397 | if (!na) | 370 | if (!na) |
398 | goto err; | 371 | goto err; |
@@ -401,7 +374,8 @@ static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid) | |||
401 | nla_nest_cancel(skb, na); | 374 | nla_nest_cancel(skb, na); |
402 | goto err; | 375 | goto err; |
403 | } | 376 | } |
404 | ret = nla_reserve(skb, TASKSTATS_TYPE_STATS, sizeof(struct taskstats)); | 377 | ret = nla_reserve_64bit(skb, TASKSTATS_TYPE_STATS, |
378 | sizeof(struct taskstats), TASKSTATS_TYPE_NULL); | ||
405 | if (!ret) { | 379 | if (!ret) { |
406 | nla_nest_cancel(skb, na); | 380 | nla_nest_cancel(skb, na); |
407 | goto err; | 381 | goto err; |
@@ -500,10 +474,9 @@ static size_t taskstats_packet_size(void) | |||
500 | size_t size; | 474 | size_t size; |
501 | 475 | ||
502 | size = nla_total_size(sizeof(u32)) + | 476 | size = nla_total_size(sizeof(u32)) + |
503 | nla_total_size(sizeof(struct taskstats)) + nla_total_size(0); | 477 | nla_total_size_64bit(sizeof(struct taskstats)) + |
504 | #ifdef TASKSTATS_NEEDS_PADDING | 478 | nla_total_size(0); |
505 | size += nla_total_size(0); /* Padding for alignment */ | 479 | |
506 | #endif | ||
507 | return size; | 480 | return size; |
508 | } | 481 | } |
509 | 482 | ||