Jan Engelhardt wrote: > On Monday 2010-03-01 20:33, Pablo Neira Ayuso wrote: > >>>>>>> +#define ALIGN_SIZE 8 >>>>>> Minor question: why align this to 64 bits? >>>>> I originally used an alignment to 32 bits, but Jan noticed it would >>>>> break if using options/values on 64 bits (and a test confirmed that). I >>>>> took 64 bits as the biggest allowed value for integers. >>>> I would need to look into this in more detail, not sure where the >>>> problem is. I think that you can use something like `struct nlattr' (see >>>> include/linux/netlink.h) and then nla_put() to add attributes in the TLV >>>> format (see lib/nlattr.c). Those are align-safe. I'm using something >>>> similar for conntrackd for the synchronization messages (src/build.c and >>>> src/parse.c). > > If they are align-safe, what's this? :-) Could you tell me where did I say that libnetfilter_conntrack is using those functions that I recommended? > 18:41 ares:/home/jengelh # conntrack -L > Bus error > 18:41 ares:/home/jengelh # file `which conntrack` > /usr/sbin/conntrack: ELF 32-bit MSB executable, SPARC32PLUS, V8+ Required, > version 1 (SYSV), dynamically linked (uses shared libs), for GNU/Linux 2.6.4, > not stripped The following patch should fix the problem that you are reporting. Let me know if you have more issues.
parse: fix access to u64 attributes in netlink messages From: Pablo Neira Ayuso <pablo@xxxxxxxxxxxxx> This patch fixes parsing of 64 bits attributes (that are unaligned) in ctnetlink. It would be better to add nfnl_get_uX() functions similar to those in include/net/netlink.h to libnfnetlink to avoid this sort of errors. Reported-by: Jan Engelhardt <jengelh@xxxxxxxxxx> Signed-off-by: Pablo Neira Ayuso <pablo@xxxxxxxxxxxxx> --- src/conntrack/parse.c | 30 +++++++++++++++++++----------- 1 files changed, 19 insertions(+), 11 deletions(-) diff --git a/src/conntrack/parse.c b/src/conntrack/parse.c index 0e0cd58..60dabe4 100644 --- a/src/conntrack/parse.c +++ b/src/conntrack/parse.c @@ -276,9 +276,11 @@ static void __parse_protoinfo_dccp(const struct nfattr *attr, set_bit(ATTR_DCCP_ROLE, ct->set); } if (tb[CTA_PROTOINFO_DCCP_SEQ-1]) { - ct->protoinfo.dccp.handshake_seq = - __be64_to_cpu(*(u_int64_t *) - NFA_DATA(tb[CTA_PROTOINFO_DCCP_SEQ-1])); + u_int64_t tmp; + memcpy(&tmp, + NFA_DATA(tb[CTA_PROTOINFO_DCCP_SEQ-1]), + sizeof(tmp)); + ct->protoinfo.dccp.handshake_seq = __be64_to_cpu(tmp); set_bit(ATTR_DCCP_HANDSHAKE_SEQ, ct->set); } } @@ -314,10 +316,13 @@ static void __parse_counters(const struct nfattr *attr, = ntohl(*(u_int32_t *) NFA_DATA(tb[CTA_COUNTERS32_PACKETS-1])); - if (tb[CTA_COUNTERS_PACKETS-1]) - ct->counters[dir].packets - = __be64_to_cpu(*(u_int64_t *) - NFA_DATA(tb[CTA_COUNTERS_PACKETS-1])); + if (tb[CTA_COUNTERS_PACKETS-1]) { + u_int64_t tmp; + memcpy(&tmp, + NFA_DATA(tb[CTA_COUNTERS_PACKETS-1]), + sizeof(tmp)); + ct->counters[dir].packets = __be64_to_cpu(tmp); + } switch(dir) { case __DIR_ORIG: @@ -335,10 +340,13 @@ static void __parse_counters(const struct nfattr *attr, = ntohl(*(u_int32_t *) NFA_DATA(tb[CTA_COUNTERS32_BYTES-1])); - if (tb[CTA_COUNTERS_BYTES-1]) - ct->counters[dir].bytes - = __be64_to_cpu(*(u_int64_t *) - NFA_DATA(tb[CTA_COUNTERS_BYTES-1])); + if (tb[CTA_COUNTERS_BYTES-1]) { + u_int64_t tmp; + memcpy(&tmp, + NFA_DATA(tb[CTA_COUNTERS_BYTES-1]), + sizeof(tmp)); + ct->counters[dir].bytes = __be64_to_cpu(tmp); + } switch(dir) { case __DIR_ORIG: