2006年12月28日星期四

使用Winsock API发送/接收UDP包

其实Winsock这堆底层API和BSD都是一样的么…… 难怪令人怀疑这个是抄袭BSD的源代码……
UDP比较简单,只要bind一下,RP好的话就可以发了。
先 s: = socket(AF_INET,SOCK_DGRAM,0);
成功的话,s已经是一个可以用的socket了。


然后要指明绑定的addr
sa:sockaddr_in;
fillchar(sa,sizeof(sa),0);
sa.sin_family:=AF_INET;
sa.sin_port:=htons(port);
sa.sin_addr.S_addr:=inet_addr(PChar(address));
address为地址(a.b.c.d形式),port为端口号。而htons作用是把以主机存储方式存储的port转换为网络上通用的port形式。


接着就可以bind了
bind(s,sa,sizeof(sa));
成功的话,netstat -an能够看见你所bind的端口号。


bind完了,这个socket和端口也就建立了联系,所有扔到这个口的UDP包都会给这个socket


接着可以直接用recv收。但是假如没有数据过来,程序就会卡在这里,多不好……
所以用WSAAsyncSelect来指明当收到包之后给窗口发消息。
WSAAsyncSelect(s,form1.Handle,WM_RECEIVED,FD_READ);
WM_RECEIVED自己定义,FD_READ说明只关心收到消息。


接着让窗口收到这种消息的时候运行给定的proc
先定义这个proc:
Procedure TForm1.OnMyMessage(var Msg:TMsg;var Handled:Boolean);
然后在FormCreate的时候告诉窗口:
application.OnMessage:=OnMyMessage;
这样就完了


收到消息后:
  if Msg.message=WM_RECEIVED then
(万一不是呢……)
lParam的底字节代表具体情况(我们只关心FD_READ,所以这里只会是FD_READ)
高字节为出错信息
wParam为收到消息的那个socket,我们这里只有s,所以其实也一样
然后就可以recv了
recv(Msg.wParam,data,MAXMSG,0);
不出意外,消息就在data里了~


好好处理消息吧~


记得最后closesocket(s)哦~ 以免资源浪费~


 


Old Blog Link: http://computer.mblogger.cn/henryhu/posts/61586.aspx

2006年12月14日星期四

图标历险记

升级KDE到3.5.4,启动之后发现:图标不见了! 大部分图标都变成默认的空图标了……
于是去网上搜,似乎有人说KDE启动着的时候不能升级云云。费了半天劲,发现似乎KDE无法找到那些图标,可供选择的图标里也没有。去/usr/local/share/icons,来到crystalsvg目录,发现似乎就是系统知道的那些图标,而程序图标都在hicolor里面。最后想了一个暂时解决方法:把hicolor里面的东西都拷过去! 一混,就好了……
结论:电脑比较傻,骗骗电脑比较容易……


Old Blog Link: http://computer.mblogger.cn/henryhu/posts/61352.aspx

被libc弄得晕头转向……

编译 kports,结果 configure 检查QT的时候报错,查看config.log:libqt-mt.so: undefined ... __res_state
我一看,又是这个…… 从前见过,但从前加了 -pthread 就好了,现在怎么不行呢?然后,去网上找,什么都没有;去 /usr/lib和/lib和/usr/local/lib grep,都没有…… 汗……
于是去 /usr/src/lib/libpthread 里面找,一点都没有发现……
偶尔灵机一动,想到可能在别的库里面。于是到 /usr/src/lib 里面找,结果在 /usr/src/lib/libc/resolv/res_state.c 里面找到了。既然这样,为什么 /lib/libc.so.6 里面会grep不到??
尝试一下,重新编译安装libc,再试,好了!!
估计是上次没有 make buildword 的结果…… 以后一定make buildworld...


Old Blog Link: http://computer.mblogger.cn/henryhu/posts/61348.aspx

2006年12月10日星期日

FreeBSD中读取Ethernet Address的一种可行方法

#include <net/if.h>
#include <ifaddrs.h>
#include <net/if_dl.h>
#include <net/ethernet.h>

int main()
{
    struct ifaddrs *list;
    if(getifaddrs(&list) < 0)
    {
        perror("getifaddrs");
        return;
    }
        
    struct ifaddrs *cur;        
    for(cur = list; cur != NULL; cur = cur->ifa_next)
    {
        if(cur->ifa_addr->sa_family != AF_LINK)
            continue;
        char *name = cur->ifa_name;
        printf(name);printf("\n");
        char *addr;
        struct sockaddr_dl *sdl;
        sdl = (struct sockaddr_dl *)cur->ifa_addr;
                addr = ether_ntoa((const struct ether_addr *)LLADDR(sdl));
        printf(addr);printf("\n");
    }
        
    freeifaddrs(list);
    return 0;
}
看了TCP/IP Illustrated Vol 2,再看看源代码,才知道当 sa_family==AF_LINK 时,ifa_addr实际上是 struct sockaddr_dl  格式, 根据书里的写法,LLADDR宏是用来从sockaddr_dl获取ether_addr的位置,而ifa_addr里面的各个数的定义也清楚了……
struct sockaddr_dl {
    u_char    sdl_len;    /* Total length of sockaddr */
    u_char    sdl_family;    /* AF_LINK */ (当前版本AF_LINK=18, socket.h)
    u_short    sdl_index;    /* if != 0, system given index for interface */(我的rl0是1)
    u_char    sdl_type;    /* interface type */ (IFT_ETHER, 0x6, if_types.h)
    u_char    sdl_nlen;    /* interface name length, no trailing 0 reqd. */ (我的为3("rl0"))
    u_char    sdl_alen;    /* link level address length */ (ETHER_ADDR_LEN, 0x6)
    u_char    sdl_slen;    /* link layer selector length */ (0, 难道还是没有用?)
    char    sdl_data[46];    /* minimum work area, can be larger;
                   contains both if name and ll address */ ("rl0"+ether_addr...)
};
而LLADDR宏也就是把sdl_data偏移掉接口的名字,接下来就是接口的地址……
另外,struct ifaddrs和struct ifaddr很象,但是原来的ifa_ifp,也就是回指接口结构的指针没了。安全考虑? 为了区分内核和用户?
所以,坚信Linus的名言,有问题去看源代码!……
Old Blog Link: http://computer.mblogger.cn/henryhu/posts/61230.aspx

2006年12月2日星期六

An ugly way to get MAC Address in FreeBSD

char ethaddr[6];
    struct ifaddrs *list;
    if(getifaddrs(&list) < 0)
    {
        perror("getifaddrs");
    }
        
    struct ifaddrs *cur;        
    for(cur = list; cur != NULL; cur = cur->ifa_next)
    {
        if(cur->ifa_addr->sa_family != AF_LINK)
            continue;
        if(cur->ifa_name!=if_name)
            continue;
        int i;
        for (i=9;i<15;i++)
        ethaddr[i-9]=cur->ifa_addr->sa_data[i];
    }
        
    freeifaddrs(list);

I'll refer to some books to see the meaning of ifa_addr->sa_data when ifa_addr is an ethernet address...
Currently I konw:
sa_data:
0 byte 1
1 byte 0
2 byte 6 // addr len?
3 byte 3 // name len?
4 byte 6 // addr len?
5 byte 0
6,7,8 name[3]
9-14 addr[6]
... and something more...
Old Blog Link: http://computer.mblogger.cn/henryhu/posts/61231.aspx

2006年12月1日星期五

TheWorld 2 Beta 3 配合 Process Explorer 使用

TheWorld 2 Beta 3 上 MSN Space,每天 TheWorld 要死十几次。只好请来 Process Explorer。
有一次又死了,打开 Process Explorer,找到切换次数最多的线程——TheWorld的多线程用在这里不错~,结束,还不行。又找了一个,两个线程看上去都在msvcrt.dll:endthreadex里面。结束之后,TheWorld 竟然活了过来……
看来多线程还是不成熟啊…… 估计只结束前面那个就行了……


 


Old Blog Link: http://computer.mblogger.cn/henryhu/posts/61123.aspx