c - pthread_create 失败并返回 -1(或 4294967295)

标签 c multithreading openssl

我正在尝试重现OpenSSL 的网络安全(由 Chandra 等人)中的示例。该程序包括 client.c server.c common.hcommon.cclient.c只需在端口 6012 上创建到服务器的连接并从 stdin 读取数据。然后将这些数据发送到服务器。 server.c从套接字读取数据并将其写回到 stdout .

问题是 server.c总是卡在 if(BIO_do_accept(acc) <= 0)在第 62 行,永远不会收到从 client.c 发送的数据,运行得非常好。后来我发现问题是pthread_create(...)在第 66 行(在 THREAD_CREATE(...) 中定义为 common.h )失败并返回 4294967295 。因为THREAD_CREATE(...)失败,程序永远没有机会运行do_server_loopserver_thread(...) ,这解释了为什么服务器从未从客户端获取数据。

我应该如何解释 pthread_create 的返回值我应该如何解决它? 附言。我后来用了strerror转换4294967295它返回“未知错误”。

任何帮助将不胜感激!

////////////////////////////////////////////////////////////////////////////////////////////

这里是server.c :

#include "common.h" 
void do_server_loop(BIO *conn)
{
    int err, nread;
    char buf[80];

    do
    {
        fprintf(stderr, "server_loop executed.\n");
        for(nread = 0; nread < sizeof(buf); nread += err)
        {
            err = BIO_read(conn, buf + nread, sizeof(buf) - nread);
            if(err <= 0){
                break;
            }
        }
        fwrite(buf, 1, nread, stdout);
    }
    while (err > 0);
}

void THREAD_CC server_thread(void *arg)
{
    fprintf(stderr, "server_thread(void *arg) executed.\n");
    BIO *client = (BIO *)arg;
#ifndef WIN32
    pthread_detach(pthread_self());
#endif
    fprintf(stderr, "Connection opened.\n");
    do_server_loop(client);
    fprintf(stderr, "Connection closed.\n");

    BIO_free(client);
    ERR_remove_state(0);
#ifdef WIN32
    _endthread();
#else
    return 0;
#endif
}


int main(int argc, char *argv[])
{
    BIO     *acc, *client;
    int thread_create_result;
    THREAD_TYPE tid;

    init_OpenSSL();

    acc = BIO_new_accept(PORT);
    if(!acc){
        int_error("Error creating server socket");
    }

    if(BIO_do_accept(acc) <= 0){
        int_error("Error binding server socket");   
    }
    for(;;)
    {
        if(BIO_do_accept(acc) <= 0){
            int_error("Error accepting connection");
        }
        client = BIO_pop(acc);
        thread_create_result = THREAD_CREATE(tid, server_thread, client);
        if(thread_create_result != 0){
            fprintf(stderr, "THREAD_CREATE failed! returns: %s.\n", \
                                            strerror(thread_create_result));
            fprintf(stderr, "thread_create_result has the value: %u.\n", \
                                            thread_create_result);
            exit(-1);
        }
    }

    BIO_free(acc);
    return 0;
}

这里是common.h

#include <errno.h>
#include <fcntl.h>
#include <netdb.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <openssl/bio.h>
#include <openssl/err.h>
#include <openssl/hmac.h>
#include <openssl/objects.h>
#include <openssl/rand.h>
#include <openssl/ssl.h>
#include <openssl/x509.h>
#include <openssl/x509v3.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <strings.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>


#ifndef WIN32
#include <pthread.h>
#define THREAD_CC *
#define THREAD_TYPE         pthread_t
#define THREAD_CREATE(tid, entry, arg)  pthread_create(&(tid), NULL, \
                                   (entry), (arg))
#else
#include <windows.h>
#include <process.h>
#define THREAD_CC              __cdecl
#define THREAD_TYPE            DWORD
#define THREAD_CREATE(tid, entry, arg) do { _beginthread((entry), 0, (arg));\
                        (tid) = GetCurrentThreadId();\
                       } while (0)
#endif

#define PORT    "6012"      //port
#define SERVER  "10.1.251.24"   //server address
#define CLIENT  "10.1.21.46"    //client address

#define int_error(msg) handle_error(__FILE__, __LINE__, msg)
void handle_error(const char *file, int lineno, const char *msg);

void init_OpenSSL(void);

生成文件:

CC = gcc
OPENSSLDIR = /usr/local/ssl
#CFLAGS = -g -Wall -W -I${OPENSSLDIR}/include -O2 -D_REENTRANT -D__EXTENSIONS__  
CFLAGS = -g -Wall -W -I${OPENSSLDIR}/include -O2

RPATH = -R${OPENSSLDIR}/lib
#LD = ${RPATH} -L${OPENSSLDIR}/lib -lssl -lcrypto -lsocket -lnsl -lpthread
LD = -L${OPENSSLDIR}/lib -lssl -lcrypto -lsocket -lnsl -pthread

OBJS = common.o

PROGS = server

all: ${PROGS}

server: server.o ${OBJS}
        ${CC} server.o ${OBJS} -o server ${LD}


clean:;
        ${RM} ${PROGS} *.ln *.BAK *.bak *.o

最佳答案

改变

if(thread_create_result = !0){

if(thread_create_result != 0){

此外,您可以使用 strerror 函数将错误代码转换为人类可读的形式。

关于c - pthread_create 失败并返回 -1(或 4294967295),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3363202/

相关文章:

iphone - 在 Xcode 中编译失败,因为找不到链接库

c - RAND_bytes 不会从同一种子给出相同的结果

c - C 中的二进制操作

c - GCC 中如何实现精确宽度的整数类型?

c# - 如何强制线程内的一段代码不间断运行(防止被操作系统抢占)

c++ - openssl RSA_private_decrypt 和 RSA_private_encrypt 有什么区别?

谁能给我解释一下这段代码?

C 标准输入不起作用

Java 线程与对象

java - 应用程序如何处理异步响应 - 通过回调