2010-3-26 page和buffer_head

      首次看页高速缓存时,以为在页高速缓存中,每个page的private都指向buffer_head链表,但是阅读完do_mpage_readpage()函数后,发现并非如此。对于普通文件的页高速缓存,如果页中的段在磁盘上是连续的,那么page中的PG_private标志就不会置位,且private不会指向buffer_head链表。

      我们可以沿着这样的思路往下走:

      当函数do_generic_mapping_read()读取文件时,与读取位置对应的页不在高速缓存中时,该函数就向系统申请一个新页,并添加到radix_tree中,之后调用a_ops->readpage()从磁盘读取内容到该页中。

      a_ops->readpage()函数最后会调用do_mpage_read()函数来构造bio,通过对这个函数的分析可以得出radix_tree中的page不一定都与buffer_head关联。前面提到分配一个新页,是想强调这个页还没有与buffer_head关联。下面废话不多说,继续分析do_mapge_read()函数(昨天我已经发了对这个函数的注释,但是有些事理解错误的,今天将会重新发一个,当然也难免会有新的错误)。

      do_mapge_read()函数通过从get_block()(对应于ext2文件系统是ext2_getblock())得到的信息来判断页中的块是否连续如果连续则通过mpage_alloc()申请一个具有多个段(包含多个io_vec)的bio结构,然后调用bio_add_page()将该页添加到bio中。如果页不连续,则调用block_read_full_page()以每次一块的方式来读取该页,对于页中的每块将调用submit_bh()来生成并提交bio,对于submit_bh() 生成的bio中的io_vec数组只有一个元素。

/** * __netdev_alloc_skb - allocate an skbuff for rx on a specific device * @dev: network device to receive on * @len: length to allocate * @gfp_mask: get_free_pages mask, passed to alloc_skb * * Allocate a new &sk_buff and assign it a usage count of one. The * buffer has NET_SKB_PAD headroom built in. Users should allocate * the headroom they think they need without accounting for the * built in space. The built in space is used for optimisations. * * %NULL is returned if there is no free memory. */ struct sk_buff *__netdev_alloc_skb(struct net_device *dev, unsigned int len, gfp_t gfp_mask) { struct page_frag_cache *nc; struct sk_buff *skb; bool pfmemalloc; void *data; len += NET_SKB_PAD; /* If requested length is either too small or too big, * we use kmalloc() for skb->head allocation. */ if (len <= SKB_WITH_OVERHEAD(1024) || len > SKB_WITH_OVERHEAD(PAGE_SIZE) || (gfp_mask & (__GFP_DIRECT_RECLAIM | GFP_DMA))) { skb = __alloc_skb(len, gfp_mask, SKB_ALLOC_RX, NUMA_NO_NODE); if (!skb) goto skb_fail; goto skb_success; } len += SKB_DATA_ALIGN(sizeof(struct skb_shared_info)); len = SKB_DATA_ALIGN(len); if (sk_memalloc_socks()) gfp_mask |= __GFP_MEMALLOC; if (in_hardirq() || irqs_disabled()) { nc = this_cpu_ptr(&netdev_alloc_cache); data = page_frag_alloc(nc, len, gfp_mask); pfmemalloc = nc->pfmemalloc; } else { local_bh_disable(); nc = this_cpu_ptr(&napi_alloc_cache.page); data = page_frag_alloc(nc, len, gfp_mask); pfmemalloc = nc->pfmemalloc; local_bh_enable(); } if (unlikely(!data)) return NULL; skb = __build_skb(data, len); if (unlikely(!skb)) { skb_free_frag(data); return NULL; } if (pfmemalloc) skb->pfmemalloc = 1; skb->head_frag = 1; skb_success: skb_reserve(skb, NET_SKB_PAD); skb->dev = dev; skb_fail: return skb; } EXPORT_SYMBOL(__netdev_alloc_skb);这里返回的指针指向skb的头还是data部分
最新发布
03-21
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值