标题: 用Python的ctypes模块调用libeay32.dll中的若干函数
http://scz.617.cn/python/201604271814.txt
Python 2.7缺省自带ctypes模块,通过它可以用纯Python代码调用外部动态链接库的
导出函数。如果被调函数的返回值、形参类型涉及结构、指针,使用ctypes并不方便,
此时在Python代码中需要以ctypes要求的格式预定义结构及函数原型。如果被调函数
的返回值类型是int,形参类形是各种整型或"char *",使用ctypes倒是挺方便的,
比如kill()、prctl()之流。
下例就是不方便的那种情形,这段C代码的意图是,已知n/e生成pem文件。
--------------------------------------------------------------------------
代码: 全选
RSA *rsa;
EVP_PKEY *pkey;
rsa = RSA_new();
BN_hex2bn( &rsa->n, "0143" );
BN_hex2bn( &rsa->e, "0B" );
pkey = EVP_PKEY_new();
EVP_PKEY_assign_RSA( pkey, rsa );
pemfile = fopen( "publickey.pem", "wb" );
PEM_write_PUBKEY( pemfile, pkey );
fclose( pemfile );
RSA_free( rsa );
代码: 全选
Python的Crypto模块很容易满足原始需求,此处不纠结这个。对于不方便的那种情形,
如果非要试一下ctypes的话,有些汇编级的技巧。
--------------------------------------------------------------------------
#! /usr/bin/env python
# -*- coding: cp936 -*-
import ctypes
so = ctypes.cdll.LoadLibrary( 'libeay32.dll' )
cso = ctypes.cdll.msvcrt
so.RSA_new.restype = ctypes.POINTER( ctypes.c_char )
so.EVP_PKEY_assign.argtypes = ( ctypes.c_int, ctypes.c_int, ctypes.POINTER( ctypes.c_char ) )
so.RSA_free.argtypes = [ ctypes.POINTER( ctypes.c_char ) ]
rsa = so.RSA_new()
n = "0143"
so.BN_hex2bn( ctypes.byref( rsa.contents, 16 ), n )
e = "0B"
so.BN_hex2bn( ctypes.byref( rsa.contents, 20 ), e )
pkey = so.EVP_PKEY_new()
so.EVP_PKEY_assign( pkey, 6, rsa )
pemfile = cso.fopen( "publickey.pem", "wb" )
so.PEM_write_PUBKEY( pemfile, pkey )
cso.fclose( pemfile )
so.RSA_free( rsa )