Chinaunix

标题: 关于struct的一个问题,很紧急,谢谢 [打印本页]

作者: k_eric    时间: 2008-03-21 09:17
标题: 关于struct的一个问题,很紧急,谢谢
最近需要写一个网络程序,其中网络传输部分采用了udt的代码,我把udt的c++库用swig封装成了pyd库文件,可以调用,但udt库中有一个函数,
udt.bind(UDTSOCKET u,struct  sockaddr* name, int* namelen)
这个函数需要一个struct类型的sockaddr指针。

其中sockaddr的定义:
struct sockaddr {
        u_short sa_family;              /* address family */
        char    sa_data[14];            /* up to 14 bytes of direct address */
};

广泛采用的等效定义:
//Socket address, internet style.
struct sockaddr_in {
        short   sin_family;
        u_short sin_port;
        struct  in_addr sin_addr;
        char    sin_zero[8];
};

//Internet address (old style... should be updated)
struct in_addr {
        uint32_t s_addr;
};


请问:我怎么用python的struct模块把sockaddr_in 做出来?
谢谢,很紧急,忙了两天了。
作者: k_eric    时间: 2008-03-21 09:35
在线等,等高手指点迷津
作者: k_eric    时间: 2008-03-21 10:00
继续等,怎么没有人答复,给点儿线索也行啊
作者: k_eric    时间: 2008-03-21 11:28
等待中
作者: seewind    时间: 2008-03-21 12:39
用python的ctypes模块,可以作出来。
import ctypes
class sockaddr(ctypes.Structure):
    """
struct sockaddr {
        u_short sa_family;              /* address family */
        char    sa_data[14];            /* up to 14 bytes of direct address */
};

    """
    _fields_ = [
        ('sa_family', ctypes.c_ushort),
        ('sa_data', ctypes.c_char * 14),
        ]

name = sockaddr()

#调用函数时需要用ctypes.byref,如:
udt.bind(u,ctypes.byref(name), namelen)

对于想和外部dll交互,使用ctypes模块是一个不错的选择。
它还提供了很多其他功能,如:
from_address
addressof
...
不过我在用ctypes模块调用一个dll,而这个dll又有使用py模块时,出现了个莫名其妙的问题。

[ 本帖最后由 seewind 于 2008-3-21 12:43 编辑 ]
作者: k_eric    时间: 2008-03-21 12:54
谢谢楼上的指点,太感谢了,原来ctypes模块这么方便
作者: k_eric    时间: 2008-03-21 13:53
还是有问题,


class sockaddr(ctypes.Structure):
    _fields_=[("sa_family",ctypes.c_ushort),
              ("sa_data",ctypes.c_char*14),
              ]
   
class in_addr(ctypes.Structure):
    _fields_=[("s_addr",ctypes.c_uint32)]

class sockaddr_in(ctypes.Structure):
    _fields_=[("sin_family",ctypes.c_short),
              ("sin_port",ctypes.c_ushort),
              ("sin_addr",in_addr),
              ("sinzero",ctypes.c_char*,
              ]

my_in_addr=in_addr(struct.unpack('I',socket.inet_aton(IP))[0])
myaddr=sockaddr_in(AF_INET,
                   socket.htons(Port),
                   my_in_addr,
                   '\x00\x00\x00\x00\x00\x00\x00\x00',
                   )
myaddr_p=ctypes.pointer(myaddr)

mysocket=udt.socket(AF_INET,SOCK_STREAM,0)
udt.bind(mysocket,myaddr_p,ctypes.sizeof(myaddr))


系统显示:
Traceback (most recent call last):
  File "C:\yuan\work\udt-dll\test_server.py", line 103, in <module>
    udt.bind(mysocket,myaddr_p,ctypes.sizeof(myaddr))
TypeError: in method 'bind', argument 2 of type 'sockaddr const *'
>>>
作者: seewind    时间: 2008-03-21 14:41
udt.bind(mysocket,myaddr_p,ctypes.sizeof(myaddr))
打错字了吧?是byref
作者: k_eric    时间: 2008-03-21 14:44
byref跟pointer效果一样,我都试过了,一样的错误
作者: k_eric    时间: 2008-03-21 14:51
主要错误原因就是,myaddr是sockaddr_in格式的,不是sockaddr格式,这两种格式内存长度是一样的,sockaddr很难赋值,所以c程序当中都是用sockaddr_in来构造,然后等调用函数的时候,再强制类型转化

附:c语言的语句
int main()
{
UDTSOCKET serv = UDT::socket(AF_INET, SOCK_STREAM, 0);

sockaddr_in my_addr;
my_addr.sin_family = AF_INET;
my_addr.sin_port = htons(9000);
my_addr.sin_addr.s_addr = INADDR_ANY;
memset(&(my_addr.sin_zero), '\0', ;

if (UDT::ERROR == UDT::bind(serv, (sockaddr*)&my_addr, sizeof(my_addr)))
{
  cout << "bind: " << UDT::getlasterror().getErrorMessage();
  return 0;
}
作者: k_eric    时间: 2008-03-21 14:54
现在需要一个可以转换类型的函数,这个就是ctypes.cast(),但这个函数,我用起来有问题,唉,痛苦啊,大家帮忙看看。
作者: k_eric    时间: 2008-03-21 15:14
bind(,,)
第一个参数为要绑定的socket.
第二个参数指定该套接字的本地地址信息,是指向sockaddr结构的指针变量,对于不同协议不同(所以我们要用到第三个参数sizeof(sockddr)),在TCP/IP协议中,我们可以用socketaddr_in结构代替sockaddr。
struct socketaddr_in
{
short sin_family;//这个固定为AF_INET,
unsigned short sin_port;//端口号,注意,这里必须用网络字节序,调用函数htons函数
struct in_addr sin_addr;//IP地址,如果指定为INADDR_ANY,则允许套接字向任何分配给本机的IP地址发送或者接收数据,这对于有多个网卡的机器来说,就简化了程序的编写,如果只想让一个IP地址接收数据,则必须指定具体某个IP,这可以用inet_addr()函数,注意,这里也必须用网络字节序,调用函数htonl函数
char sin_zero[8];//只是个填充,以保证socketaddr_in与sockaddr长度一样
}
bind()函数调用成功返回0,失败就会返回一个SOCKET_ERROR,同样的错误信息可以通过WSAGetLastError函数返回。
作者: k_eric    时间: 2008-03-21 15:15
把资料补齐,大家帮忙看看,很紧急,谢谢了。
作者: seewind    时间: 2008-03-21 15:28
1、“udt的c++库用swig封装成了pyd库文件”,你已经封装成pyd库了,那应该可以不需要ctypes模块拉,这些回调函数什么的,在pyd里面就可以过滤掉,最终在py里面使用的是py的对象、方法。。。
2、“bind(,,)第一个参数为要绑定的socket.第二个参数指定该套接字的本地地址信息”
指针的具体类型应该是没啥具体用处的,只是编译器用来语法校验的吧。python里面没有什么sockaddr、sockaddr_in结构,所以,在调用c函数里面如果需要指针参数,就用byref一个ctypes的变量就行。
作者: k_eric    时间: 2008-03-21 15:44
我用ctypes的目的,不是调用windows版的udt.dll,我已经打包为python的udt.pyd,而是为了构造c语言的struct格式,udt里面的函数都是可以访问的,但调用的话,需要传一个c语言的struct格式的参数,就是上面的sockaddr

现在我用ctypes可以构造出sockaddr_in的格式,而且sockaddr的格式和sockaddr_in应该是一样的,但python就是不认,一直出现上面的错误,说我传的参数类型不对
作者: seewind    时间: 2008-03-21 15:57
1、“而是为了构造c语言的struct格式、调用的话,需要传一个c语言的struct格式的参数”,
这些可以写在pyd里面阿,用c代码构造、调用就可以拉,经过pyd后,py看到的udt应该是一个完全的py模块才对。
之前我也在写py扩展,我用pyrex,简单方便,用bcb编译。

2、有没简单例子,传个上来
作者: k_eric    时间: 2008-03-21 16:05
我没有用pyrex,我用的是swig,自动封装好的,源代码是C++格式的,现在udt在python眼里确实是一个类,里面包含了很多方法,包括异常处理都包含了,但具体到调用方法的时候,参数还是c++格式的,是不是我打包不彻底,还是其他的原因

谢谢seewind
作者: seewind    时间: 2008-03-21 16:13
“但具体到调用方法的时候,参数还是c++格式”,这应该是包装不彻底吧。
你试下将特殊的调用方法也包装下
作者: k_eric    时间: 2008-03-21 16:18
具体的bind函数包装代码,我改一下这段代码试试

SWIGINTERN PyObject *_wrap_bind(PyObject *SWIGUNUSEDPARM(self), PyObject *args) {
  PyObject *resultobj = 0;
  UDTSOCKET arg1 ;
  sockaddr *arg2 = (sockaddr *) 0 ;
  int arg3 ;
  int result;
  int val1 ;
  int ecode1 = 0 ;
  void *argp2 = 0 ;
  int res2 = 0 ;
  int val3 ;
  int ecode3 = 0 ;
  PyObject * obj0 = 0 ;
  PyObject * obj1 = 0 ;
  PyObject * obj2 = 0 ;
  
  if (!PyArg_ParseTuple(args,(char *)"OOO:bind",&obj0,&obj1,&obj2)) SWIG_fail;
  ecode1 = SWIG_AsVal_int(obj0, &val1);
  if (!SWIG_IsOK(ecode1)) {
    SWIG_exception_fail(SWIG_ArgError(ecode1), "in method '" "bind" "', argument " "1"" of type '" "UDTSOCKET""'");
  }
  arg1 = static_cast< UDTSOCKET >(val1);
  res2 = SWIG_ConvertPtr(obj1, &argp2,SWIGTYPE_p_sockaddr, 0 |  0 );
  if (!SWIG_IsOK(res2)) {
    SWIG_exception_fail(SWIG_ArgError(res2), "in method '" "bind" "', argument " "2"" of type '" "sockaddr const *""'");
  }
  arg2 = reinterpret_cast< sockaddr * >(argp2);
  ecode3 = SWIG_AsVal_int(obj2, &val3);
  if (!SWIG_IsOK(ecode3)) {
    SWIG_exception_fail(SWIG_ArgError(ecode3), "in method '" "bind" "', argument " "3"" of type '" "int""'");
  }
  arg3 = static_cast< int >(val3);
  result = (int)UDT::bind(arg1,(sockaddr const *)arg2,arg3);
  resultobj = SWIG_From_int(static_cast< int >(result));
  return resultobj;
fail:
  return NULL;
}
作者: lincolnrainbow    时间: 2008-09-27 14:57
此问题我已解决,具体请看我的文章
http://blog.chinaunix.net/u2/79621/showart_1210923.html




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2