参考链接
Server
层次结构
- lib存放静态编译gmssl代码生成的静态库,即libssl.a和libcrypto.a
- pem存放证书文件,因为SSL双向验证,服务端会验证客户端的证书,因此服务端存放客户端的证书和私钥,以及生成客户端证书时用于签名的ca证书

Code
void ShowCerts(SSL * ssl)
cert = SSL_get_peer_certificate(ssl);
if(SSL_get_verify_result(ssl) == X509_V_OK){
line = X509_NAME_oneline(X509_get_subject_name(cert), 0, 0);
printf("证书: %s\n", line);
line = X509_NAME_oneline(X509_get_issuer_name(cert), 0, 0);
printf("颁发者: %s\n", line);
int main(int argc, char **argv) {
struct sockaddr_in my_addr, their_addr;
unsigned int myport, lisnum;
OpenSSL_add_all_algorithms();
SSL_load_error_strings();
ctx = SSL_CTX_new(SSLv23_server_method());
ERR_print_errors_fp(stdout);
SSL_CTX_set_verify(ctx, SSL_VERIFY_PEER|SSL_VERIFY_FAIL_IF_NO_PEER_CERT, NULL);
if (SSL_CTX_load_verify_locations(ctx, "/home/chy-cpabe/CLionProjects/ssl_client/src/pem/CaCert.pem",NULL)<=0){
ERR_print_errors_fp(stdout);
if (SSL_CTX_use_certificate_file(ctx, argv[3], SSL_FILETYPE_PEM) <= 0) {
ERR_print_errors_fp(stdout);
if (SSL_CTX_use_PrivateKey_file(ctx, argv[4], SSL_FILETYPE_PEM) <= 0) {
ERR_print_errors_fp(stdout);
if (!SSL_CTX_check_private_key(ctx)) {
ERR_print_errors_fp(stdout);
if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
printf("socket created\n");
bzero(&my_addr, sizeof(my_addr));
my_addr.sin_family = PF_INET;
my_addr.sin_port = htons(myport);
my_addr.sin_addr.s_addr = INADDR_ANY;
if (bind(sockfd, (struct sockaddr *) &my_addr, sizeof(struct sockaddr))
if (listen(sockfd, lisnum) == -1) {
printf("begin listen\n");
len = sizeof(struct sockaddr);
if ((new_fd = accept(sockfd, (struct sockaddr *) &their_addr, &len))
printf("server: got connection from %s, port %d, socket %d\n",
inet_ntoa(their_addr.sin_addr), ntohs(their_addr.sin_port),
if (SSL_accept(ssl) == -1) {
strcpy(buf, "server->client");
len = SSL_write(ssl, buf, strlen(buf));
printf("消息'%s'发送失败!错误代码是%d,错误信息是'%s'\n", buf, errno,
printf("消息'%s'发送成功,共发送了%d个字节!\n", buf, len);
len = SSL_read(ssl, buf, MAXBUF);
printf("接收消息成功:'%s',共%d个字节的数据\n", buf, len);
printf("消息接收失败!错误代码是%d,错误信息是'%s'\n",
CMakeLists.txt
cmake_minimum_required(VERSION 3.22)
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_FLAGS "-Wno-error=deprecated-declarations -Wno-deprecated-declarations ")
link_directories(${PROJECT_SOURCE_DIR}/lib)
include_directories(/usr/local/gmssl/include)
add_executable(${PROJECT_NAME} ssl_server.cpp)
target_link_libraries(${PROJECT_NAME} libssl.a libcrypto.a pthread dl)
配置执行输入参数

执行结果

Client
层次结构
- lib存放静态编译gmssl代码生成的静态库,即libssl.a和libcrypto.a
- pem存放证书文件,因为SSL双向验证,客户端同样会验证客户端的证书,因此客户端存放服务端的证书和私钥,以及生成服务端证书时用于签名的ca证书

Code
void ShowCerts(SSL * ssl)
cert = SSL_get_peer_certificate(ssl);
if(SSL_get_verify_result(ssl) == X509_V_OK){
line = X509_NAME_oneline(X509_get_subject_name(cert), 0, 0);
printf("证书: %s\n", line);
line = X509_NAME_oneline(X509_get_issuer_name(cert), 0, 0);
printf("颁发者: %s\n", line);
int main(int argc, char **argv)
printf("参数格式错误!正确用法如下:\n\t\t%s IP地址 端口\n\t比如:\t%s 127.0.0.1 80\n此程序用来从某个"
"IP 地址的服务器某个端口接收最多 MAXBUF 个字节的消息",
OpenSSL_add_all_algorithms();
SSL_load_error_strings();
ctx = SSL_CTX_new(SSLv23_client_method());
ERR_print_errors_fp(stdout);
SSL_CTX_set_verify(ctx, SSL_VERIFY_PEER|SSL_VERIFY_FAIL_IF_NO_PEER_CERT, NULL);
if (SSL_CTX_load_verify_locations(ctx, "/home/chy-cpabe/CLionProjects/ssl_client/src/pem/CaCert.pem",NULL)<=0){
ERR_print_errors_fp(stdout);
if (SSL_CTX_use_certificate_file(ctx, argv[3], SSL_FILETYPE_PEM) <= 0) {
ERR_print_errors_fp(stdout);
if (SSL_CTX_use_PrivateKey_file(ctx, argv[4], SSL_FILETYPE_PEM) <= 0) {
ERR_print_errors_fp(stdout);
if (!SSL_CTX_check_private_key(ctx)) {
ERR_print_errors_fp(stdout);
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
printf("socket created\n");
bzero(&dest, sizeof(dest));
dest.sin_family = AF_INET;
dest.sin_port = htons(atoi(argv[2]));
if (inet_aton(argv[1], (struct in_addr *) &dest.sin_addr.s_addr) == 0) {
printf("address created\n");
if (connect(sockfd, (struct sockaddr *) &dest, sizeof(dest)) != 0) {
printf("server connected\n");
if (SSL_connect(ssl) == -1)
ERR_print_errors_fp(stderr);
printf("Connected with %s encryption\n", SSL_get_cipher(ssl));
bzero(buffer, MAXBUF + 1);
len = SSL_read(ssl, buffer, MAXBUF);
printf("接收消息成功:'%s',共%d个字节的数据\n",
("消息接收失败!错误代码是%d,错误信息是'%s'\n",
bzero(buffer, MAXBUF + 1);
strcpy(buffer, "from client->server");
len = SSL_write(ssl, buffer, strlen(buffer));
("消息'%s'发送失败!错误代码是%d,错误信息是'%s'\n",
buffer, errno, strerror(errno));
printf("消息'%s'发送成功,共发送了%d个字节!\n",
CMakeLists.txt
cmake_minimum_required(VERSION 3.22)
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_FLAGS "-Wno-error=deprecated-declarations -Wno-deprecated-declarations ")
link_directories(${PROJECT_SOURCE_DIR}/lib)
include_directories(/usr/local/gmssl/include)
link_libraries(ssl crypto)
add_executable(${PROJECT_NAME} ssl_client.cpp)
target_link_libraries(${PROJECT_NAME} libssl.a libcrypto.a pthread dl)
配置执行输入参数
- 127.0.0.1
7838
/home/chy-cpabe/CLionProjects/ssl_client/src/pem/TerminalCert.pem
/home/chy-cpabe/CLionProjects/ssl_client/src/pem/TerminalKey.pem

执行结果
