Linux ip_-append_数据中的一个问题

Linux ip_-append_数据中的一个问题,linux,networking,kernel,Linux,Networking,Kernel,当我看到以下代码时,我对linux网络协议栈中的ip_append_数据感到困惑: if (!(rt->u.dst.dev->features&NETIF_F_SG)) { unsigned int off; off = skb->len; if (getfrag(from, skb_put(skb, copy), offset, copy, off, skb) < 0)

当我看到以下代码时,我对linux网络协议栈中的ip_append_数据感到困惑:

    if (!(rt->u.dst.dev->features&NETIF_F_SG)) {
        unsigned int off;

        off = skb->len;
        if (getfrag(from, skb_put(skb, copy), 
                offset, copy, off, skb) < 0) {
            __skb_trim(skb, off);
            err = -EFAULT;
            goto error;
        }
    } else {
        int i = skb_shinfo(skb)->nr_frags;
        skb_frag_t *frag = &skb_shinfo(skb)->frags[i-1];
        struct page *page = sk->sk_sndmsg_page;
        int off = sk->sk_sndmsg_off;
        unsigned int left;

        if (page && (left = PAGE_SIZE - off) > 0) {
            if (copy >= left)
                copy = left;
            if (page != frag->page) {
                if (i == MAX_SKB_FRAGS) {
                    err = -EMSGSIZE;
                    goto error;
                }
                get_page(page);
                skb_fill_page_desc(skb, i, page, sk->sk_sndmsg_off, 0);
                frag = &skb_shinfo(skb)->frags[i];
            }
        } else if (i < MAX_SKB_FRAGS) {
            if (copy > PAGE_SIZE)
                copy = PAGE_SIZE;
            page = alloc_pages(sk->sk_allocation, 0);
            if (page == NULL)  {
                err = -ENOMEM;
                goto error;
            }
            sk->sk_sndmsg_page = page;
            sk->sk_sndmsg_off = 0;

            skb_fill_page_desc(skb, i, page, 0, 0);
            frag = &skb_shinfo(skb)->frags[i];
            skb->truesize += PAGE_SIZE;
            atomic_add(PAGE_SIZE, &sk->sk_wmem_alloc);
        } else {
            err = -EMSGSIZE;
            goto error;
        }
        if (getfrag(from, page_address(frag->page)+frag->page_offset+frag->size, offset, copy, skb->len, skb) < 0) {
            err = -EFAULT;
            goto error;
        }
        sk->sk_sndmsg_off += copy;
        frag->size += copy;
        skb->len += copy;
        skb->data_len += copy;
    }
条件是
(page!=frag->page)
,为什么?是否可以用
(!frag->page)
?我认为
(page==frag->page)
(!frag->page)
为真

如果
(page==frag->page)
(!frag->page)
为真时始终为真,
(page!=frag->page)
(!frag->page)
为真时为假,将
(page!=frag->page)
替换为
(!frag->page)
将以真替换为假

如果你想写
(page!=frag->page)
(!frag->page)
为真时总是真的,如果这是对的,那就不足以让
(page!=frag->page)
等同于
(!frag->page)
,因为
(page!=frag->page)
时不一定总是假的(!frag->page)
为false

            if **(page != frag->page)** {
                if (i == MAX_SKB_FRAGS) {
                    err = -EMSGSIZE;
                    goto error;
                }
                get_page(page);
                skb_fill_page_desc(skb, i, page, sk->sk_sndmsg_off, 0);
                frag = &skb_shinfo(skb)->frags[i];
            }