摘要:阐述了嵌入式系统的概念及嵌入式网络通信系统的结构。给出了基于嵌入式微处理器S3C44B0X完成具有以太网络接口的嵌入式网络通信系统的电路设计方法,同时给出了在一个裁剪的uCliunx操作系统环境下实现系统与主机Socket通信的应用端用户程序。
关键词:嵌入式系统;uClinux;Socket通信;S3C44B0X
随着微电子技术的不断创新和发展,嵌入式系统已经广泛渗透到科学研究、工程设计、国防军事、自动化控制领域以及人们日常生活的方方面面。由嵌入式微控制器组成的系统其最明显的优势就是可以嵌入到任何微型或小型仪器和设备中。
嵌入式系统是指将应用程序、操作系统与计算机硬件集成在一起的系统。它以应用为中心、以计算机技术为基础,而且软硬件
1 嵌入式网络通信系统结构
作为一类特殊的计算机系统, 嵌入式系统通常由嵌入式处理器、嵌入式外围设备、嵌入式操作系统和嵌入式应用软件等几大部分组成。嵌入式处理器是嵌入式系统的核心部件,它可分为嵌入式微处理器、嵌入式微控制器、嵌入式DSP(Digital Signal Processor)和高度集成的嵌入式SoC(System on Chip)。嵌入式外围设备是指嵌入式硬件系统中除中心控制部件以外的存储、通信、保护、调试、显示等其它辅助部件。嵌入式操作系统一般在比较大型或需要多任务的应用场合才考虑使用,它可方便嵌入式应用软件的设计,大大提高嵌入式系统的功能,但同时也要占用宝贵的嵌入式资源。嵌入式应用软件和普通的应用软件有一定的区别,它是针对特定的实际专业领域、基于相应的嵌入式硬件平台、能完成用户预期任务的计算机软件。
嵌入式处理器有许多种流行的处理器核,本文主要介绍采用Samsung公司基于ARM公司32位RISC的ARM7TDMI核的S3C44B0X处理器来进行Socket通信的软硬件平台的设计和实现方法。
S3C44B0X是高性价比、高性能的微控制器,它采用ARM7TDMI核,可工作在66MHz。ARM7TDMI是一种32位嵌入式RISC处理器,但也配备了16位压缩指令集Thumb。它支持片上调试,允许处理器响应调试请求暂停,芯片内的增强型乘法器(multiplier)可进行两个32位数相乘从而直接产生64位结果,也可为嵌入式ICE硬件提供片上断点和调试点支持。此外,它还可以提供三级流水线及冯•诺依曼结构。实际上,S3C44B0X已在ARM7TDMI内容基础上扩展了一系列完整的通用外围器件。图1所示是一种嵌入式网络通信系统的硬件体系结构。
2 外围电路设计
作为优秀的网络控制器,基于S3C44B0X处理器的系统必须要有一个与之匹配的控制芯片。这里,笔者选用了Cirrus logic公司的CS8900A。CS8900A是一个单芯片全双工的以太网解决方案,片内集成了用于完成以太网电路所必需的所有模拟和数字电路。图2为系统中的CS8900A以太网接口电路。图中的信号发送和接收端通过网络隔离变压器和RJ45接口接入传输媒体。另外,为了系统能够正常工作,还需要外接一个20MHz的晶振。
&nb
本系统的软件是基于嵌入式操作系统uClinux而设计的。uClinux是一个完全符合GNU/GPL公约的操作系统,它与UNIX系统兼容,其代码完全开放。uClinux是在标准Linux基础上进行适当裁剪和优化后的操作系统,uClinux是一个高度优化、代码紧凑的Linux的嵌入式子集。虽然体积很小,但仍然保留了Linux的大多数优点,如:稳定、良好的移植性;优秀的网络功能;对各种文件系统的完备支持,以及标准丰富的API等。uClinux是专门面向没有存储器管理单元(MMU)的处理器的嵌入式操作系统,并且专为嵌入式系统做了许多小型化工作,它可以直接在Flash上运行,也可以加载到内存中运行。u-Cliunx带有一个完整的TCP/IP协议,同时也支持其他许多网络协议,对于嵌入式系统来说,它是一个网络完备的操作系统,因此得到了广泛的应用。
为了实现基于uClinux的
Socket即“套接字”,表示网络通信进程的ID。最常用的有流式套接字和数据报套接字两种。在Linux中,分别称为“SOCK STREAM”和“SOCK DGRAM”。经裁剪的uClinux保留了Linux中的大部分Socket库函数。基于S3C44B0X处理器的嵌入式Socket通信所要调用的主要库函数如下:
(1)socket int socket(int domainint typeint proto-col)
此函数用来建立一个新的socket,以通知系统建立一个通信端口。函数中的domain参数用于指定使用何种地址类型;type参数用于指定套接字类型;protocol参数通常为0,表示使用默认协议。
(2)bind int bind(int sockfd,struct sockaddr* myaddr,int addrlen)
bind函数可把socket返回的套接字端口与网络上的物理位置相关联。其中sockfd参数是函数socket返回的套接字描述符;myaddr参数是本地地址;ad-drlen参数是套接字地址结构的长度。服务器和客户机都可以调用函数bind来绑定套接字地址,但一般是由服务器调用函数bind来绑定自己的公认端口号。
(3)listen int listen(int sockfd,int backlog)
利用该函数可以使socket端口接受从客户机发送来的连接请求。backlog参数是所能接受的客户机的最大数目。对socket、bind、listen三个函数的综合调用最终可在服务器上产生一个能接受客户机请求的监听文件描述符sockfd。
(4)accept int accept(int sockfd,struct sockaddr*address,int*address_len)
当有客户机发出连接请求时,此函数初始化这个连接。其中参数address用来存储客户机的信息,此信息由accept填入。当与客户机连接时,客户机的地址与端口将填到此处;address len是客户机地址长度的字节数,也由accept填入。
(5)connect int connect(int sockfd,struct sockaddr *address,size_t address_len)
&n
本文设计的Socket通信采用server/client模式,即服务器端的应用程序用于接受客户端的连接请求、接收客户端的信息、处理客户端的计算请求、向客户端发送计算结果以及应答信息等。客户端的应用程序用于申请与服务器的连接、向服务器发送计算请求、处理服务器发回的计算结果和其它信息。
服务器在创建一个socket后,接着会将该socket与本地地址/端口号进行捆绑,成功之后就在相应的socket上监听,当accpet捕捉到一个连接服务请求时,即完成一个新的连接,稍后可向客户端发送数据。
客户端代码相对来说要简单一些,首先通过服务器域名获得其IP地址,然后创建一个
综上所述,网络程序客户端和服务器端进行Socket通信的建立步骤如下
服务器端:socket →bind →listen→accept
客户端:socket →bind →connect
本设计使用的套接字为流式套接字,下面是运行于32位微处理器S3C44B0X嵌入式开发平台的客户端应用程序
#include <netdb.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
int main(int argc,char *argv[])
{
int s;
char buffer[256];
struct sockaddr_in addr;
struct hostent *hp;
struct in_addr in;
struct sockaddr_in local_addr;
if (argc < 2) return;
if(!(hp=gethostbyname (argv[1]))){
fprintf(stderr,"Can't resolve host.\n");
exit (1);
}
if ((s = socket(AF_INET,SOCK_STREAM,0))<0){
perror ("socket");
exit (1);
}
bzero (&addr,sizeof (addr));
addr.sin_family = AF_INET;
addr.sin_port=htons((unsigned short)atoi(argv[2]));
hp = gethostbyname(argv[1]);
memcpy(&local_addr.sin_addr.s_addr,hp->h addr,4);
in.s_addr = local_addr.sin_addr.s_addr;
printf ("Domain Name%s\n"argv
printf ("IP address :%s\n",inet_ntoa (in));
printf("%s,%s\n",hp->h_name,argv[2]);
addr.sin_addr.s_addr=inet_addr(hp->h_name);
ifconnect(s,(struct sockaddr *)&addr,sizeof (addr))<0){
perror ("connect");
exit (1);
}
recv (s,buffer,sizeof (buffer),0);
printf ("%s\n",buffer);
while(1);
bzero(buffer,sizeof (buffer));
read (STDIN_FILENO,buffer,sizeof(buffer));
if (send (s,buffer,sizeof (buffer),0)<0){
perror ("send");
exit (1);
}
}
}
4 结束语