• Florian Westphal's avatar
    netfilter: bridge: convert skb_make_writable to skb_ensure_writable · c1a83116
    Florian Westphal authored
    Back in the day, skb_ensure_writable did not exist.  By now, both functions
    have the same precondition:
    
    I. skb_make_writable will test in this order:
      1. wlen > skb->len -> error
      2. if not cloned and wlen <= headlen -> OK
      3. If cloned and wlen bytes of clone writeable -> OK
    
    After those checks, skb is either not cloned but needs to pull from
    nonlinear area, or writing to head would also alter data of another clone.
    
    In both cases skb_make_writable will then call __pskb_pull_tail, which will
    kmalloc a new memory area to use for skb->head.
    
    IOW, after successful skb_make_writable call, the requested length is in
    linear area and can be modified, even if skb was cloned.
    
    II. skb_ensure_writable will do this instead:
       1. call pskb_may_pull.  This handles case 1 above.
          After this, wlen is in linear area, but skb might be cloned.
       2. return if skb is not cloned
       3. return if wlen byte of clone are writeable.
       4. fully copy the skb.
    
    So post-conditions are the same:
    *len bytes are writeable in linear area without altering any payload data
    of a clone, all header pointers might have been changed.
    
    Only differences are that skb_ensure_writable is in the core, whereas
    skb_make_writable lives in netfilter core and the inverted return value.
    skb_make_writable returns 0 on error, whereas skb_ensure_writable returns
    negative value.
    
    For the normal cases performance is similar:
    A. skb is not cloned and in linear area:
       pskb_may_pull is inline helper, so neither function copies.
    B. skb is cloned, write is in linear area and clone is writeable:
       both funcions return with step 3.
    
    This series removes skb_make_writable from the kernel.
    
    While at it, pass the needed value instead, its less confusing that way:
    There is no special-handling of "0-length" argument in either
    skb_make_writable or skb_ensure_writable.
    
    bridge already makes sure ethernet header is in linear area, only purpose
    of the make_writable() is is to copy skb->head in case of cloned skbs.
    Signed-off-by: default avatarFlorian Westphal <fw@strlen.de>
    Signed-off-by: default avatarPablo Neira Ayuso <pablo@netfilter.org>
    c1a83116
ebt_dnat.c 2.28 KB