本文主要是介绍1. Linux 内存的meminfo 获取过程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
1. Linux 内存的大小获取过程
硬件:tiny4412
软件:linux3.0.86,uboot-tiny4412
目的:梳理linux start-kernel阶段,获取内核大小的过程,并将内存bank个数,start ,size信息,放入meminfo中
tiny4412平台,传递内存的信息,是通过 u-boot,传递tag的方式来传递内存的大小的,而不是通过dtb的方式。
【start_kernel()->setup_arch()】
//首先判断是否从fdt中,获取machine desc mdesc = setup_machine_fdt(__atags_pointer);if (!mdesc)//以tags的方式,获取machine descmdesc = setup_machine_tags(machine_arch_type);
static struct machine_desc * __init setup_machine_tags(unsigned int nr)
{struct tag *tags = (struct tag *)&init_tags;struct machine_desc *mdesc = NULL, *p;char *from = default_command_line;init_tags.mem.start = PHYS_OFFSET;/** locate machine in the list of supported machines.*/for_each_machine_desc(p)if (nr == p->nr) {printk("Machine: %s\n", p->name);mdesc = p;break;}
...if (tags->hdr.tag == ATAG_CORE) {if (meminfo.nr_banks != 0)squash_mem_tags(tags);save_atags(tags);parse_tags(tags);}
首先会通过for_each_machine_desc查找对应的 machine ID。
machine ID 是放置在 .arch.info.init 这个section的数据。 通过MACHINE_START 注册的machine将会放在在 .arch.info.init 这个section 中。
以tag的形式传递过来的数据 ,不同的数据有不同的tag及进行标识
#define ATAG_CORE 0x54410001
#define ATAG_MEM 0x54410002
#define ATAG_CMDLINE 0x54410003
#define ATAG_RDIMG 0x54410004
parse_tags(tags);static void __init parse_tags(const struct tag *t)
{//编译传递过来的tagfor (; t->hdr.size; t = tag_next(t))if (!parse_tag(t))printk(KERN_WARNING"Ignoring unrecognised tag 0x%08x\n",t->hdr.tag);
}static int __init parse_tag(const struct tag *tag)
{extern struct tagtable __tagtable_begin, __tagtable_end;struct tagtable *t;for (t = &__tagtable_begin; t < &__tagtable_end; t++)if (tag->hdr.tag == t->tag) {t->parse(tag);break;}return t < &__tagtable_end;
}
在setup.c 中,会通过__tagtable 注册以下执行的函数,方式在taglist.init这个setction中,
struct tagtable {u32 tag;int (*parse)(struct tag *);
};#define __tag __used __attribute__((__section__(".taglist.init")))
#define __tagtable(tag, fn) \static struct tagtable __tagtable_##fn __tag = { tag, fn static int __init parse_tag_mem32(const struct tag *tag)
{return arm_add_memory(tag->u.mem.start, tag->u.mem.size);
}__tagtable(ATAG_MEM, parse_tag_mem32);
有u-boot传递过来的带有ATAG_MEM 的信息,将会执行 arm_add_memory函数
int __init arm_add_memory(phys_addr_t start, unsigned long size)
{struct membank *bank = &meminfo.bank[meminfo.nr_banks];if (meminfo.nr_banks >= NR_BANKS) {printk(KERN_CRIT "NR_BANKS too low, ""ignoring memory at 0x%08llx\n", (long long)start);return -EINVAL;}/** Ensure that start/size are aligned to a page boundary.* Size is appropriately rounded down, start is rounded up.*/size -= start & ~PAGE_MASK;bank->start = PAGE_ALIGN(start);bank->size = size & PAGE_MASK;/** Check whether this memory region has non-zero size or* invalid node number.*/if (bank->size == 0)return -EINVAL;meminfo.nr_banks++;return 0;
}
arm_add_memory函数,拿到由u-boot传递过来的哪个bank的start地址和长度,放到变量meminfo中。
这篇关于1. Linux 内存的meminfo 获取过程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!