嵌入式Linux 高级阶段笔记

嵌入式Linux 高级阶段笔记
嵌入式Linux 高级阶段笔记

Linux C编程学习

1.C++ list使用

list初始化,并定义一个链表。

list list_student;

Struct Student_st *pStu = (Struct Student_st *)malloc(sizeof(Struct Student_st));

pStu->no = 100;

list_student. push_back(pStu);

链表中获取一个结构体对象。

list::itegrator it = list_student.begin();

Struct Student_st *pTmp =*it; //注意it存储的是指针的指针,*it才是对象指针

链表排序。sort() 和sort(comp compFun) –不建议使用排序

默认的sort()仅仅进行值比较,及< 、> 或者=。sort(Compare comp) 采取的是把比较函数传入。

2.C++ Hash_Map使用

引入库:#include

及命名空间:using namespace __gnu_cxx;

HashMap 不能够储存重复值。

定义结构类型:hash_map _clientMap;

添加记录:_clientMap.insert(pair(psamNo,psam));

--注意: 必须以键值对的方式添加。

返回值是一个键值对,存储了添加是否成功标

识。

pair insert ( const value_type& x );

例子:

ret=mymap.insert (pair(‘z’,500) );

if (ret.second==true)

printf(“OK\n”);

删除记录:可以有下列方式删除,通过位置、起始位置到终止位置及键值。void erase( iterator pos );

void erase( iterator start, iterator end );

size_type erase( const KEY_TYPE &key );

size_type 删除元素个数

清除记录:删除所有记录。

void clear();

查询记录:注意若没有查询到返回指向Map尾部的迭代器。

iterator find( const KEY_TYPE &key );

遍历记录:使用迭代器访问每一个元素。it->first it->second获取key和value P_PSAMINFO psam = NULL;

hash_map::iterator it = _clientMap.begin();

while(it != _clientMap.end()){

psam = it->second;

printf("psamcode=%s ,psamsocket

=%d\n",psam->psamcode,psam->getFD());

it++;

free(psam);

}

3.static_cast使用

将一个值以合逻辑的方式转换。这可看作是【利用原值重建一个临时物件,并在设立初值时使用类型转换】。意味着有使用临时对象。

例子:

float x = 100.1;

cout<< static_cast(x); //print x as int

f( static_cast("Hello")); // call f() for string instead of char*

4.dynamic_cast使用

将类型向下转换为其实际类型。这是唯一在执行期进行检验的转换动作。你可以用它来检验某个类型,例如:

5.socket函数

5.1socket函数

int socket(int domain, int type, int protocol);

参数1:

windows下目前仅支持AF_INET格式,也就是说ARPA Internet地址格式。linux 下支持PF_INET和AF_INET.

参数2:

新套接口的类型描述。SOCK_STREAM(TCP模式)SOCK_DGRAM(UDP模式) SOCK_RAW(原始套接字)

参数3:

protocol:特殊用途, 默认用缺省值0

返回值:成功则返回新的socket文件描述,失败则返回-1

例子:int socket_fd = socket(AF_INET, SOCK_STREAM, 0);

5.2bind函数

int bind(int sockfd, struct sockaddr* addr, socklen_t addrlen);

参数sockfd

指定地址与哪个套接字绑定,这是一个由之前的socket函数调用返回的套接字。调用bind的函数之后,该套接字与一个相应的地址关联,发送到这个地址的数据可以通过这个套接字来读取与使用。

参数addr

指定地址。这是一个地址结构,并且是一个已经经过填写的有效的地址结构。调用bind之后这个地址与参数sockfd指定的套接字关联,从而实现上面所说的效果。

//旧的结构体

struct sockaddr {

u_short sa_family;

char sa_data[14];

};

//新的结构体

struct sockaddr_in {

short int sin_family; /* Address family */

unsigned short int sin_port; /* Port number */

struct in_addr sin_addr; /* Internet address */

unsigned char sin_zero[8]; /* Same size as struct sockaddr */

};

sin_family指代协议族,在socket编程中只能是AF_INET

sin_port存储端口号(使用网络字节顺序)

sin_addr存储IP地址,使用in_addr这个数据结构

sin_zero是为了让sockaddr与sockaddr_in两个数据结构保持大小相同而保留的空字节。

s_addr按照网络字节顺序存储IP地址

sockaddr_in和sockaddr是并列的结构,指向sockaddr_in的结构体的指针也可以指向sockaddr的结构体,并代替它。也就是说,你可以使用sockaddr_in建立你所需要的信息, 然后用进行类型转换就可以了

参数addrlen

正如大多数socket接口一样,内核不关心地址结构,当它复制或传递地址给驱动的时候,它依据这个值来确定需要复制多少数据。这已经成为socket接口中最常见的参数之一了。

返回值:成功返回0,否则失败

例子:

struct sockaddr_in adr_server;

memset(&adr_server, 0x00, sizeof(struct sockaddr_in));

adr_server.sin_family = AF_INET; //通讯协议

adr_server.sin_port = htons(8891); //端口

adr_server.sin_addr.s_addr = htonl(INADDR_ANY); //本机IP

int z = bind(socket_fd, (struct sockaddr *)&adr_server, sizeof(adr_server));

if(z != 0) printf("fun bind: error...\n");

5.3listen函数

int listen(int sockfd, int backlog);

参数socketfd:需要监听的套接字值

参数backlog:值为需要维护的最大连接缓冲个数。一般为5个

返回值:成功返回0,否则失败

5.4accept函数

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);

参数sockfd:服务端套接字

参数addr:

客户端地址存储空间,一般采用struct sockaddr_in 类型,然后强转成参数类型。参数sockfd:存放结构体addr的长度,可以定义变量int client_len

返回值:客户端套接字

注意:accpt用于阻塞模式Socket服务,当有连接上上来时返回客户端套接字,若无连接时,等待连接。参数中如果addr与addrlen中有一个为零NULL,将不返回所接受的套接口远程地址的任何信息。

例子:int socket_client =accept(socket_fd,(struct sockaddr *)&adr_client,&client_len);

5.5read/write函数

read和write函数是对文件的操作。跟文件操作一样处理。

例如:

int rSize = read(client_fd, ch, 512);

int wSize = write(client_fd,ch1,sizeof(ch1));

5.6、setsockopt函数

6.select模型

select模型是非阻塞模式。

#include

int select(int maxfd, fd_set *readset, fd_set * writeset, fd_set *exceptset, const struct timeval *timeout);

参数maxfd:最大描述符+1

参数readset:回传描述符读

参数writeset:回传描述符写

参数exceptset:回传描述符异常

参数timeout:超时设置

struct timeval

{

time_t tv_sec;

time_t tv_usec;

};

有三种情况,

a、永远等待,超时参数设置为NULL

b、等待一定时间,时间自己设定

c、不等待,超时设置为0。

返回值:错误-1、超时0、执行成功返回变动描述符个数

对描述符的处理(宏):

FD_CLR(inr fd,fd_set* set);//用来清除描述词组set中相关fd 的位

FD_ISSET(int fd,fd_set *set); //用来测试描述词组set中相关fd 的位是否为真

FD_SET(int fd,fd_set*set); //用来设置描述词组set中相关fd的位

FD_ZERO(fd_set *set); //用来清除描述词组set的全部位

备注:select 函数主要用于非阻塞式文件描述符监听,可以统一管理多个事件的情况。例子:

while(1)

{

FD_ZERO(&readfd);//初始化set

FD_SET(0, &readfd); //每次轮询都需要添加到readFd中

https://www.360docs.net/doc/a14041749.html,_sec = 5;

https://www.360docs.net/doc/a14041749.html,_usec = 0; //设置等待时间,必须在每次循环中设置

rlt = select(0+1, &readfd, NULL, NULL, &outTime);

if (rlt < 0)

{

printf("error...\n");

break;

}

else if (rlt == 0)

{

printf("out off time...\n");

continue;

}else if (rlt > 0)

{

printf("发送变化的句柄个数: %d\n", rlt);

if (FD_ISSET(0, &readfd))

{

read(0, recvbuf, 50);

write(1, recvbuf, strlen(recvbuf)); //直接写入到标准输出屏幕上}

}

}

7.epoll模型

epoll用到的所有函数都是在头文件sys/epoll.h中声明的,下面简要说明所用到的数据结构和函数:

所用到的数据结构

typedef union epoll_data {

void *ptr;

int fd; //发生事件的文件描述符

__uint32_t u32;

__uint64_t u64;

} epoll_data_t;

struct epoll_event {

__uint32_t events; /* Epoll events */

epoll_data_t data; /* User data variable */

};

结构体epoll_event 被用于注册所感兴趣的事件和回传所发生待处理的事件,其中epoll_data 联合体用来保存触发事件的某个文件描述符相关的数据,例如一个client连接到服务器,服务器通过调用accept函数可以得到于这个client对应的socket文件描述符,可以把这文件描述符赋给epoll_data的fd字段以便后面的读写操作在这个文件描述符上进行。epoll_event 结构体的events字段是表示感兴趣的事件和被触发的事件可能的取值为:EPOLLIN :表示对应的文件描述符可以读;

EPOLLOUT:表示对应的文件描述符可以写;

EPOLLPRI:表示对应的文件描述符有紧急的数据可读(我不太明白是什么意思,可能是类似client关闭socket连接这样的事件);

EPOLLERR:表示对应的文件描述符发生错误;

EPOLLHUP:表示对应的文件描述符被挂断;

EPOLLET:表示对应的文件描述符有事件发生;

所用到的函数:

1、epoll_create函数

函数声明:int epoll_create(int size)

该函数生成一个epoll专用的文件描述符。它其实是在内核申请一空间,用来存放你想关注的socket fd上是否发生以及发生了什么事件。size就是你在这个epoll fd上能关注的最大socket fd数。随你定好了。只要你有空间。

2、epoll_ctl函数

函数声明:int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event)

该函数用于控制某个文件描述符上的事件,可以注册事件,修改事件,删除事件。

参数:epfd:由epoll_create 生成的epoll专用的文件描述符;

op:要进行的操作例如注册事件,可能的取值EPOLL_CTL_ADD 注册、EPOLL_CTL_MOD 修改、EPOLL_CTL_DEL 删除

fd:关联的文件描述符;

event:指向epoll_event的指针;

如果调用成功返回0,不成功返回-1

struct epoll_event ev;

//设置与要处理的事件相关的文件描述符

ev.data.fd=listenfd;

//设置要处理的事件类型

ev.events=EPOLLIN|EPOLLET;

//注册epoll事件

epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,&ev);

常用的事件类型:

EPOLLIN :表示对应的文件描述符可以读;

EPOLLOUT:表示对应的文件描述符可以写;

EPOLLPRI:表示对应的文件描述符有紧急的数据可读

EPOLLERR:表示对应的文件描述符发生错误;

EPOLLHUP:表示对应的文件描述符被挂断;

EPOLLET:表示对应的文件描述符有事件发生;

3、epoll_wait函数

函数声明:int epoll_wait(int epfd,struct epoll_event * events,int maxevents,int timeout)

该函数用于轮询I/O事件的发生;

参数:

epfd:由epoll_create 生成的epoll专用的文件描述符;

epoll_event:用于回传代处理事件的数组;

maxevents:每次能处理的事件数;

timeout:等待I/O事件发生的超时值(毫秒);-1相当于阻塞,0相当于非阻塞。一般用-1即可返回发生事件数

唯一有点麻烦是epoll有2种工作方式:LT和ET。

LT(level triggered)是缺省的工作方式,并且同时支持block和no-block socket.在这种做法中,内核告诉你一个文件描述符是否就绪了,然后你可以对这个就绪的fd进行IO操作。如果你不作任何操作,内核还是会继续通知你的,所以,这种模式编程出错误可能性要小一点。传统的select/poll都是这种模型的代表.

ET (edge-triggered)是高速工作方式,只支持no-block socket。在这种模式下,当描述符从未就绪变为就绪时,内核通过epoll告诉你。然后它会假设你知道文件描述符已经就绪,并且不会再为那个文件描述符发送更多的就绪通知,直到你做了某些操作导致那个文件描述符不再为就绪状态了(比如,你在发送,接收或者接收请求,或者发送接收的数据少于一定量时导致了一个EWOULDBLOCK 错误)。但是请注意,如果一直不对这个fd作IO操作(从而导致它再次变成未就绪),内核不会发送更多的通知(only once),不过在TCP协议中,ET模式的加速效用仍需要更多的benchmark确认。

4. close(epollFd);

8.进程间通讯

进程间通讯可以采用System v IPC 、POSIX IPC和BSD Socket方式,其中BSD Socket是基于socket方式。socket不仅支持进程间通讯,也支持网络通信。

System V IPC 接口是AT&T的贝尔实验室发展起来的,其通讯起到包括:管道、FIFO、消息队列、信号灯、共享内存。

POSIX标准非常广泛,POSIX IPC标准用于进程间通信。接口包括:消息队列、信号灯、共享内存。

先对System V进程间通讯的资料做一下介绍。

System v进程间通讯的参数信息,请查看man svipc 7 帮助文档。下面对重要信息的概述:

简写:svipc 意思: System V interprocess communication mechanisms

相关的参考头文件为:

# include

# include

# include

# include

# include

在进程通讯中创建各种机制,比如信号灯、管道、FIFO、内存共享或消息时,要求设置读写权限,即mode参数。此参数与open函数中的mode相识。可以采用下面方式标识它:0400 Read by user.

0200 Write by user.

0040 Read by group.

0020 Write by group.

0004 Read by others.

0002 Write by others.

一般采取0666的方式。

除了权限参数外,需要或上相应的操作类型,如下:

IPC_CREAT Create entry if key doesn’t exist.

IPC_EXCL Fail if key exists.

IPC_NOWAIT Error if request must wait.

IPC_PRIVATE Private key.

IPC_RMID Remove resource.

IPC_SET Set resource options.

IPC_STAT Get resource options.

比如:

消息队列的创建int msgId = msgget((key_t)1234, 0666|IPC_CREAT);

System V方式进程间通信的函数都采用下列格式:

创建通讯方式方式:xxxget();

设置或删除通讯方式:xxxctl();

接下介绍POSIX IPC的通讯方式,POSIX(Portalble Operating System Interface)已经得到

Unix/Linux系统的广泛支持。

接口格式为:

int xxx_open();

int xxx_close();

int xxx_unlink();

8.1 内存共享

内存共享有System V和POSIX两种方式。内存共享仅仅是设置了数据的共享,数据的一致性需要通过信号灯或者其他方式实现同步。下面简单介绍System V的方式。

#include

#include

int shmget(key_t key, size_t size, int shmflg);

功能:创建内存共享。

参数key:整型类型相似,用于标识共享内存名字。比如(key_t)1234

参数size:设置共享内存区域的大小,必须是系统页的整数倍。通过int getpagesize(void)函数获取系统页大小。linux下一般是4K。

参数shmflg:通过特定常量的按位或操作来。

IPC_CREAT:这个标志表示应创建一个新的共享内存块。通过指定这个标志,我们可以创建一个具有指定键值的新共享内存块。

IPC_EXCL:这个标志只能与IPC_CREAT 同时使用。当指定这个标志的时候,如果已有一个具有这个键值的共享内存块存在,则shmget会调用失败。也就是说,这个标志将使线程获得一个“独有”的共享内存块。如果没有指定这个标志而系统中存在一个具有相同键值的共享内存块,shmget会返回这个已经建立的共享内存块,而不是重新创建一个。

标志:这个值由9个位组成,分别表示属主、属组和其它用户对该内存块的访问权限。其中表示执行权限的位将被忽略。指明访问权限的一个简单办法是利用中指定,并且在手册页第二节stat条目中说明了的常量指定。例如,S_IRUSR和S_IWUSR分别指定了该内存块属主的读写权限,而S_IROTH和S_IWOTH则指定了其它用户的读写权限。下面例子中shmget函数创建了一个新的共享内存块(当shm_key已被占用时则获取对一个已经存在共享内存块的访问),且只有属主对该内存块具有读写权限,其它用户不可读写。

int segment_id = shmget (shm_key, getpagesize (), IPC_CREAT | S_IRUSR| S_IWUSR ); 如果调用成功,shmget将返回一个共享内存标识符。如果该共享内存块已经存在,系统会检查访问权限,同时会检查该内存块是否被标记为等待摧毁状态

void *shmat(int shmid, const void *shmaddr, int shmflg);

功能:函数把shmid虚拟内存空间的地址映射到进程自有内存地址shmaddr中。

参数shmid:shmget()创建的共享内存空间。

参数shmaddr:等于NULL时,系统会申请合适的地址(未使用)与shmid关联;不等于NULL 时,….

shmflg标志:

返回值:成功时返回共享内存的起始地址,失败时返回-1。注意失败比较为(void*)-1

int shmdt(const void *shmaddr);

功能:取消地址shmaddr上面的共享内存地址映射。成功则修改共享内存的结构体shmid_ds 信息参数,shm_dtime 设置为但前时间。shm_nattch数量减去1个。

返回值:成功shmdt() 返回0; 失败返回-1。

#include #include

int shmctl(int shmid, int cmd, struct shmid_ds *buf);

缩写:shmctl: shared memory control

功能:通过shmctl()函数设置共享内存shmid的信息选项。

参数buf:指向一个shmid_ds 结构体。

struct shmid_ds {

struct ipc_perm shm_perm; /* Ownership and permissions */

size_t shm_segsz; /* Size of segment (bytes) */

time_t shm_atime; /* Last attach time */

time_t shm_dtime; /* Last detach time */

time_t shm_ctime; /* Last change time */

pid_t shm_cpid; /* PID of creator */

pid_t shm_lpid; /* PID of last shmat()/shmdt() */

shmatt_t shm_nattch; /* No. of current attaches */

...

};

ipc_perm 结构体定义在单元:as follows (the highlighted fields are settable using IPC_SET):

struct ipc_perm {

key_t key; /* Key supplied to shmget() */

uid_t uid; /* Effective UID of owner */

gid_t gid; /* Effective GID of owner */

uid_t cuid; /* Effective UID of creator */

gid_t cgid; /* Effective GID of creator */

unsigned short mode; /* Permissions + SHM_DEST and

SHM_LOCKED flags */

unsigned short seq; /* Sequence number */

};

IPC_STAT:拷贝内存中的共享内存信息到参数buf中。

IPC_SET:设置buf参数中结构体ipc_perm的相关信息,如权限等。

IPC_RMID:标识共享内存将被释放。真正释放的动作发生在最后一个进程调用了断开映射,即shm_nattch参数为0时。设置释放参数的进程必须是共享内存创建进程或者本身是私有进程。

请在设置时,确保内存被释放了,否则共享内存将一直存在。

查看共享内存使用情况:

[root@localhost linux_high]# ipcs -m

------ Shared Memory Segments --------

key shmid owner perms bytes nattch status

0x00000000 65536 root 600 393216 2 dest

0x00000000 98305 root 600 393216 2 dest

0x00000000 131074 root 600 393216 2 dest

0x00000000 163843 root 600 393216 2 dest

0x00000000 196612 root 600 393216 2 dest

0x00000000 229381 root 600 393216 2 dest

0x00000000 262150 root 600 393216 2 dest

0x00000000 294919 root 600 393216 2 dest

0x00000000 327688 root 600 393216 2 dest

0x00000000 360457 root 600 393216 2 dest

8.2 消息队列

消息队列有System V和POSIX两种方式。

下面先介绍System V调用方式。

#include

#include

#include

int msgget(key_t key, int msgflg);

功能:创建消息队列

参数key:队列名字

参数msgflg:参考内存共享shmget()函数

返回值:成功返回队列标识(非负数), 否则-1。

8.3 信号灯

信号灯可以采用System v和POSIX两种方式实现。下面介绍POSIX信号灯用于进程或线程间的信号同步。信号灯是一个整数,它的值不能低于0。又称之为计数器,用于表示可用资源的数量。对信号灯的操作包括获取(sem_wait)和释放(sem_post)。获取信号灯时若资源为0,进程将阻塞知道它直到大于0。

POSIX信号灯有两种形式:1.命名信号灯。2.无名信号灯。

命名信号灯:命名信号灯通过名字,如:”/somename”,进行标识。两个进程可以使用相同的这个标识名,通过sem_open(3)函数进行创建。创建完成后,进程间可以通过sem_wait()和sem_post()进行信号间的同步。当所有进程都使用信号灯完毕后,采用unlink(0删除信号灯。

无名信号灯(又称基于内存的信号灯):需要名字进行标识。信号被放在多进程或者多线程之间的共享内存区。线程间无名信号变量需要设置为全局的,进程间信号灯则被存储在内存共享区。无名信号灯使用前需要被初始化sem_init(),在释放内存共享区前使用sem_destroy销毁无名信号灯。

Linux下命名信号灯的创建在虚拟文件系统中,通常在/dev/shm下,名字为https://www.360docs.net/doc/a14041749.html,。

信号灯头文件及函数:

#include

typedef union //联合体

{

char __size[__SIZEOF_SEM_T];

long int __align;

} sem_t;

a.sem_open

sem_t *sem_open(const char *name, int oflag);

sem_t *sem_open(const char *name, int oflag, mode_t mode, unsigned int value);

上述函数为打开信号灯函数,用途如下:

sem_open()创建一个新的POSIX信号灯或者打开一个已经存在的信号灯。

参数name:信号灯名字

参数oflag:标识操作{O_CREAT:创建信号灯,这个时候需要设置mode和value参数}

参数mode:参考文件open函数的权限设置{S_IRWXU |S_IRUSR | S_IWUSR| S_IXUSR等..} 参数value:初始值

返回值:成功返回sem_t类型指针失败SEM_FAILED

b.int sem_unlink(const char *name);

删除信号灯。

参数name:需要删除的名字。

返回值:成功0 失败-1

c.sem_post(sem_t *sem);

释放信号灯.

d.sem_wait(sem_t *sem);

请求信号灯.

e.sem_getvalue(sem_t *sem, int *sval);

设置但前信号量为sval值。

f.int sem_init(sem_t *sem, int pshared, unsigned int value);

初始化无名信号灯。

参数sem:无名信号灯指针

参数pshared:确定在进程间还是线程间共用.{0:线程间非0: 进程间}

参数value:信号灯初始化值

g.int sem_destroy(sem_t *sem);

销毁信号灯指针。

参数sem:无名信号灯指针

例子:

/******************************************

单元: pro1.c

用途:

通过pro1.c和pro2.c之间的信号灯同步,

来演示进程间同时打印数据一个数组。

pro1.c先打印个数字,然后等待pro2.c打印一个后,

再打印数字。

*******************************************/

#include

#include

#include

#include

#include

#include

#include

int main()

{

int i=0;

sem_t *mySem = sem_open("/mySem", O_CREAT, S_IRUSR|S_IWUSR, 4);

if (mySem != SEM_FAILED)

{

for (i=0; i<10; i++)

{

sem_wait(mySem);

printf("I am %d, sleep 1 second, then notifing pro2.c had done.\n", i);

sleep(1);

}

}

sem_close(mySem);

return 0;

}

/******************************************

单元: pro2.c

用途:

通过pro1.c和pro2.c之间的信号灯同步,

来演示进程间同时打印数据一个数组。

pro1.c先打印个数字,然后等待pro2.c打印一个后,

再打印数字。

*******************************************/

#include

#include

#include

#include

#include

#include

#include

int main()

{

int i=0;

sem_t *mySem = sem_open("/mySem", O_CREAT, S_IRUSR|S_IWUSR, 0);

if (mySem != SEM_FAILED)

{

for (i=0; i<10; i++)

{

printf("I am %d, sleep 1 second, then notifing pro1.c had done.\n", i);

sem_post(mySem);

sleep(1);

}

}

sem_close(mySem);

return 0;

}

8.4 管道

这里的管道是指无名管道,使用很简单。但是注意只能用于父子进程。

#include

int pipe(int filedes[2]);

创建一对文件描述,filedes[0]是读文件描述,filedes[1]是写文件描述。

成功返回0,失败-1

8.5 FIFO

命名管道也称之为FIFO,它在形式上是一种特殊的文件,可以独立于任何进程而存在。分为阻塞式和非阻塞式两种,具体设置在打开文件式采用O_NONBLOCK标识,默认是阻塞式的。

#include

#include

int mkfifo(const char *pathname, mode_t mode);

参数pathname: 管道名

参数mode:权限设置

返回值:0成功-1失败。

注意: linux下支持管道机制,windows下不支持。因此/mnt/hgfs共享目录下无法创建管道。

例子:

/**********************************

管道文件的创建, 注意/mnt/hgfs/下面

不能创建管道文件, 即不允许在共享目

录下面创建管道文件,需要拷贝到其他

目录下。

**********************************/

#include

#include

#include

#define pError(str) do{ \

printf("error: %s\n", str);\

}while(1<0);

int main(int argc, char *argv[])

{

/* 从命令行输入需要创建的FIFO文件*/ if (argc <2)

{

pError("params nums <2");

exit(0);

}

/* 清除文件掩码*/

umask(0);

/* 创建FIFO文件*/

if (mkfifo(argv[1], 0777)==-1)

{

pError("mkfifo error");

exit(1);

}

printf("success to create fifo %s\n", argv[1]);

return 0;

}

/******************************* readFiFo.c

********************************/

#include

#include

#include

#include

#include

#include

#define pError(str) do{ \

printf("error: %s\n", str);\

}while(1<0);

int main()

{

int count = 0;

char buf[128] = {0};

int fd = -1;

fd = open("myfifo", O_RDONLY);

if (fd < 0)

{

pError("params nums <2");

exit(1);//异常退出

}

while(1)

{

printf("ready to read fifo....\n");

count = read(fd, buf, 128);

if (count == 0)

break;

printf("recv msg: %s", buf);

}

close(fd);

return 0;

}

/************************************** writeFiFo.c

**************************************/

#include

#include

#include

#include

#include

#include

#define pError(str) do{ \

printf("error: %s\n", str);\

}while(1<0);

int main()

{

int count = 0;

char buf[128] = {0};

int fd = -1;

fd = open("myfifo", O_WRONLY | O_NONBLOCK);

if (fd < 0)

{

pError("params nums <2");

exit(1);//异常退出

}

printf("open fifo success to write....\n");

while(1)

{

//read(STDIN_FILENO, buf, 128);

sprintf(buf, "i am %d", count);

if (write(fd, buf, 128) < 0)

break;

count++;

sleep(1);

}

close(fd);

return 0;

}

9.线程学习

创建线程

pthread_create是UNIX环境创建线程函数

头文件

#include

函数声明

int pthread_create(pthread_t*restrict tidp, const pthread_attr_t *restrict_attr,void*(*start_rtn)(void*),void *restrict arg);

返回值

若成功则返回0,否则返回出错编号

返回成功时,由tidp指向的内存单元被设置为新创建线程的线程ID。attr参数用于制定各种不同的线程属性。新创建的线程从start_rtn函数的地址开始运行,该函数只有一个万能指针参数arg,如果需要向start_rtn函数传递的参数不止一个,那么需要把这些参数放到一个结构中,然后把这个结构的地址作为arg的参数传入。

linux下用C开发多线程程序,Linux系统下的多线程遵循POSIX线程接口,称为pthread。

由restrict 修饰的指针是最初唯一对指针所指向的对象进行存取的方法,仅当第二个指针基于第一个时,才能对对象进行存取。对对象的存取都限定于基于由restrict 修饰的指针表达式中。由restrict 修饰的指针主要用于函数形参,或指向由malloc() 分配的内存空间。restrict 数据类型不改变程序的语义。编译器能通过作出restrict 修饰的指针是存取对象的唯一方法的假设,更好地优化某些类型的例程。

参数

第一个参数为指向线程标识符的指针。

第二个参数用来设置线程属性。

第三个参数是线程运行函数的起始地址。

最后一个参数是运行函数的参数。

另外,在编译时注意加上-lpthread参数,以调用静态链接库。因为pthread并非Linux 系统的默认库

线程属性设置

介绍了线程的创建和退出,以及相关函数的使用。其中pthread_create函数的第二个参数,是关于线程属性的设置,这也是今天所有讲述的。这些属性主要包括邦定属性、分离属性、堆栈地址、堆栈大小、优先级。其中系统默认的是非邦定、非分离、缺省1M的堆栈、与父进程同样级别的优先级。在pthread_create中,把第二个参数设置为NULL的话,将采用默认的属性配置。

(1)邦定属性。

在LINUX中,采用的是“一对一”的线程机制。也就是一个用户线程对应一个内核线程。邦定属性就是指一个用户线程固定地分配给一个内核线程,因为CPU时间片的调度是面向内核线程(轻量级进程)的,因此具有邦定属性的线程可以保证在需要的时候总有一个内核线程与之对应,而与之对应的非邦定属性就是指用户线程和内核线程的关系不是始终固定的,而是由系统来分配。

(2)分离属性。

分离属性是决定以一个什么样的方式来终止自己。在非分离情况下,当一个线程结束时,它多占用的线程没有得到释放,也就是没有真正的终止,需要通过pthread_join来释放资源。而在分离属性情况下,一个线程结束时会立即释放它所占有的系统资源。但这里有一点要注意的是,如果设置一个线程分离属性,而这个线程又运行得非常快的话,那么它很可能在pthread_create函数返回之前就终止了线程函数的运行,它终止以后就很有可能将线程号和系统资源移交给其他的线程使用,这时调用 pthread_create的线程就得到错误的线程号。

这些属性都是通过一些函数来完成的,通常先调用pthread_attr_init来初始化,之后来调用相应的属性设置函数。

1、pthread_attr_init

功能:对线程属性变量的初始化。

头文件:

函数原型: int pthread_attr_init (pthread_attr_t* attr);

函数返回值:成功: 0

失败: -1 errno

2、pthread_attr_setscope

功能:设置线程绑定属性。

头文件:

函数原型: int pthread_attr_setscope (pthread_attr_t* attr, int scope);

函数传入值:attr: 线程属性。

scope:PTHREAD_SCOPE_SYSTEM(绑定)

PTHREAD_SCOPE_PROCESS(非绑定)

函数返回值得:同1。

3、pthread_attr_setdetachstate

功能:设置线程分离属性。

头文件:

函数原型: int pthread_attr_setdetachstate (pthread_attr_t* attr, int detachstate);函数传入值:attr:线程属性。

detachstate:PTHREAD_CREATE_DETACHED(分离)

PTHREAD_CREATE_JOINABLE(非分离)

函数返回值得:同1。

4、pthread_attr_getschedparam

功能:得到线程优先级。

头文件:

函数原型: int pthread_attr_getschedparam (pthread_attr_t* attr, struct sched_param* param);

param:线程优先级;

函数返回值:同1。

5、pthread_attr_setschedparam

功能:设置线程优先级。

头文件:

函数原型: int pthread_attr_setschedparam (pthread_attr_t* attr, struct sched_param* param);

函数传入值:attr:线程属性。

param:线程优先级。

函数返回值:同1。

函数实现:

#include

#include

#include

#include

void *pthread_func_1 (void* data)

{

int i = 0;

for (; i < 6; i++)

{

printf ("This is pthread_1.\n");

if (i == 2)

{

pthread_exit (0);

}

}

return NULL;

}

void *pthread_func_2 (void *data)

{

嵌入式linux基本操作实验一的实验报告

实验一linux基本操作实验的实验报告 一实验目的 1、熟悉嵌入式开发平台部件,了解宿主机/目标机开发模式; 2、熟悉和掌握常用Linux的命令和工具。 二实验步骤 1、连接主机和目标板;(三根线,网线直接连接实验箱和PC机,实验箱UART2连接主机的UART口)。 2、Linux命令的熟悉与操作 PC端:在PC机的桌面上打开虚拟机,并启动Linux系统,打开命令终端,操作Linux基本命令,如:查看:ls,进入目录:cd,创建文件:mkdir,删除文件:rmdir,配置网络:ifconfig,挂载:mount,设置权限:chmod,编辑器:vi,拷贝:cp等命令,要求能熟练操作。 使用方法: 1.查看:ls Ls列出文件和目录 Ls–a 显示隐藏文件 Ls–l 显示长列格式ls–al 其中:蓝:目录;绿:可执行文件;红:压缩文件;浅蓝:链接文件;灰:其他文件;红底白字:错误的链接文件 2.进入目录:cd 改变当前目录:cd 目录名(进入用户home目录:cd ~;进入上一级目录:cd -) 3.创建文件:mkdir 建立文件/目录:touch 文件名/mkdir目录名 4.删除文件:rmdir 删除空目录:rmdir目录名 5.配置网络:ifconfig 网络- (以太网和WIFI无线) ifconfig eth0 显示一个以太网卡的配置 6.挂载:mount mount /dev/hda2 /mnt/hda2 挂载一个叫做hda2的盘- 确定目录'/ mnt/hda2' 已经存在 umount /dev/hda2 卸载一个叫做hda2的盘- 先从挂载点'/ mnt/hda2' 退出fuser -km /mnt/hda2 当设备繁忙时强制卸载 umount -n /mnt/hda2 运行卸载操作而不写入/etc/mtab文件- 当文件为只读或当磁盘写满时非常有用 mount /dev/fd0 /mnt/floppy 挂载一个软盘 mount /dev/cdrom /mnt/cdrom挂载一个cdrom或dvdrom mount /dev/hdc /mnt/cdrecorder挂载一个cdrw或dvdrom mount /dev/hdb /mnt/cdrecorder挂载一个cdrw或dvdrom mount -o loop file.iso /mnt/cdrom挂载一个文件或ISO镜像文件

什么是嵌入式linux系统

什么是嵌入式linux系统? 一、什么是嵌入式linux? Linux从1991年问世到现在,短短的十几年时间已经发展成为功能强大、设计完善的操作系统之一,不仅可以与各种传统的商业操作系统分庭抗争,在新兴的嵌入式操作系统领域内也获得了飞速发展。嵌入式Linux(Embedded Linux)是指对标准Linux经过小型化裁剪处理之后,能够固化在容量只有几K或者几M字节的存储器芯片或者单片机中,适合于特定嵌入式应用场合的专用Linux操作系统。嵌入式Linux既继承了intelnet上无限的开放原代码资源,又具有嵌入式操作系统的特性。 二、嵌入式Linux的特点版权费:免费; 购买费用:媒介成本; 技术支持:全世界的自由软件开发者提供支持; 网络特性:免费而且性能优异; 软件移植:容易,代码开放,有许多应用软件支持; 应用产品开发周期:短,新产品上市迅速,因为有许多公开的代码可以参考和移植; 实时性能:RT_Linux,hardhat Linux 等嵌入式Linux支持实时性能; 稳定性:好; 安全性:好。 三、嵌入式Linux的市场前景和商业机会 嵌入式Linux有巨大的市场前景和商业机会,出现了大量的专业公司和产品,如Montavista、Lineo、Emi等。有行业协会,如Embedded Linux Consortum等。得到世界著名计算机公司和oem板级厂商的支持,例如IBM、Motorola、Intel等。传统的嵌入式系统厂商也采用了Linux策略如Lynxworks 、Windriver、QNX等。还有intelnet上的大量嵌入式Linux 爱好者的支持。嵌入式Linux支持几乎所有的嵌入式cpu和被移植到几乎所有的嵌入式oem板。 四、嵌入式Linux的应用领域嵌入式Linux的应用领域非常广泛,主要的应用领域有,信息家电:PDA,STB-Set-stopbox,Digital Telephone,Answering Machine,Screen Phone、数据网络:Ethernet switches,Router,Bridge,Hub,Remote access servers,ATM,Frame relay、远程通信、医疗电子、交通运输、计算机外设、工业控制、航空领域等。 五、嵌入式linux的优势嵌入式Linux的开发和研究是操作系统领域中的一个热点,目前已经开发成功的嵌入式系统中,大约有一半使用的是Linux。Linux之所以能在嵌入式系统市场上取得如此辉煌的成果,与其

《嵌入式系统与开发》构建嵌入式Linux系统-实验报告

《嵌入式数据库sqlite移植及使用》 实验报告 学生姓名:陈彤 学号:13004405 专业班级:130044 指导教师:孙国梓 完成时间:2016.5.31 实验3 嵌入式数据库sqlite移植及使用 一.实验目的 理解嵌入式软件移植的基本方法,掌握sqlite数据库软件移植的步骤,掌握sqlite开发的两种方式—命令模式和C代码开发模式的使用方法,并编程实现简单通讯录查询实验。 二.实验内容 实验3.1 移植嵌入式数据库sqlite 实验3.2 简单通讯录查询实例设计和测试 三.预备知识 Linux使用、数据库相关知识等 四.实验设备及工具(包括软件调试工具) 硬件:ARM 嵌入式开发平台、PC 机Pentium100 以上、串口线。 软件:WinXP或UBUNTU开发环境。 五.实验步骤 5.1 移植嵌入式数据库sqlite 步骤【参看教材103页】: 第一步,解压缩sqlite源码,命令tar zxvf sqlite-autoconf-3080900.tar.gz,在解压后的文件夹下,可以看到源码文件有shell.c 和sqlite3.c文件,生成Makefile的配置脚本文件configure.ac ,并检查当前文件夹下__A__(A.存在 B.不存在)Makefile文件。 第二步利用configure脚本文件生成基于ARM实验台的Makefile,具体命令为./configure CC=arm-linux-gcc –prefix=/opt/sqlite –host=arm-linux(假设安装目录为/opt/sqlite),并检查当前文件夹下___A__(A.存在 B.不存在)Makefile文件。 第三步,编译sqlite,命令为_make_,编译过程中使用的编译器为_ arm-linux-gcc _。 第四步,安装sqlit,命令为_make install_。安装完成后到_/opt/sqlite_文件夹下去查看相关文件,可以看到该文件夹下有_bin_、_include_、__lib__和share文件夹,其中可执行文件sqlite3位于_./bin_文件夹,库位于_./lib_文件夹。 第五步,将sqlite3拷贝到开发板bin目录下,将库下的文件拷贝到开发板的lib目录下【注意链接文件的创建】 第六步,数据库的使用 方式1:命令操纵数据库 在超级终端环境下创建数据库stucomm.db,命令为_sqlite3 stucomm.db_; 创建数据表stutable,字段包括id 整型,name 字符型,phoneNum 字符型,具体命令为_sqlite> create table stutable (id int(20),name char(20),phoneNum char(20));_; 插入2条记录,记录信息如下 001,zhangsan,10086 002,lisi,10000

嵌入式Linux之我行 史上最牛最详细的uboot移植,不看别后悔

嵌入式Linux之我行——u-boot-2009.08在2440上的移植详解(一) 嵌入式Linux之我行,主要讲述和总结了本人在学习嵌入式linux中的每个步骤。一为总结经验,二希望能给想入门嵌入式Linux 的朋友提供方便。如有错误之处,谢请指正。 ?共享资源,欢迎转载:https://www.360docs.net/doc/a14041749.html, 一、移植环境 ?主机:VMWare--Fedora 9 ?开发板:Mini2440--64MB Nand,Kernel:2.6.30.4 ?编译器:arm-linux-gcc-4.3.2.tgz ?u-boot:u-boot-2009.08.tar.bz2 二、移植步骤 本次移植的功能特点包括: ?支持Nand Flash读写 ?支持从Nor/Nand Flash启动 ?支持CS8900或者DM9000网卡 ?支持Yaffs文件系统 ?支持USB下载(还未实现) 1.了解u-boot主要的目录结构和启动流程,如下图。

u-boot的stage1代码通常放在cpu/xxxx/start.S文件中,他用汇编语言写成;u-boot的stage2代码通常放在lib_xxxx/board.c文件中,他用C语言写成。各个部分的流程图如下:

2. 建立自己的开发板项目并测试编译。 目前u-boot对很多CPU直接支持,可以查看board目录的一些子目录,如:board/samsung/目录下就是对三星一些ARM 处理器的支持,有smdk2400、smdk2410和smdk6400,但没有2440,所以我们就在这里建立自己的开发板项目。 1)因2440和2410的资源差不多,主频和外设有点差别,所以我们就在board/samsung/下建立自己开发板的项目,取名叫my2440 2)因2440和2410的资源差不多,所以就以2410项目的代码作为模板,以后再修改

南邮嵌入式系统B实验报告2016年度-2017年度-2

_* 南京邮电大学通信学院 实验报告 实验名称:基于ADS开发环境的程序设计 嵌入式Linux交叉开发环境的建立 嵌入式Linux环境下的程序设计 多线程程序设计 课程名称嵌入式系统B 班级学号 姓名 开课学期2016/2017学年第2学期

实验一基于ADS开发环境的程序设计 一、实验目的 1、学习ADS开发环境的使用; 2、学习和掌握ADS环境下的汇编语言及C语言程序设计; 3、学习和掌握汇编语言及C语言的混合编程方法。 二、实验内容 1、编写和调试汇编语言程序; 2、编写和调试C语言程序; 3、编写和调试汇编语言及C语言的混合程序; 三、实验过程与结果 1、寄存器R0和R1中有两个正整数,求这两个数的最大公约数,结果保存在R3中。 代码1:使用C内嵌汇编 #include int find_gcd(int x,int y) { int gcdnum; __asm { MOV r0, x MOV r1, y LOOP: CMP r0, r1 SUBLT r1, r1, r0 SUBGT r0, r0, r1 BNE LOOP MOV r3, r0 MOV gcdnum,r3 //stop // B stop // END } return gcdnum; } int main() { int a; a = find_gcd(18,9);

printf("gcdnum:%d\n",a); return 0; } 代码2:使用纯汇编语言 AREA example1,CODE,readonly ENTRY MOV r0, #4 MOV r1, #9 start CMP r0, r1 SUBLT r1, r1, r0 SUBGT r0, r0, r1 BNE start MOV r3, r0 stop B stop END 2、寄存器R0 、R1和R2中有三个正整数,求出其中最大的数,并将其保存在R3中。 代码1:使用纯汇编语言 AREA examp,CODE,READONL Y ENTRY MOV R0,#10 MOV R1,#30 MOV R2,#20 Start CMP R0,R1 BLE lbl_a CMP R0,R2 MOVGT R3,R0 MOVLE R3,R2 B lbl_b lbl_a CMP R1,R2 MOVGT R3,R1 MOVLE R3,R2 lbl_b B . END 代码2:使用C内嵌汇编语言 #include int find_maxnum(int a,int b,int c)

嵌入式Linux系统开发教程很完整的习题答案资料

参考答案 第一章 一、填空题。 1、嵌入式系统主要融合了计算机软硬件技术、通信技术和微电子技术,它是将计算机直接嵌入到应用系统中,利用计算机的高速处理能力以实现某些特定的功能。 2、目前国内对嵌入式系统普遍认同的定义是:以应用为中心、以计算机技术为基础、软硬件可裁剪、适应应用系统对功能、可靠性、成本、体积、功耗严格要求的专用计算机系统。 3、嵌入式系统一般由嵌入式计算机和执行部件组成,其中嵌入式计算机主要由四个部分组成,它们分别是:硬件层、中间层、系统软件层以及应用软件层。 4、嵌入式处理器目前主要有ARM、MIPS、Power PC、68K等,其中arm处理器有三大特点:体积小、低功耗、的成本和高性能,16/32位双指令集,全球合作伙伴众多。 5、常见的嵌入式操作系统有:Linux、Vxworks、WinCE、Palm、uc/OS-II和eCOS。 6、嵌入式系统开发的一般流程主要包括系统需求分析、体系结构设计、软硬件及机械系统设计、系统集成、系统测试,最后得到最终产品。 二、选择题 1、嵌入式系统中硬件层主要包含了嵌入式系统重要的硬件设备:、存储器(SDRAM、ROM等)、设备I/O接口等。(A) A、嵌入式处理器 B、嵌入式控制器 C、单片机 D、集成芯片 2、20世纪90年代以后,随着系统应用对实时性要求的提高,系统软件规模不断上升,实时核逐渐发展为,并作为一种软件平台逐步成为目前国际嵌入式系统的主流。(D) A、分时多任务操作系统 B、多任务操作系统 C、实时操作系统 D、实时多任务操作系统 3、由于其高可靠性,在美国的火星表面登陆的火星探测器上也使用的嵌入式操作系统是。(B) A、Palm B、VxWorks C、Linux D、WinCE [在此处键入]

嵌入式程序设计实验报告

实验一开发环境的搭建与配置 【实验目的】 1)熟悉嵌入式Linux开发平台。 2)掌握嵌入式Linux开发平台的开发环境搭建与配置。 3)了解minicom配置串口通信参数的过程。 4)了解嵌入式Linux的启动过程。 5)掌握程序交叉编译运行及调试的一般方法。 【实验内容】 1)连接实验开发板与宿主机。 2)在虚拟机中的CentOS(宿主机)搭建开发环境。 3)在宿主机中配置minicom。 4)分析嵌入式Linux的启动过程。 5)在宿主机上编写简单的C语言程序并用交叉编译工具进行编译,然后传输到目标机上运行。 6)在宿主机上编写简单的C语言程序并用交叉编译工具进行编译,用gdbserver进行远程调试。 【实验步骤】 连接实验开发板,对虚拟机进行设置 1)首先把实验开发板打开,用网线和串口线连接宿主机,并连接电源(注意这时不要拨动实验 开发板的开关按钮)。 2)在桌面上点击打开vmware 软件,选择“编辑虚拟机设置”,如下图所示:

图1 3)进入虚拟机配置界面后把网络连接方式设置为“桥接方式”,如图2所示: 图2

4)添加串口,如下图所示: 图3 5)完成串口的添加后,选择“OK”,完成对虚拟机的设置。如下图所示:

图4 6)选择虚拟机的“Edit”、“Virtual Network Editor...”,如下图所示:

图5 7)进入虚拟机网络参数设置界面后对VMnet0进行设置(注意这里桥接的网卡应选择与实验开 发板相连接的那块儿网卡),然后点击“Apply”、“OK”如下图所示:

图6 8)上述设置完成后启动CentOS(CentOS的用户名为“root”,密码为“xidianembed”)。 工具链的配置 1)在CentOS的根目录下创建一个名为“EELiod”的目录,把实验中要用到的文件(主要是一 些rpm包)拷贝到该目录下。(可以用U盘、WinSCP等工具进行,此处不再做详细说明)。 2)交叉编译工具链位于/opt/buildroot-2011.02/output/host/usr目录下,进入工具链的bin目录下, 可以看到一些编译工具,这些工具将会在之后的交叉编译过程中使用到。

嵌入式Linux应用软件开发流程

从软件工程的角度来说,嵌入式应用软件也有一定的生命周期,如要进行需求分析、系统设计、代码编写、调试和维护等工作,软件工程的许多理论对它也是适用的。 但和其他通用软件相比,它的开发有许多独特之处: ·在需求分析时,必须考虑硬件性能的影响,具体功能必须考虑由何种硬件实现。 ·在系统设计阶段,重点考虑的是任务的划分及其接口,而不是模块的划分。模块划分则放在了任务的设计阶段。 ·在调试时采用交叉调试方式。 ·软件调试完毕固化到嵌入式系统中后,它的后期维护工作较少。 下面主要介绍分析和设计阶段的步骤与原则: 1、需求分析 对需求加以分析产生需求说明,需求说明过程给出系统功能需求,它包括:·系统所有实现的功能 ·系统的输入、输出 ·系统的外部接口需求(如用户界面) ·它的性能以及诸如文件/数据库安全等其他要求 在实时系统中,常用状态变迁图来描述系统。在设计状态图时,应对系统运行过程进行详细考虑,尽量在状态图中列出所有系统状态,包括许多用户无需知道的内部状态,对许多异常也应有相应处理。 此外,应清楚地说明人机接口,即操作员与系统间地相互作用。对于比较复杂地系统,形成一本操作手册是必要的,为用户提供使用该系统的操作步骤。为使系统说明更清楚,可以将状态变迁图与操作手册脚本结合起来。

在对需求进行分析,了解系统所要实现的功能的基础上,系统开发选用何种硬件、软件平台就可以确定了。 对于硬件平台,要考虑的是微处理器的处理速度、内存空间的大小、外部扩展设备是否满足功能要求等。如微处理器对外部事件的响应速度是否满足系统的实时性要求,它的稳定性如何,内存空间是否满足操作系统及应用软件的运行要求,对于要求网络功能的系统,是否扩展有以太网接口等。 对于软件平台而言,操作系统是否支持实时性及支持的程度、对多任务的管理能力是否支持前面选中的微处理器、网络功能是否满足系统要求以及开发环境是否完善等都是必须考虑的。 当然,不管选用何种软硬件平台,成本因素都是要考虑的,嵌入式Linux 正是在这方面具有突出的优势。 2、任务和模块划分 在进行需求分析和明确系统功能后,就可以对系统进行任务划分。任务是代码运行的一个映象,是无限循环的一段代码。从系统的角度来看,任务是嵌入式系统中竞争系统资源的最小运行单元,任务可以使用或等待CPU、I/O设备和内存空间等系统资源。 在设计一个较为复杂的多任务应用系统时,进行合理的任务划分对系统的运行效率、实时性和吞吐量影响都极大。任务分解过细会不断地在各任务之间切换,而任务之间的通信量也会很大,这样将会大大地增加系统的开销,影响系统的效率。而任务分解过粗、不够彻底又会造成原本可以并行的操作只能按顺序串行执行,从而影响系统的吞吐量。为了达到系统效率和吞吐量之间的平衡折中,在划分任务时应在数据流图的基础上,遵循下列步骤和原则:

嵌入式系统实验报告

郑州航空工业管理学院 嵌入式系统实验报告 (修订版) 20 – 20第学期 赵成,张克新 院系: 姓名: 专业: 学号: 电子通信工程系 2014年3月制

实验一ARM体系结构与编程方法 一、实验目的 了解ARM9 S3C2410A嵌入式微处理器芯片的体系结构,熟悉ARM微处理器的工作模式、指令状态、寄存器组及异常中断的概念,掌握ARM指令系统,能在ADS1.2 IDE中进行ARM汇编语言程序设计。 二、实验内容 1.ADS1.2 IDE的安装、环境配置及工程项目的建立; 2.ARM汇编语言程序设计(参考附录A): (1)两个寄存器值相加; (2)LDR、STR指令操作; (3)使用多寄存器传送指令进行数据复制; (4)使用查表法实现程序跳转; (5)使用BX指令切换处理器状态; (6)微处理器工作模式切换; 三、预备知识 了解ARM嵌入式微处理器芯片的体系结构及指令体系;熟悉汇编语言及可编程微处理器的程序设计方法。 四、实验设备 1. 硬件环境配置 计算机:Intel(R) Pentium(R) 及以上; 内存:1GB及以上; 实验设备:UP-NETARM2410-S嵌入式开发平台,J-Link V8仿真器; 2. 软件环境配置 操作系统:Microsoft Windows XP Professional Service Pack 2; 集成开发环境:ARM Developer Suite (ADS) 1.2。 五、实验分析 1.安装的ADS1.2 IDE中包括和两个软件组件。在ADS1.2中建立类型的工程,工程目标配置为;接着,还需要对工程进行、及链接器设置;最后,配置仿真环境为仿真方式。 2.写出ARM汇编语言的最简程序结构,然后在代码段中实现两个寄存器值的加法运算,给出运算部分相应指令的注释。 ; 文件名:

linux嵌入式实验2

嵌入式实验报告(二) 姓名:董辰辰 学号:111180031 专业:电子信息与科学 一、 实验目的 1、 了解Linux 内核源代码的目录结构以及各个目录的相关内容。 2、 了解Linux 内核各配置选项内容和作用。 3、 掌握Linux 内核的编译过程。 4、 了解嵌入式操作系统中文件系统的类型和作用。 5、 了解JFFS2文件系统的优点以及在嵌入式系统中的作用。 6、 掌握利用busybox 软件制作嵌入式文件系统的方法。 7、 掌握嵌入式Linux 文件系统的挂载过程。 二、实验内容和要求 1、配置完整的内核,尽可能理解配置选项在操作系统中的作用。 2、以 Busybox 为基础,构建一个合适的文件系统。 3、制作ramdiak 文件系统映像。 4、将自己编译生成的内核和文件系统下载进开发板。 5、讨论自己的嵌入式系统所具备的的功能。 6、比较romfs 、ext2fs/ext3fs 、JFFS2等文件系统的优缺点。 三、实验设计和分析 实验分为两个部分:Linux 内核配置编译和文件系统构建。本次实验的目的 就是自己搞一个内核和文件系统。当然自己编写代码是不太可能啦,而是根据 linux 提供的源代码自己配置编译出一个自己的内核,在构建一个自己需要的文 件系统。实验室用的内核版本是2.6.35.7 实验室的开发板有了内核和文件系统 才能够对其进行开发。内核是操作系统最基本的部分,可以说是一个软件,实验 室的开发板有处理器、内存、Flash 闪存等硬件组成,可以说内核是调度它们的 软件,有了内核各种应用程序才能够调用硬件资源,总的来说内核文件是操作系 统的核心,负责系统的进程管理,内存管理,设备和文件管理等,决定着系统的 性能和稳定性。文件系统是操作系统的一个重要组成部分,通过对操作系统所管 理的存储空间的抽象,向用户提供统一的、对象化的访问接口,屏蔽对物理设备 的直接操作和资源管理。我觉得内核的本质其实是程序,而文件系统是设置一些 规则来用来管理存储的,在一个完整的操作系统中两者都是不可或缺的,内核在 加载的时候需要有很多的挂载指令,应该挂载的就是文件系统的文件夹,比如将 proc 文件系统挂载到proc 文件夹,这就是为什么在开发板启动时内核和文件系 本页已使用福昕阅读器进行编辑。福昕软件(C)2005-2009,版权所有,仅供试用。

淮阴工学院嵌入式系统开发与应用实验报告实验四嵌入式Linux开发环境的搭建

实验四嵌入式Linux开发环境的搭建 一、实验目的 1、了解嵌入式Linux开发环境的作用 2、掌握相关服务器的安装 二、实验准备 硬件:JXARM9-2410教学实验箱,PC机 软件:Windows XP操作系统,ADS集成开发工具 三、实验过程 1、tftp网络配置 (1) 虚拟机网络配置 a. 点击虚拟机-设置,将网络适配器设置为自定义,并 指定虚拟网络为VMnet0。 b. 点击编辑-虚拟机网络参数-主机虚拟网络映射,并 且指定其桥接的网卡

c. Red Hat IP配置 ●点击系统设置-网络,双击eth0配置IP信息如下: 地址:172.20.11.243 子网掩码:255.255.255.0 默认网关地址:172.20.11.254 DNS: 210.29.152.4 ●点击激活,在弹出的对话框点击是按钮 ●

测试网络连接是否正常 方法一:在linux下,点击系统工具-终端,输入命令ping 172.20.11.243 方法二:在主系统中打开命令提示符,输入ping 172.20.11.243 (2)修改tftp的配置文件 a.在终端下输入gedit /etc/xinetd.d/tftp b.修改”disable=yes”为”=no”,点击保存,关闭gedit

c.重启xinetd服务,使刚才的更改生效,在终端里输入, /etc/init.d/xinetd restart d.进入tftpboot文件夹创建一个测试文件testfile e.取得tftpboot文件夹的所有权限,命令chmod 777 /tftpboot f.启动tftp测试上传和下载 2配置NFS服务器 a.设置:开始->系统设置->服务器设置->NFS服务器打开配置对话 框 第一步:点击添加nfs共享 第二步:输入目录/tftpboot 第三步:输入主机172.20.11.243 第四步:基本权限为读/写

嵌入式LINUX开发工具选择

嵌入式Linux具有稳定、可伸缩及开放源代码等特点,可兼容多 种处理器和主机,广泛适用于各种产品和应用。但是,交叉编译、 设备驱动程序开发/调试,以及更小尺寸等要求对嵌入式Linux开 发者来说都是严峻的挑战。为应对这些挑战,针对嵌入式Linux开 发的专用工具应运而生,而且发展十分迅猛。 但是,许多这类开发工具都不兼容非X86平台,而且也没有很好 地实现归档备案或集成。在其它开发环境下,组件间的高度集成并 没有完全兑现。因此,要想完全从这些免费的软件组件开始创建 一个完整的跨平台开发环境,开发者应意识到这将需要大量的调 研、实施、培训和维护方面的工作。 Linux 是少数既可以在嵌入式设备上运行也可作为开发环境的操 作系统之一。这一特性可让开发者在转向此开发系统之前于常用硬 件(比如X86桌面系统)之上开发、调试和测试应用程序和库,因 此可减少对标准参考平台和指令集仿真器的依赖。这一技术仅适用于应用程序和库,但不适用于设备驱动程序,因为后者的开发依赖于 Linux架构。 开放源代码团体及一些软件供应商可提供设备驱动程序开发工具。由于设备驱动程序比标准应用程序距离硬件更近,因此它们的开发比较困难。所幸的是,Linux 桌面系统可以利用一些Windows及其它操作系统所没有的工具。有足够经验开发设备驱动程序的开发人员可能已经习惯将Linux作为他们的桌面开发系统了。 Linux的快速发展及其桌面方案的不断涌现提出了一个重要问题:所选择的工具方案怎样在不同的Linux分布式系统上运行?它们依赖于主机平台的软件配置吗? 有些Linux工具提供独立于主机平台的开发环境,包括一系列可支持开发工具的应用软件、库和实用程序。这一方法几乎将开发环境与主机配置完全隔离开来,因此主机可以是任何Linux分布式系统,而且任何更新和修改都不会影响开发环境的功能。 这种方法的主要缺点是对存储空间的要求有所增加――约200MB,因为它自己实际上相当于一个微型Linux分布式系统。 可用的工具 一个嵌入式Linux产品的开发需要几个阶段,包括为目标板配置和构建基本Linux OS;调试应用程序、库、内核及设备驱动程序/内核模块;出货前最终方案的优化、测试和验证。 有数百种开放源代码开发工具可供选择。只要开发者原意花时间和精力去调研、实施和维护一系列各不相同的工具,总能找出一个完整的解决方案,完成几乎任何开发任务。

嵌入式基础实验报告

嵌入式基础实验报告 ——Linux下编译环境的设置 姓名:张耀丹 学号:131012692 班级:13级网络工程二班

一、实验目的 1、熟悉嵌入式Linux 开发环境,学会基于UP-CUP IOT-4412-II 型网关部分平台的Linux 开 发环境的配置和使用 2、利用arm-none-linux-gnueabi-gcc 交叉编译器编译程序,使用基于NFS 的挂载方式进行 实验,了解嵌入式开发的基本过程 二、实验环境 1、硬件:UP-CUP IOT-4412-II 型网关部分嵌入式实验平台,PC 机Pentium 500 以上, 硬盘 40G 以上,内存大于256M 2、软件:Vmware Workstation + Fedora Core 14 + 超级终端/Xshell + ARM-LINUX 交叉编译开 发环境 三、实验内容 1、本次实验使用Fedora14 操作系统环境,安装ARM-Linux 的开发库及编译器。创建一个新 目录,并在其中编写hello.c 和Makefile 文件。 2、学习在Linux 下的编程和编译过程,以及ARM 开发板的使用和开发环境的设置。将已 经编译好的文件通过NFS 方式挂载到目标开发板上运行 四、实验步骤 实验目录:/UP-CUP4412/SRC/exp/basic/Cortex/ 1、编译源程序 (1)在宿主机端任意目录下建立工作目录CortexA9,实际光盘目录中已经给出本次实验所需全面文件及代码,存放在Cortex目录下。 [root@localhost ~]# mkdir CortexA9 [root@localhost ~]# cd CortexA9 (2)编写程序源代码 在Linux 下的文本编辑器有许多,常用的是vim 和Xwindow 界面下的gedit 等,我们在开发过程中推荐使用vim,用户需要学习vim 的操作方法,请参考相关书籍中的关于vim 的操作指南。Kdevelope、anjuta 软件的界面与vc6.0 类似,使用它们对于熟悉windows 环境下开发的用户更容易上手。 实际的CortexA9.c 源代码较简单,如下: #include main() { printf(“**********\n” ); printf(“CortexA9 \n”); printf(“**********\n” ); return 0; }

嵌入式Linux应用程序开发实验报告期末作业

软件学院大作业设计报告课程名称: 题目:专业:班级:姓名学号: 基于UP-CUP2440平台的驱动程序开发和QT程序开发 计算机软件 计算机软件111班 鲁飞8000611038 卢惠民8000611021 戚成林8000611032 慕一聪8000611018 刘备8000611006 李岚职称:副教授 2014年6月11日 嵌入式Linux应用程序开发 任课教师: 完成时间:

一、小组成员分工 二、实验任务 三、主要仪器设备及耗材 四、实验步骤 、驱动 1. 基本知识: 2. 实验原理: 3. 具体实现: 二、QT计算器 1.QT程序设计 2. 虚拟机下进行编译: 3. 下载到开发板上运行: 六、实验数据及处理结果10 15 17 18 七、思考讨论题或体会或对改进实验的建议18 19 八、参考资料:

、小组成员分工 分工: 鲁飞:QT设计与设计报告卢惠民:QT设计与设计报告戚成琳:驱动与设计报告慕一聪、刘备:设计报告 二、实验任务 1?编写基于UP-CUP2440硬件平台的GPIO驱动程序 必选功能:使中断按键按下后,开发板上的LED灯能作如下闪动: a)连续性闪动,跑马灯:如:1-2-3-1-2-3…或3-2-1-3-2-1 b)间隔性闪动:如:1-3-2-1-3-2…或3-1-2-3-1-2… 进阶功能:改变中断按键的控制功能,使中断按键按下后,LED灯不断闪亮,再 次按下后,LED灯灭。 2.QT计算器 三、主要仪器设备及耗材 PC, Windows Xp , H-JTAG,H-Flasher,DNW,开发实验箱。 四、实验步骤 、驱动 思路: 前后台思想: 在中断模块中设置一个计数的变量,每一次中断计数器加一。 然后在GP10驱动模块的ioctIO函数中分情况使跑马灯按要求闪动或熄灭。

嵌入式linux系统开发概述

嵌入式linux系统开发概述 作者:谷丰,[email=您可以通 过%3Ca%20href=]gufeng77@https://www.360docs.net/doc/a14041749.html,[/email]" target="_blank">您可以通过 gufeng77@https://www.360docs.net/doc/a14041749.html,和他联系 基于linux的嵌入式系统开发是一个很大的课题,涵盖了从硬件到软件设计的多个领域,由于linux的开源特性,导致开发中可以使用的软件和工具多不胜数,从最底层与系统硬件直接打交道的引导装载程序(bootloader),到linux操作系统的分发版(distribution),再到上层的图形用户界面(GUI)乃至应用程序(application),可供选择的软件实在是太多了,这对开发者来说是一种恩赐。但由于标准的不统一,对于刚刚步入这个领域的初学者来说,很难在短时间内全部了解和掌握它们。本文论述了嵌入式linux开发的基本模式和概念,给出了一些常用的软件和工具,旨在带领他们更快的走入这个奇妙的世界。 1 引导装载程序(bootloader) 引导装载程序通常是在任何硬件上执行的第一段代码,它的主要任务视装载设备的不同而不同。在台式机和笔记本这样的常规系统中,经常存在多个操作系统并存的情况,因此bootloader的主要作用就是选择系统使用何种操作系统来引导。常用的引导程序有LILO或GRUB,通常将它们装入硬盘的主引导记录(Master Boot Record)中,或者装入linux 驻留的磁盘的第一个扇区。 在嵌入式系统中,情况有些不同。首先,嵌入式设备通常需要经常地移 动,考虑到在移动过程中的震动,一般不会采用机械式结构设计的硬盘为存 储设备;而且从成本控制上说,硬盘的价格比较高,除非是需要大容量存储 的场合,硬盘不适合作为嵌入式设备的存储介质。目前采用得比较多的是闪 存设备,闪存设备是与存储设备功能类似的特殊芯片,而且它

嵌入式Linux课程学习心得

第一篇、嵌入式系统学习心得 嵌入式Linux课程学习心得 篇一嵌入式心得体会 这学 期才接触嵌入式系统感觉还称不上入门,我通过学习知道了嵌入式的发展前景很大,各个领 域都用到了嵌入式,学好嵌入式不愁没饭吃。 广义上讲,凡是带 有微处理器的专用软硬件系统都是嵌入式系统。如各类单片机和dsp系统。从狭义上讲,那 些使用嵌入式微处理器构成独立系统,具有自己操作系统,具有特定功能,用于特定场合的

专用软硬件系统称为嵌入式系统。嵌入式系统由嵌入式硬件与嵌入式软件组成; 嵌入式硬件 以芯片、模板、组件、控制器形式埋藏于设备内部。 理解“嵌入”的概 念主要从三个方面上来理解。 1、从硬件上,将基 于cpu的处围器件,整合到cpu芯片内部,比如早期基于x86体系结构下的计算机,cpu只 是有运算器和累加器的功能,一切芯片要造外部桥路来扩展实现,象串口之类的都是靠外部 的16c550/2的串口控制器芯片实现,而目前的这种串口控制器芯片早已集成到cpu内部,还 有pc机有显卡,而多数嵌入式处理器都带有lcd控制器,但其种意义上就

相当于显卡。比较 高端的arm类intel xscale架构下的ixp网络处理器cpu内部集成pci控制器(可配成支持 4个pci从设备或配成自身为cpi从设备);还集成3个npe网络处理器引擎,其中两个对应 于两个mac地址,可用于网关交换用,而另外一个npe网络处理器引擎支持dsl,只要外面 再加个phy芯片即可以实现dsl上网功能。ixp系列最高主频可以达到8g,支持2g内存, 1g×10或10g×1的以太网口或febre channel的光通道。ixp系列应该是目标基于arm体系 统结构下由intel进行整合后成xscale内核的最高的处理器了。 2、从软件上前,就 是在定制操作系统内核里将应用一并选入,编译后将内核下载到rom中。而

嵌入式Linux系统开发教程实验报告

嵌入式实验报告 : 学号: 学院: 日期:

实验一熟悉嵌入式系统开发环境 一、实验目的 熟悉Linux 开发环境,学会基于S3C2410 的Linux 开发环境的配置和使用。使用Linux的armv4l-unknown-linux-gcc 编译,使用基于NFS 方式的下载调试,了解嵌入式开发的基本过程。 二、实验容 本次实验使用Redhat Linux 9.0 操作系统环境,安装ARM-Linux 的开发库及编译器。创建一个新目录,并在其中编写hello.c 和Makefile 文件。学习在Linux 下的编程和编译过程,以及ARM 开发板的使用和开发环境的设置。下载已经编译好的文件到目标开发板上运行。 三、实验设备及工具 硬件::UP-TECH S2410/P270 DVP 嵌入式实验平台、PC 机Pentium 500 以上, 硬盘10G 以上。 软件:PC 机操作系统REDHAT LINUX 9.0+超级终端(或X-shell)+AMR-LINUX 开发环境。 四、实验步骤 1、建立工作目录 [rootlocalhost root]# mkdir hello [rootlocalhost root]# cd hello 2、编写程序源代码 我们可以是用下面的命令来编写hello.c的源代码,进入hello目录使用vi命令来编辑代码: [rootlocalhost hello]# vi hello.c 按“i”或者“a”进入编辑模式,将上面的代码录入进去,完成后按Esc 键进入

命令状态,再用命令“:wq!”保存并退出。这样我们便在当前目录下建立了一个名为hello.c的文件。 hello.c源程序: #include int main() { char name[20]; scanf(“%s”,name); printf(“hello %s”,name); return 0; } 3、编写Makefile 要使上面的hello.c程序能够运行,我们必须要编写一个Makefile文件,Makefile文件定义了一系列的规则,它指明了哪些文件需要编译,哪些文件需要先编译,哪些文件需要重新编译等等更为复杂的命令。使用它带来的好处就是自动编译,你只需要敲一个“make”命令整个工程就可以实现自动编译。Makefile源程序: CC= armv4l-unknown-linux-gcc EXEC = hello OBJS = hello.o CFLAGS += LDFLAGS+= –static all: $(EXEC) $(EXEC): $(OBJS) $(CC) $(LDFLAGS) -o $ $(OBJS) clean: -rm -f $(EXEC) *.elf *.gdb *.o 下面我们来简单介绍这个Makefile 文件的几个主要部分: CC 指明编译器 EXEC 表示编译后生成的执行文件名称 OBJS 目标文件列表

LINUX实训心得体会

linux学习心得 我们这一代90后,从小接触的是windows98,家里条件好的自己有电脑装的是2000,后来又有了xp,上大学时又有了win7。说实话小时候没想过搞it,也计算机了解也只是一些皮毛,至于什么unix,linux,听过没见过,就更别说用过了。 以前觉得linux就跟dos一样,全是用命令窗口,相对于窗口界面来说多麻烦呀。直到学习linux这门课以后,我才知道,原来我错了。 一.为什么要学linux呢?每个人都有不同的看法,下面我说说自己的感想吧。 首先linux是开源的,这也是最主要的原因,想学windows,unix,对不起我们没源代码。也正是因为这样,linux才能够像滚雪球一样越滚越大,发展到现在这种规模。其中不乏很多it精英的心血。我们学透以后更可以做成自己的os! 其次,linux简单易学,因为我们初学者只是学的基础部分,linux的结构体系非常清晰,再加上老师循序渐进的教学以及耐心的讲解,使我们理解起来很快,短期内就基本掌握了操作和运行模式。对我们学习操作系统有很大的帮助,加深我们对os的理解。 再次,linux是用c语言编写的,我们有学习c语言的基础,读程序和编写代码方面存在的困难小一点,也是我们能较快掌握的原因之一。 二.学习情况 在学习的过程中,我们用的是vm虚拟机,开始时真的不真的该怎么去做,特别是我的是命令窗口界面,别人的是图形界面,我都不知道怎么调过来。后来通过自学老师给的资料和向同学请教,掌握了一些基本的操作,比如挂载优盘,编译程序,在linux环境下运行,转换目录等等。学了这些基础才能进行下面的模拟os程序。 老实说,第一个程序是在c中编译好的,调试好了才在linux下运行,感觉用vi比较麻烦,因为有错了不能调试,只是提示错误。但是一些显而易见的小错误还是用vi改正比较方便。以后的大一点的程序就得在linux下调试了,因为有的头文件在vc里面说找不到。 就这样,我们一边上os理论课,一边上这个实验,这样挺互补的,老师讲课,一步一步地布置任务,我们自学,就这个循环的过程中,我们学习了基本操作,用vi,shell,模拟内存的分配过程等一些os管理。可以说自己收获很大,基本上完成了老师布置的任务,对于拔高的题目没有去做,因为我了解我的水平,没有时间和精力去做。 期间我阅读了不少关于linux的相关资料,其中也不乏一些有趣的小故事,这既丰富了我的课余生活,也让我加深了对一些术语的理解,比玩游戏强多了。 三.对linux未来的展望 众所周知,目前windows操作系统是主流,在以后相当长的时间内不会有太大的改变,其方便友好的图形界面吸引了众多的用户,甚至目前许多应用软件都是基于它的。可是没有哪一个系统是十分完美的,这也正是有别的os得以存在的原因,每个系统都有其自身的优点。 linux最大的特点就是其开源性,这一点是十分难得的,这也是它能够存在到现在的原因之一,随着it从业人员越来越多,理论上会有更多的人使用linux,可以肯定,linux在以后

嵌入式linux学习心得(精选多篇)

嵌入式linux学习心得(精选多篇) 第一篇:嵌入式linux学习内容 知识结构 1嵌入式处理器与裸机程序开发2linux系统管理3linux 应用程序开发4linux驱动程序开发5linux内核开发与系统移植 一、处理器 1arm处理器工作模式2arm系统寄存器3arm寻址方式4arm汇编指令集5arm环境c语言编程6arm中断与异常7ads 集成开发环境 8裸机程序开发(串口、lcd、时钟、led、按键……) 二、系统管理 1linux定制安装2linux命令详解 3samba、nfs、tftp、wireshark使用4shell编程 三、应用程序开发 1gcc、gdb、makefile2文件、时间编程

3多进程、多线程程序设计4进程间通讯5网络编程 6qt图形化应用程序开发7android图形化应用程序开发 四、内核开发 1linux内核配置与裁剪2linux内核模块开发3根文件系统制作4进程子系统5内存子系统6proc文件系统7系统调用8内核定时器9内核异常分析 五、驱动程序开发 1字符设备驱动程序2总线、设备、驱动模型3硬件访问技术4中断处理5input设备驱动6platform驱动程序7pci、usb 驱动程序8网卡驱动程序9触摸屏驱动程序xx串口驱动程序 学习顺序 1嵌入式处理器与裸机程序开发2linux系统管理3linux 应用程序开发4linux内核开发基础5嵌入式linux环境搭建6linux驱动程序开发7深入学习linux内核 第二篇:嵌入式linux学习步骤 嵌入式linux学习步骤 作者:phantom时间:xxxx-8-6文章来源:来自网络

相关文档
最新文档