- 论坛徽章:
- 0
|
刚分析do_munmap()代码,对其中一处代码十分不解:- int do_munmap(struct mm_struct *mm, unsigned long addr, size_t len)
- {
- struct vm_area_struct *mpnt, *prev, **npp, *free, *extra;
- if ((addr & ~PAGE_MASK) || addr > TASK_SIZE || len > TASK_SIZE-addr)
- return -EINVAL;
- if ((len = PAGE_ALIGN(len)) == 0)
- return -EINVAL;
- /* Check if this memory area is ok - put it on the temporary
- * list if so.. The checks here are pretty simple --
- * every area affected in some way (by any overlap) is put
- * on the list. If nothing is put on, nothing is affected.
- */
- mpnt = find_vma_prev(mm, addr, &prev);
- if (!mpnt)
- return 0;
- /* we have addr < mpnt->vm_end */
- if (mpnt->vm_start >= addr+len)//落入了空洞中,直接返回。
- return 0;
- /* If we'll make "hole", check the vm areas limit */
- if ((mpnt->vm_start < addr && mpnt->vm_end > addr+len)
- && mm->map_count >= max_map_count)//要拆分VMA,则势必要多出一个VMA,看是否超出个数限制。
- return -ENOMEM;
- /*
- * We may need one additional vma to fix up the mappings ...
- * and this is the last chance for an easy error exit.
- */
- extra = kmem_cache_alloc(vm_area_cachep, SLAB_KERNEL);
- if (!extra)
- return -ENOMEM;
- npp = (prev ? &prev->vm_next : &mm->mmap);
- free = NULL; //要删掉的VMA形成一个链表,由free指向。
- spin_lock(&mm->page_table_lock);
- for ( ; mpnt && mpnt->vm_start < addr+len; mpnt = *npp) {
- *npp = mpnt->vm_next;
- mpnt->vm_next = free;
- free = mpnt;
- rb_erase(&mpnt->vm_rb, &mm->mm_rb);
- }
- mm->mmap_cache = NULL; /* Kill the cache. */
- spin_unlock(&mm->page_table_lock);
- /* Ok - we have the memory areas we should free on the 'free' list,
- * so release them, and unmap the page range..
- * If the one of the segments is only being partially unmapped,
- * it will put new vm_area_struct(s) into the address space.
- * In that case we have to be careful with VM_DENYWRITE.
- */ //现在要删掉的VMA在free指向的链表中
- while ((mpnt = free) != NULL) {
- unsigned long st, end, size;
- struct file *file = NULL;
- free = free->vm_next;
- st = addr < mpnt->vm_start ? mpnt->vm_start : addr;
- end = addr+len;
- end = end > mpnt->vm_end ? mpnt->vm_end : end;
- size = end - st;
- if (mpnt->vm_flags & VM_DENYWRITE &&
- (st != mpnt->vm_start || end != mpnt->vm_end) &&
- (file = mpnt->vm_file) != NULL) {
- atomic_dec(&file->f_dentry->d_inode->i_writecount);
- }
- remove_shared_vm_struct(mpnt);
- mm->map_count--;
- zap_page_range(mm, st, size);
- /*
- * Fix the mapping, and free the old area if it wasn't reused.
- */
- extra = unmap_fixup(mm, mpnt, st, size, extra);
- if (file) //这个file一定继续有效吗?????????前面没有递增file的引用计数,但是unmap_fixup()里面却是有可能fput()的。。不解。。
- atomic_inc(&file->f_dentry->d_inode->i_writecount);
- }
- validate_mm(mm);
- /* Release the extra vma struct if it wasn't used */
- if (extra)
- kmem_cache_free(vm_area_cachep, extra);
- free_pgtables(mm, prev, addr, addr+len);
- return 0;
- }
复制代码- static struct vm_area_struct * unmap_fixup(struct mm_struct *mm,
- struct vm_area_struct *area, unsigned long addr, size_t len,
- struct vm_area_struct *extra)
- {//撤销一个VMA中[addr, addr+len)这段线性地址。如果撤销这范围的地址刚好使VMA一分为2(刚好撤销中间的),则使用extra,并返回NULL。
- struct vm_area_struct *mpnt;
- unsigned long end = addr + len;
- area->vm_mm->total_vm -= len >> PAGE_SHIFT;
- if (area->vm_flags & VM_LOCKED)
- area->vm_mm->locked_vm -= len >> PAGE_SHIFT;
- /* Unmapping the whole area. */
- if (addr == area->vm_start && end == area->vm_end) { //撤销整个VMA
- if (area->vm_ops && area->vm_ops->close)
- area->vm_ops->close(area); //如果有要调用close()方法
- if (area->vm_file)
- fput(area->vm_file);
- kmem_cache_free(vm_area_cachep, area);
- return extra; //返回非NULL,则说明extra没有使用(即没有增加VMA)
- }
- /* Work out to one of the ends. */
- if (end == area->vm_end) {//只撤销VMA尾部一段
- /*
- * here area isn't visible to the semaphore-less readers
- * so we don't need to update it under the spinlock.
- */
- area->vm_end = addr;
- lock_vma_mappings(area);
- spin_lock(&mm->page_table_lock);
- } else if (addr == area->vm_start) {//只撤销VMA头部一段
- area->vm_pgoff += (end - area->vm_start) >> PAGE_SHIFT;
- /* same locking considerations of the above case */
- area->vm_start = end;
- lock_vma_mappings(area);
- spin_lock(&mm->page_table_lock);
- } else {//撤销VMA中部一段,则要多出一个额外的VMA并插入
- /* Unmapping a hole: area->vm_start < addr <= end < area->vm_end */
- /* Add end mapping -- leave beginning for below */
- mpnt = extra;
- extra = NULL;
- mpnt->vm_mm = area->vm_mm;
- mpnt->vm_start = end;
- mpnt->vm_end = area->vm_end;
- mpnt->vm_page_prot = area->vm_page_prot;
- mpnt->vm_flags = area->vm_flags;
- mpnt->vm_raend = 0;
- mpnt->vm_ops = area->vm_ops;
- mpnt->vm_pgoff = area->vm_pgoff + ((end - area->vm_start) >> PAGE_SHIFT);
- mpnt->vm_file = area->vm_file;
- mpnt->vm_private_data = area->vm_private_data;
- if (mpnt->vm_file)
- get_file(mpnt->vm_file);
- if (mpnt->vm_ops && mpnt->vm_ops->open)
- mpnt->vm_ops->open(mpnt);
- area->vm_end = addr; /* Truncate area */
- /* Because mpnt->vm_file == area->vm_file this locks
- * things correctly.
- */
- lock_vma_mappings(area);
- spin_lock(&mm->page_table_lock);
- __insert_vm_struct(mm, mpnt);
- }
- __insert_vm_struct(mm, area);
- spin_unlock(&mm->page_table_lock);
- unlock_vma_mappings(area);
- return extra;
- }
复制代码 疑问在:- if (file)
- atomic_inc(&file->f_dentry->d_inode->i_writecount);
复制代码 这个file一定继续有效吗?????????前面没有临时递增file的引用计数,但是如果某个VMA全部落在要撤销的地址范围之内,则这个VMA要释放掉,同时file被fput掉。还是我忽略了某些地方?
还有这个递增动作,是不是和前面的递减不对称?
内核版本:2.4.22 |
|