2017-10-18 99 views
1

我想創建一個* .so文件,以便在使用SWIG的Python中進一步使用。特別是我使用了openssl的一些庫(例如opensll/bn.h)。但不知何故,它返回錯誤ImportError: [...]/auxchash.so: undefined symbol: BN_bn2hexswig c + + python與openssl/bn.h

我有file.cpp,auxchash.cpp:

#include auxchash.h 

int keygen(int bits, char *p, char *q, char *g, char *hk, char *tk){ 
    BN_CTX *ctx = BN_CTX_new(); 
    BIGNUM *bn_p = BN_new(); 
    BIGNUM *bn_q = BN_new(); 
    BIGNUM *bn_g = BN_new(); 
    BIGNUM *bn_hk = BN_new(); 
    BIGNUM *bn_tk = BN_new(); 
    BIGNUM *bn_two = BN_new(); 

    BN_CTX_init(ctx); 

    BN_dec2bn(&bn_two, "2"); //initialize a BIGNUM with value 2 

    //on non-unix platform needs to initialize the PRNG with randomness 
    //or BN_generate_prime_ex may fail 

    //computing the safe prime p and q = (p-1)/2 
    BN_generate_prime_ex(bn_p, bits, 1, NULL, NULL, NULL); 
    BN_sub(bn_q, bn_p, BN_value_one()); 
    BN_div(bn_q, NULL, bn_q, bn_two, ctx); 

    //finding the generator g (for the group QR_p) 
    BN_rand_range(bn_g, bn_p); 
    BN_mod_exp(bn_g, bn_g, bn_two, bn_p, ctx); 

    //choosing the keys hk and tk 
    BN_rand_range(bn_tk, bn_q); 
    BN_mod_exp(bn_hk, bn_g, bn_tk, bn_p, ctx); 

    //converting from BIGNUM to hex 
    p = BN_bn2hex(bn_p); 
    q = BN_bn2hex(bn_q); 
    g = BN_bn2hex(bn_g); 
    hk = BN_bn2hex(bn_hk); 
    tk = BN_bn2hex(bn_tk); 

    //freeing the resources 
    BN_CTX_free(ctx); 
    BN_free(bn_two); 
    BN_free(bn_p); 
    BN_free(bn_q); 
    BN_free(bn_g); 
    BN_free(bn_hk); 
    BN_clear_free(bn_tk); 

    return 0; 
} 

的file.h,auxchash.h:

#include<stdio.h> 
#include<string.h> 
#include<stdlib.h> 
#include<openssl/bn.h> 
#include<openssl/sha.h> 
#include<openssl/rand.h> 

int keygen(int bits, char *p, char *q, char *g, char *hk, char *tk); 

爲痛飲模塊file.i,auxchash.i :

%module auxchash 
%{ 
#define SWIG_FILE_WITH_INIT 
#include "auxchash.h" 
#include<stdio.h> 
#include<string.h> 
#include<stdlib.h> 
#include<openssl/bn.h> 
#include<openssl/sha.h> 
#include<openssl/rand.h> 
%} 

%include "typemaps.i" 
%include "cstring.i" 

%cstring_bounded_output(char *p, 1024); 
%cstring_bounded_output(char *q, 1024); 
%cstring_bounded_output(char *g, 1024); 
%cstring_bounded_output(char *hk, 1024); 
%cstring_bounded_output(char *tk, 1024); 
extern int keygen(int bits, char *p, char *q, char *g, char *hk, char *tk); 

最後一個文件setup.py創建所有需要的文件痛飲,setup.py:

from distutils.core import setup, Extension 


auxchash_module = Extension('_auxchash', 
         sources=['auxchash_wrap.cxx', 'auxchash.cpp'], 
         ) 

setup (name = 'auxchash', 
    version = '0.1', 
    author  = "SWIG Docs", 
    description = """Simple swig example from docs""", 
    ext_modules = [auxchash_module], 
    py_modules = ["auxchash"], 
    ) 

而且所有的人都編譯終端命令:

swig -c++ -python auxchash.i 
python setup.py build_ext --inplace 

到目前爲止好,它編譯沒有錯誤。但後來當我運行主蟒蛇:

import auxchash 
res,p,q,g,hk,tk = auxchash.keygen(10) 

它給我以下錯誤:

File: "[...]/auxchash.py" import auxchash 
File: "[...]/auxchash.py" auxchash=swig_import_helper() 
File: "[...]/auxchash.py" return=importlib.import_module('_auxchash') 
File: "[...]/__init.py__" __import__(name)` 
ImportError: [...]/auxchash.so: undefined symbol: BN_bn2hex 

我不知道如何計算出來。

+0

在一個快速閱讀,我認爲你需要'int keygen(int bits,char ** p,char ** q,char ** g,char ** hk,char ** tk)''所以函數可以改變調用者的指針。類似'* p = BN_bn2hex(bn_p)'。 – jww

+0

關於'auxchash.so:未定義的符號:BN_bn2hex','ldd auxchash.so'顯示了什麼?是否存在對'libcrypto.so'的依賴? – jww

+0

是@jww你是對的,你寫的是絕對正確的。但即使使用swig和char **,我也遇到了麻煩。我決定採用這種方法,因爲我無法處理swig中char *的雙指針。看到這[鏈接](https://stackoverflow.com/questions/46776757/swig-char-as-a-pointer-to-a-char?noredirect=1#comment80538743_46776757)。 – lorsp000

回答

0

您需要將自己對OpenSSL的模塊,像egsomething:

auxchash_module = Extension('_auxchash', 
         sources=['auxchash_wrap.cxx', 'auxchash.cpp'], 
         libraries=['crypto', 'ssl'], 
         ) 

(您可能只需要密碼在該列表中,我不太記得了/你能告訴現在)

+0

非常感謝!是的,添加庫參數,它似乎工作。 (: – lorsp000

+0

它可以工作,但現在我又遇到了另一個問題:在我的python main上打印返回變量會打印一些難以理解的字符,另一方面,在C++函數中,打印相同的值,我得到了正確的值。在swing模塊的輸出中是錯誤的,但'%cstring_bounded_output'應該正確地返回一個char *。這是鏈接到庫嗎? – lorsp000

+0

爲了更好理解,我嘗試了在簡單字符串「hello」中寫入返回變量。蟒蛇主要是無法得到它們正確的(即它仍然打印難以辨認的字符 – lorsp000