aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/taskstats.c
diff options
context:
space:
mode:
authorNicolas Dichtel <nicolas.dichtel@6wind.com>2016-04-22 11:31:24 -0400
committerDavid S. Miller <davem@davemloft.net>2016-04-23 20:13:25 -0400
commit80df554275c21edca22ece02448bdb378c2ee9f1 (patch)
treeea4a3f342c07929025c193b2b67e206e1008afa9 /kernel/taskstats.c
parentde95c4a46a6e608444ccaf541087594553c7df11 (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.c37
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
364static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid) 360static 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