⑴ vmlinux.lds.s怎么被编译成vmlinux.lds
vmlinux.lds.S用于对ld的输出进行组版,这个文件的格式在ld.info手册中有详细的说明。vmlinu
x.lds.S的主要目的是对输出文件中段进行排序,并定义相关的符号名,以下是简要注释。
/* ld script to make i386 Linux kernel
* Written by Martin Mares ;
*/
OUTPUT_FORMAT("elf32-i386", "elf32-i386", "elf32-i386")
OUTPUT_ARCH(i386) /* 输出格式 */
ENTRY(_start) /* 定义_start作为入口点 */
SECTIONS
{
. = PAGE_OFFSET_RAW + 0x100000; /* 定义当前段的偏移量(.代表当前计数器) */
_text = .; /* 定义符号_text为当前位置 */
.text : { /* 定义段.text (": {"是段定义符)*/
*(.text) /* 将所有输入文件中.text段合并到这里 */
*(.fixup) /* 将所有输入文件中的.fixup段合并到这里 */
*(.gnu.warning) /* 将所有输入文件中的.gnu.warning段合并到这里 */
} = 0x9090 /* 合并中的空隙用0x9090填充 */
/* 以下的语法含义可以类推 */
.text.lock : { *(.text.lock) } /* out-of-line lock text */
.rodata : { *(.rodata) }
.kstrtab : { *(.kstrtab) }
. = ALIGN(16); /* Exception table */
__start___ex_table = .; /* 定义__start_ex_table符号为当前位置 */
__ex_table : { *(__ex_table) }
__stop___ex_table = .;
__start___ksymtab = .; /* Kernel symbol table */
__ksymtab : { *(__ksymtab) }
__stop___ksymtab = .;
_etext = .; /* End of text section */
.data : { /* Data */
*(.data)
CONSTRUCTORS /* 将C++的构造函数指针段合并到这里 */
}
_edata = .; /* End of data section */
. = ALIGN(8192); /* init_task */
.data.init_task : { *(.data.init_task) }
. = ALIGN(4096); /* Init code and data */
__init_begin = .;
.text.init : { *(.text.init) }
.data.init : { *(.data.init) }
. = ALIGN(4096); /* 输出计数器在页边界上对齐 */
__init_end = .;
. = ALIGN(32);
.data.cacheline_aligned : { *(.data.cacheline_aligned) }
. = ALIGN(4096);
.data.page_aligned : { *(.data.idt) }
__bss_start = .; /* BSS */
.bss : {
*(.bss)
}
_end = . ;
/* Stabs debugging sections. */
.stab 0 : { *(.stab) } /* 0 是段属性,代表段的起始地址 */
.stabstr 0 : { *(.stabstr) }
.stab.excl 0 : { *(.stab.excl) }
.stab.exclstr 0 : { *(.stab.exclstr) }
.stab.index 0 : { *(.stab.index) }
.stab.indexstr 0 : { *(.stab.indexstr) }
.comment 0 : { *(.comment) }
}
⑵ Linux环境变量顺序
内核启动的时候,各个驱动初始化的工作在文件init/main.c中的do_basic_setup()函数中做.
------------------------------------------------------------------------------------------------------
static void __init do_basic_setup(void)
{
/* drivers will send hotplug events */
init_workqueues();
usermodehelper_init();
driver_init();
#ifdef CONFIG_SYSCTL
sysctl_init();
#endif
/* Networking initialization needs a process context */
sock_init();
do_initcalls();
}
------------------------------------------------------------------------------------------------------
其中的driver_init()做一些核心的初始化,看看代码就明白了.
相应的驱动程序的初始化在do_initcalls()中做.
------------------------------------------------------------------------------------------------------
static void __init do_initcalls(void)
{
initcall_t *call;
int count = preempt_count();
for (call = __initcall_start; call < __initcall_end; call++) {
char *msg;
if (initcall_debug) {
printk(KERN_DEBUG "Calling initcall 0x%p", *call);
print_fn_descriptor_symbol(": %s()", (unsigned long) *call);
printk("\n");
}
(*call)();
msg = NULL;
if (preempt_count() != count) {
msg = "preemption imbalance";
preempt_count() = count;
}
if (irqs_disabled()) {
msg = "disabled interrupts";
local_irq_enable();
}
if (msg) {
printk(KERN_WARNING "error in initcall at 0x%p: "
"returned with %s\n", *call, msg);
}
}
/* Make sure there is no pending stuff from the initcall sequence */
flush_scheled_work();
}
------------------------------------------------------------------------------------------------------
这个__initcall_start是在文件 arch/xxx/kernel/vmlinux.lds.S (其中的xxx 是你的体系结构的名称,例如i386)
这个文件是内核ld的时候使用的.其中定义了各个sectioin,看看就明白了。
在这个文件中有个.initcall.init, 代码如下:
------------------------------------------------------------------------------------------------------
__initcall_start = .;
.initcall.init : {
*(.initcall1.init)
*(.initcall2.init)
*(.initcall3.init)
*(.initcall4.init)
*(.initcall5.init)
*(.initcall6.init)
*(.initcall7.init)
}
------------------------------------------------------------------------------------------------------
这里有7个初始化的优先级,内核会按照这个优先级的顺序依次加载.
这些优先级是在文件include/linux/init.h 中定义的. 你注意一下宏 __define_initcall的实现就明白了.
相关代码如下:
#define __define_initcall(level,fn) \
static initcall_t __initcall_##fn __attribute_used__ \
__attribute__((__section__(".initcall" level ".init"))) = fn
#define core_initcall(fn) __define_initcall("1",fn)
#define postcore_initcall(fn) __define_initcall("2",fn)
#define arch_initcall(fn) __define_initcall("3",fn)
#define subsys_initcall(fn) __define_initcall("4",fn)
#define fs_initcall(fn) __define_initcall("5",fn)
#define device_initcall(fn) __define_initcall("6",fn)
#define late_initcall(fn) __define_initcall("7",fn)
我们可以看到,我们经常写的设备驱动程序中常用的mole_init其实就是对应了优先级6:
#define __initcall(fn) device_initcall(fn)
#define mole_init(x) __initcall(x);
文章出处:http://www.diybl.com/course/6_system/linux/Linuxjs/2008628/128990.html
请采纳。
⑶ linux 下的lds链接脚本有没有什么书籍
Linux下有两个清屏命令: clear 这个命令将会刷新屏幕,本质上只是让终端显示页向后翻了一页,如果向上滚动屏幕还可以看到之前的操作信息。一般都会使用这个命令。 reset 这个命令将完全刷新终端屏幕,之前的终端输入操作信息将都会被清空,这样虽然比较清爽,但整个命令过程速度有点慢,使用较少。
⑷ linux lds 脚本是直接生成吗
Linux中,周期执行的任务一般由cron这个守护进程来处理ps-ef|grepcroncron读取一个或多个配置文件,这些配置文件中包含了命令行及其调用时间。cron的配置文件称为“crontab”,是“crontable”的简写。一、cron在3个地方查找配置文件(设置shell脚本):1、/var/spool/cron/yanggang这个目录下存放的是每个用户(包括root)的crontab任务,每个任务以创建者的名字命名,比如用户tom建的crontab任务对应的文件就是/var/spool/cron/tomyanggang@barry$sudols-l/var/spool/cron/(或有时是/var/spool/cron/crontabs/)-rw-------1rootcrontab10712011-09-1917:20root-rw-------1yanggangcrontab11762011-09-2411:07yanggang一般一个用户最多只有一个crontab文件(如:root,yanggang等),其对应日志在/var/spool/mail/root(或/var/spool/mail/yanggang)文件里2、/etc/crontab这个文件负责安排由系统管理员制定的维护系统以及其他任务的crontab。SHELL=/bin/bashPATH=/sbin:/bin:/usr/sbin:/usr/binMAILTO=rootHOME=/#.----------------minute(0-59)#|.-------------hour(0-23)#||.----------dayofmonth(1-31)#|||.-------month(1-12)ORjan,feb,mar,apr#||||.----dayofweek(0-6)(Sunday=0or7)OR#sun,mon,tue,wed,thu,fri,sat#|||||#*****commandtobeexecuted例如:*/5****root/usr/libexec/atrunminute:代表一小时内的第几分,范围0-59。hour:代表一天中的第几小时,范围0-23。mday:代表一个月中的第几天,范围1-31。month:代表一年中第几个月,范围1-12。wday:代表星期几,范围0-7(0及7都是星期天)。who:要使用什么身份执行该指令,当您使用crontab-e时,不必加此字段。command:所要执行的指令。3、/etc/cron.d/这个目录用来存放任何要执行的crontab文件或脚本。二、权限crontab权限问题到/var/adm/cron/下一看,文件cron.allow和cron.deny是否存在用法如下:1、如果两个文件都不存在,则只有root用户才能使用crontab命令。2、如果cron.allow存在但cron.deny不存在,则只有列在cron.allow文件里的用户才能使用crontab命令,如果root用户也不在里面,则root用户也不能使用crontab。3、如果cron.allow不存在,cron.deny存在,则只有列在cron.deny文件里面的用户不能使用crontab命令,其它用户都能使用。4、如果两个文件都存在,则列在cron.allow文件中而且没有列在cron.deny中的用户可以使用crontab,如果两个文件中都有同一个用户,以cron.allow文件里面是否有该用户为准,如果cron.allow中有该用户,则可以使用crontab命令。AIX中普通用户默认都有crontab权限,如果要限制用户使用crontab,就需要编辑/var/adm/cron/cron.denyHP-UNIX中默认普通用户没得crontab权限,要想放开普通用户的crontab权限可以编三、创建cron脚本第一步:写cron脚本文件,命名为crontest.cron。15,30,45,59****echo"xgmtest..">>xgmtest.txt表示,每隔15分钟,执行打印一次命令第二步:添加定时任务。执行命令“crontabcrontest.cron”。搞定第三步:"crontab-l"查看定时任务是否成功或者检测/var/spool/cron下是否生成对应cron脚本四、cron服务cron是一个linux下的定时执行工具,可以在无需人工干预的情况下运行作业。/sbin/servicecrondstart//启动服务/sbin/servicecrondstop//关闭服务/sbin/servicecrondrestart//重启服务/sbin/servicecrondreload//重新载入配置/sbin/servicecrondstatus//查看服务状态例如:使用sudo停止与启动服务yanggang@barry$sudoservicecronstopcronstop/waitingyanggang@barry$sudoservicecronstartcronstart/running,process7502五、crontab用法crontab命令用于安装、删除或者列出用于驱动cron后台进程的表格,用户把需要执行的命令序列放到crontab文件中以获得执行。每个用户都可以有自己的crontab文件,/var/spool/cron下的crontab文件不可以直接创建或者直接修改,该crontab文件是通过crontab-e命令创建的在crontab文件中如何输入需要执行的命令和时间,该文件中每行都包括六个域,其中前五个域是指定命令被执行的时间,最后一个域是要被执行的命令。每个域之间使用空格或者制表符分隔。格式如下:minutehourday-of-monthmonth-of-yearday-of-weekcommands合法值00-5900-2301-3101-120-6(0issunday)除了数字还有几个个特殊的符号就是"*"、"/"与"-"、",",*代表所有的取值范围内的数字"/"代表每的意思,"/5"表示每5个单位,"-"代表从某个数字到某个数字,","分开几个离散的数字。-l在标准输出上显示当前的crontab。-r删除当前的crontab文件。-e使用VISUAL或者EDITOR环境变量所指的编辑器编辑当前的crontab文件。当结束编辑离开时,编辑后的文件将自动安装。几个例子:每天早上6点06***echo"Goodmorning.">>/tmp/test.txt//注意单纯echo,从屏幕上看不到任何输出,因为cron把任何输出都email到root的信箱了。每两个小时0*/2***echo"Haveabreaknow.">>/tmp/test.txt晚上11点到早上8点之间每两个小时和早上八点023-7/2,8***echo"Haveagooddream">>/tmp/test.txt每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点0114*1-3commandline1月1日早上4点0411*commandlineSHELL=/bin/bashPATH=/sbin:/bin:/usr/sbin:/usr/binMAILTO=root//如果出现错误,或者有数据输出,数据作为邮件发给这个帐号HOME=/每小时(第一分钟)执行/etc/cron.hourly内的脚本01****rootrun-parts/etc/cron.hourly每天(凌晨4:02)执行/etc/cron.daily内的脚本024***rootrun-parts/etc/cron.daily每星期(周日凌晨4:22)执行/etc/cron.weekly内的脚本224**0rootrun-parts/etc/cron.weekly每月(1号凌晨4:42)去执行/etc/cron.monthly内的脚本4241**rootrun-parts/etc/cron.monthly注意:"run-parts"这个参数了,如果去掉这个参数的话,后面就可以写要运行的某个脚本名,而不是文件夹名。每天的下午4点、5点、6点的5min、15min、25min、35min、45min、55min时执行命令。5,15,25,35,45,5516,17,18***command每周一,三,五的下午3:00系统进入维护状态,重新启动系统。0015**1,3,5shutdown-r+5每小时的10分,40分执行用户目录下的innd/bbslin这个指令:10,40****innd/bbslink每小时的1分执行用户目录下的bin/account这个指令:1****bin/account每天早晨三点二十分执行用户目录下如下所示的两个指令(每个指令以;分隔):203***(/bin/rm-fexpire.lslogins.bad;bin/expire$#@62;expire.1st)每年的一月和四月,4号到9号的3点12分和3点55分执行/bin/rm-fexpire.1st这个指令,并把结果添加在mm.txt这个文件之后(mm.txt文件位于用户自己的目录位置)。12,5534-91,4*/bin/rm-fexpire.1st$#@62;$#@62;mm.txt六、几个问题crond进程没有运行1.启动crond进程/etc/init.d/crondstart2.开机自启动crond进程chkconfigcrondoncrontab编辑:输入编辑命令:crontab-e#mhdommondowcommandSHELL=/bin/bash30****cd/home/barry/top800/top10/top10_fruits/&&./top10_all.shCtrl+O(写入)——》enter键(保存文件名)——》Ctrl+X(退出)输入查看命令:crontab-l#mhdommondowcommandSHELL=/bin/bash30****cd/home/barry/top800/top10/top10_fruits/&&./top10_all.sh(建议使用此方式)语法错误:Syntaxerror:"("unexpected路径错误:在/var/spool/crontab/yanggang中,添加了如下命令,在日志文件/var/spool/mail/yanggang中提示找不到xxx.sh路径30****/home/barry/top800/top10/top10_fruits/top10_all.sh或30****bash/home/barry/top800/top10/top10_fruits/top10_all.sh这是因为在crontab中使用了绝对路径执行脚本top10_all.sh,因此在脚本top10_all.sh中引用的其它脚本也都需要使用绝对路径,才能被crontab找到并执行如何避免绝对路径复杂的设置呢,如上文六、几个问题所示,采用如下格式:30****cd/home/barry/top800/top10/top10_fruits/&&./top10_all.sh(建议使用此方式,先进入该目录,然后在执行脚本;否则,执行脚本中的其它脚本都需要加绝对路径)
⑸ lds是什么文件
下面是vmlinux.lds.S文件,我老是觉得晕晕的,有哪位大虾能解释一下或推荐一个有相关学习资料的站点?
/* ld script to make i386 Linux kernel
* Written by Martin Mares ;
*/
OUTPUT_FORMAT("elf32-i386", "elf32-i386", "elf32-i386")
OUTPUT_ARCH(i386)
ENTRY(_start)
SECTIONS
{
. = PAGE_OFFSET_RAW + 0x100000;
_text = .; /* Text and read-only data */
.text : {
*(.text)
*(.fixup)
*(.gnu.warning)
} = 0x9090
.text.lock : { *(.text.lock) } /* out-of-line lock text */
.rodata : { *(.rodata) }
.kstrtab : { *(.kstrtab) }
. = ALIGN(16); /* Exception table */
__start___ex_table = .;
__ex_table : { *(__ex_table) }
__stop___ex_table = .;
__start___ksymtab = .; /* Kernel symbol table */
__ksymtab : { *(__ksymtab) }
__stop___ksymtab = .;
_etext = .; /* End of text section */
.data : { /* Data */
*(.data)
CONSTRUCTORS
}
_edata = .; /* End of data section */
. = ALIGN(8192); /* init_task */
.data.init_task : { *(.data.init_task) }
. = ALIGN(4096); /* Init code and data */
__init_begin = .;
.text.init : { *(.text.init) }
.data.init : { *(.data.init) }
. = ALIGN(4096);
__init_end = .;
. = ALIGN(32);
.data.cacheline_aligned : { *(.data.cacheline_aligned) }
. = ALIGN(4096);
.data.page_aligned : { *(.data.idt) }
__bss_start = .; /* BSS */
.bss : {
*(.bss)
}
_end = . ;
/* Stabs debugging sections. */
.stab 0 : { *(.stab) }
.stabstr 0 : { *(.stabstr) }
.stab.excl 0 : { *(.stab.excl) }
.stab.exclstr 0 : { *(.stab.exclstr) }
.stab.index 0 : { *(.stab.index) }
.stab.indexstr 0 : { *(.stab.indexstr) }
.comment 0 : { *(.comment) }
}
⑹ 如何解读vmlinux.lds.S
vmlinux.lds.S用于对ld的输出进行组版,这个文件的格式在ld.info手册中有详细的说明。vmlinu x.lds.S的主要目的是对输出文件中段进行排序,并定义相关的符号名,以下是简要注释。 /* ld script to make i386 Linux kernel * Written by Martin M...