taskstats: pad taskstats netlink response for aligment issues on ia64

The taskstats structure is internally aligned on 8 byte boundaries but the
layout of the aggregrate reply, with two NLA headers and the pid (each 4
bytes), actually force the entire structure to be unaligned.  This causes
the kernel to issue unaligned access warnings on some architectures like
ia64.  Unfortunately, some software out there doesn't properly unroll the
NLA packet and assumes that the start of the taskstats structure will
always be 20 bytes from the start of the netlink payload.  Aligning the
start of the taskstats structure breaks this software, which we don't
want.  So, for now the alignment only happens on architectures that
require it and those users will have to update to fixed versions of those
packages.  Space is reserved in the packet only when needed.  This ifdef
should be removed in several years e.g.  2012 once we can be confident
that fixed versions are installed on most systems.  We add the padding
before the aggregate since the aggregate is already a defined type.

Commit 85893120 ("delayacct: align to 8 byte boundary on 64-bit systems")
previously addressed the alignment issues by padding out the pid field.
This was supposed to be a compatible change but the circumstances
described above mean that it wasn't.  This patch backs out that change,
since it was a hack, and introduces a new NULL attribute type to provide
the padding.  Padding the response with 4 bytes avoids allocating an
aligned taskstats structure and copying it back.  Since the structure
weighs in at 328 bytes, it's too big to do it on the stack.

Signed-off-by: Jeff Mahoney <jeffm@suse.com>
Reported-by: Brian Rogers <brian@xyzw.org>
Cc: Jeff Mahoney <jeffm@suse.com>
Cc: Guillaume Chazarain <guichaz@gmail.com>
Cc: Balbir Singh <balbir@in.ibm.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
Jeff Mahoney 2010-12-21 17:24:30 -08:00 committed by Linus Torvalds
parent 4e06fd14d5
commit 4be2c95d1f
3 changed files with 47 additions and 14 deletions

View File

@ -516,6 +516,7 @@ int main(int argc, char *argv[])
default: default:
fprintf(stderr, "Unknown nla_type %d\n", fprintf(stderr, "Unknown nla_type %d\n",
na->nla_type); na->nla_type);
case TASKSTATS_TYPE_NULL:
break; break;
} }
na = (struct nlattr *) (GENLMSG_DATA(&msg) + len); na = (struct nlattr *) (GENLMSG_DATA(&msg) + len);

View File

@ -33,7 +33,7 @@
*/ */
#define TASKSTATS_VERSION 7 #define TASKSTATS_VERSION 8
#define TS_COMM_LEN 32 /* should be >= TASK_COMM_LEN #define TS_COMM_LEN 32 /* should be >= TASK_COMM_LEN
* in linux/sched.h */ * in linux/sched.h */
@ -188,6 +188,7 @@ enum {
TASKSTATS_TYPE_STATS, /* taskstats structure */ TASKSTATS_TYPE_STATS, /* taskstats structure */
TASKSTATS_TYPE_AGGR_PID, /* contains pid + stats */ TASKSTATS_TYPE_AGGR_PID, /* contains pid + stats */
TASKSTATS_TYPE_AGGR_TGID, /* contains tgid + stats */ TASKSTATS_TYPE_AGGR_TGID, /* contains tgid + stats */
TASKSTATS_TYPE_NULL, /* contains nothing */
__TASKSTATS_TYPE_MAX, __TASKSTATS_TYPE_MAX,
}; };

View File

@ -349,25 +349,47 @@ static int parse(struct nlattr *na, struct cpumask *mask)
return ret; return ret;
} }
#ifdef CONFIG_IA64
#define TASKSTATS_NEEDS_PADDING 1
#endif
static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid) static struct taskstats *mk_reply(struct sk_buff *skb, int type, u32 pid)
{ {
struct nlattr *na, *ret; struct nlattr *na, *ret;
int aggr; int aggr;
/* If we don't pad, we end up with alignment on a 4 byte boundary.
* This causes lots of runtime warnings on systems requiring 8 byte
* alignment */
u32 pids[2] = { pid, 0 };
int pid_size = ALIGN(sizeof(pid), sizeof(long));
aggr = (type == TASKSTATS_TYPE_PID) aggr = (type == TASKSTATS_TYPE_PID)
? TASKSTATS_TYPE_AGGR_PID ? TASKSTATS_TYPE_AGGR_PID
: TASKSTATS_TYPE_AGGR_TGID; : TASKSTATS_TYPE_AGGR_TGID;
/*
* The taskstats structure is internally aligned on 8 byte
* boundaries but the layout of the aggregrate reply, with
* two NLA headers and the pid (each 4 bytes), actually
* force the entire structure to be unaligned. This causes
* the kernel to issue unaligned access warnings on some
* architectures like ia64. Unfortunately, some software out there
* doesn't properly unroll the NLA packet and assumes that the start
* of the taskstats structure will always be 20 bytes from the start
* of the netlink payload. Aligning the start of the taskstats
* structure breaks this software, which we don't want. So, for now
* the alignment only happens on architectures that require it
* and those users will have to update to fixed versions of those
* packages. Space is reserved in the packet only when needed.
* This ifdef should be removed in several years e.g. 2012 once
* we can be confident that fixed versions are installed on most
* systems. We add the padding before the aggregate since the
* aggregate is already a defined type.
*/
#ifdef TASKSTATS_NEEDS_PADDING
if (nla_put(skb, TASKSTATS_TYPE_NULL, 0, NULL) < 0)
goto err;
#endif
na = nla_nest_start(skb, aggr); na = nla_nest_start(skb, aggr);
if (!na) if (!na)
goto err; goto err;
if (nla_put(skb, type, pid_size, pids) < 0)
if (nla_put(skb, type, sizeof(pid), &pid) < 0)
goto err; goto err;
ret = nla_reserve(skb, TASKSTATS_TYPE_STATS, sizeof(struct taskstats)); ret = nla_reserve(skb, TASKSTATS_TYPE_STATS, sizeof(struct taskstats));
if (!ret) if (!ret)
@ -456,6 +478,18 @@ out:
return rc; return rc;
} }
static size_t taskstats_packet_size(void)
{
size_t size;
size = nla_total_size(sizeof(u32)) +
nla_total_size(sizeof(struct taskstats)) + nla_total_size(0);
#ifdef TASKSTATS_NEEDS_PADDING
size += nla_total_size(0); /* Padding for alignment */
#endif
return size;
}
static int cmd_attr_pid(struct genl_info *info) static int cmd_attr_pid(struct genl_info *info)
{ {
struct taskstats *stats; struct taskstats *stats;
@ -464,8 +498,7 @@ static int cmd_attr_pid(struct genl_info *info)
u32 pid; u32 pid;
int rc; int rc;
size = nla_total_size(sizeof(u32)) + size = taskstats_packet_size();
nla_total_size(sizeof(struct taskstats)) + nla_total_size(0);
rc = prepare_reply(info, TASKSTATS_CMD_NEW, &rep_skb, size); rc = prepare_reply(info, TASKSTATS_CMD_NEW, &rep_skb, size);
if (rc < 0) if (rc < 0)
@ -494,8 +527,7 @@ static int cmd_attr_tgid(struct genl_info *info)
u32 tgid; u32 tgid;
int rc; int rc;
size = nla_total_size(sizeof(u32)) + size = taskstats_packet_size();
nla_total_size(sizeof(struct taskstats)) + nla_total_size(0);
rc = prepare_reply(info, TASKSTATS_CMD_NEW, &rep_skb, size); rc = prepare_reply(info, TASKSTATS_CMD_NEW, &rep_skb, size);
if (rc < 0) if (rc < 0)
@ -570,8 +602,7 @@ void taskstats_exit(struct task_struct *tsk, int group_dead)
/* /*
* Size includes space for nested attributes * Size includes space for nested attributes
*/ */
size = nla_total_size(sizeof(u32)) + size = taskstats_packet_size();
nla_total_size(sizeof(struct taskstats)) + nla_total_size(0);
is_thread_group = !!taskstats_tgid_alloc(tsk); is_thread_group = !!taskstats_tgid_alloc(tsk);
if (is_thread_group) { if (is_thread_group) {