嵌入式系统是指将应用程序?操作系统与计算机硬件集成在一起的系统?它以应用为中心?以计算机技术为基础,而且软硬件可以裁剪,因而是能满足应用系统对功能?可靠性?成本?体积和功耗的严格要求的专用计算机系统。
本文给出一种适合于中/低端应用的通信平台设计方案,它可支持Ethernet网络之间的数据传输,并且具有RS232?RS485? USB等接口?
文中围绕嵌入式uClinux环境设计了基于嵌入式通信微处理器S3C44B0X的硬件通信平台,从而实现了嵌入式SOCKET通信?
1 嵌入式网络通信系统结构
作为一类特殊的计算机系统, 嵌入式系统通常由嵌入式处理器?嵌入式外围设备?嵌入式操作系统和嵌入式应用软件等几大部分组成?嵌入式处理器是嵌入式系统的核心部件,它可分为嵌入式微处理器?嵌入式微控制器?嵌入式DSP(DigitalSignal 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处理器的系统必须要有一个与之匹配的控制芯片?这里,笔者选用了Cirruslogic公司的CS8900A?CS8900A是一个单芯片全双工的以太网解决方案,片内集成了用于完成以太网电路所必需的所有模拟和数字电路?图2为系统中的CS8900A以太网接口电路?图中的信号发送和接收端通过网络隔离变压器和RJ45接口接入传输媒体?另外,为了系统能够正常工作,还需要外接一个20MHz的晶振?
3 基于uCliunx的SOCKET通信
本系统的软件是基于嵌入式操作系统uClinux而设计的?uClinux是一个完全符合GNU/GPL公约的操作系统,它与UNIX系统兼容,其代码完全开放?uClinux是在标准Linux基础上进行适当裁剪和优化后的操作系统,uClinux是一个高度优化?代码紧凑的Linux的嵌入式子集?虽然体积很小,但仍然保留了Linux的大多数优点,如:稳定?良好的移植性;优秀的网络功能;对各种文件系统的完备支持,以及标准丰富的API等?uClinux是专门面向没有存储器管理单元(MMU)的处理器的嵌入式操作系统,并且专为嵌入式系统做了许多小型化工作,它可以直接在Flash上运行,也可以加载到内存中运行?u-Cliunx带有一个完整的TCP/IP协议,同时也支持其他许多网络协议,对于嵌入式系统来说,它是一个网络完备的操作系统,因此得到了广泛的应用?
为了实现基于uClinux的应用系统的开发,建立或拥有一个完备的uCliunx开发环境是十分必要的?基于uClinux操作系统的应用开发环境一般由目标系统硬件开发板和宿主PC机构成3?通常需在安装有Linux的宿主PC机上安装交叉编译器,以把用户应用程序编译成目前uClinux只支持的flat格式的可执行文件和编译操作系统内核?目标硬件开发板用于运行操作系统和系统应用软件?目标硬件开发板和PC宿主机之间一般通过串口?并口或以太网接口来建立连接?本文SOCKET通信使用的软件开发及仿真环境如图3所示?
SOCKET即“套接字”,表示网络通信进程的ID?最常用的有流式套接字和数据报套接字两种?在Linux中,分别称为“SOCK STREAM”和“SOCKDGRAM”?经裁剪的uClinux保留了Linux中的大部分SOCKET库函数?基于S3C44B0X处理器的嵌入式SOCKET通信所要调用的主要库函数如下:
(1)SOCKETint SOCKET(int domain int type int proto-col)
此函数用来建立一个新的SOCKET,以通知系统建立一个通信端口?函数中的domain参数用于指定使用何种地址类型;type参数用于指定套接字类型;protocol参数通常为0,表示使用默认协议?
(2)bind intbind(int sockfd,struct sockaddr*myaddr,int addrlen)
bind函数可把SOCKET返回的套接字端口与网络上的物理位置相关联?其中sockfd参数是函数SOCKET返回的套接字描述符;myaddr参数是本地地址;ad-drlen参数是套接字地址结构的长度?服务器和客户机都可以调用函数bind来绑定套接字地址,但一般是由服务器调用函数bind来绑定自己的公认端口号?
(3)listenint listen(int sockfd,int backlog)
利用该函数可以使SOCKET端口接受从客户机发送来的连接请求?backlog参数是所能接受的客户机的最大数目?对SOCKET?bind?listen三个函数的综合调用最终可在服务器上产生一个能接受客户机请求的*文件描述符sockfd?
(4)acceptint accept(int sockfd,struct sockaddr*address,int*address_len)
当有客户机发出连接请求时,此函数初始化这个连接?其中参数address用来存储客户机的信息,此信息由accept填入?当与客户机连接时,客户机的地址与端口将填到此处;addresslen是客户机地址长度的字节数,也由accept填入?
(5)connectint connect(int sockfd,struct sockaddr*address,size_t address_len)
客户机调用SOCKET建立传输端口后,接着将调用connect函数来建立与远程服务器相连的连接线路?此函数的参数调用同bind?
本文设计的SOCKET通信采用server/client模式,即服务器端的应用程序用于接受客户端的连接请求?接收客户端的信息?处理客户端的计算请求?向客户端发送计算结果以及应答信息等?客户端的应用程序用于申请与服务器的连接?向服务器发送计算请求?处理服务器发回的计算结果和其它信息?
服务器在创建一个SOCKET后,接着会将该SOCKET与本地地址/端口号进行捆绑,成功之后就在相应的SOCKET上*,当accpet捕捉到一个连接服务请求时,即完成一个新的连接,稍后可向客户端发送数据?
客户端代码相对来说要简单一些,首先通过服务器域名获得其IP地址,然后创建一个SOCKET,接着调用connect函数来与服务器建立连接,连接成功之后再接收从服务器发送过来的数据,最后在通信结束后关闭SOCKET?
mso-ascii-font-family: 'Times New Roman'; mso-hansi-font-family: 'Times New Roman
'">综上所述,网络程序客户端和服务器端进行SOCKET通信的建立步骤如下
服务器端:SOCKET →bind →listen→accept
客户端:SOCKET →bind →connect
本设计使用的套接字为流式套接字,下面是运行于32位微处理器S3C44B0X嵌入式开发平台的客户端应用程序:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
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 %sn" argv
printf ("IP address :%sn",inet_ntoa (in));
printf("%s,%sn",hp->h_name,argv[2]);
addr.sin_addr.s_addr=inet_addr(hp->h_name);
if connect(s,(struct sockaddr *)&addr,sizeof (addr))<0){
perror ("connect");
exit (1);
}
recv (s,buffer,sizeof (buffer),0);
printf ("%sn",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 结束语
设计开发了基于S3C44B0X的以太网通信系统,并利用uClinux操作系统强大的网络功能实现了SOCKET通信?由于uClinux目前只支持flat格式的可执行文件,因此本设计是在先建立好uClinux开发环境的情况下,把用户自行开发的应用程序通过宿主PC机上的交叉编译器转化为flat格式,然后再通过FTP添加到目标硬件开发平台上运行的?文中的客户端应用程序已经在基于ARM7TDMI核的S3C44B0X处理器的嵌入式开发平台上成功运行,并成功地和宿主PC机的服务器端应用程序实现了SOCKET通信?当然,也可以把目标硬件开发平台当作服务器端,而把宿主PC机当作客户端来进行双向文件传输?