从linux内核模块发送数据

djk*_*h11 0 c sockets linux tcp linux-kernel

我想通过 tcp 从我的 linux 内核模块发送一些数据。我尝试使用来自http://www.avrfreaks.net/sites/default/files/tcp-server-send-recv.c 的一些代码 ,但代码太旧(它使用旧的 linux 内核 api)。另外,我试图了解https://github.com/abysamross/simple-linux-kernel-tcp-client-server/blob/master/network_server.c,但它对我来说太复杂了:)

我只想通过 tcp 向指定的 IP 地址发送一些小数据。我该怎么做?

Jai*_*ime 5

检查同一项目中的客户端示例。如果您可以重用某些功能,则必须仅了解和修改该tcp_client_connect功能(第 124-198 行)。在该模块中,tcp_client_connect连接在加载模块时创建连接,并在network_client_exit卸载模块时关闭连接。

tcp_client_connect函数中:

  1. (第 144 行)它创建了一个套接字

     struct socket *conn_socket = NULL;
    
     ret = sock_create(PF_INET, SOCK_STREAM, IPPROTO_TCP, &conn_socket);
    
    Run Code Online (Sandbox Code Playgroud)
  2. 然后,(第 153 到 155 行)它创建一个目标地址

     struct sockaddr_in saddr;                               /* a socket address */
    
     saddr.sin_family = AF_INET;                             /* for internet */
     saddr.sin_port = htons(PORT);                           /* using the port PORT */
     saddr.sin_addr.s_addr = htonl(create_address(destip));  /* and address destip */
    
    Run Code Online (Sandbox Code Playgroud)
  3. (第 157 行)它使用该地址打开套接字(创建连接)

     int ret = -1;
    
     ret = conn_socket->ops->connect(conn_socket, (struct sockaddr *)&saddr\
                    , sizeof(saddr), O_RDWR);     
    
     /* if it gets a response and it is not "in progress" */
     if(ret && (ret != -EINPROGRESS))
     {
         /* error creating the socket*/
     }
    
    Run Code Online (Sandbox Code Playgroud)
  4. (第 166 到 168 行)它使用套接字发送消息。

    int len = 49;
    char reply[len+1];
    
    memset(&reply, 0, len+1);   /* sets 0s into all the string space */
    strcat(reply, "HOLA");      /* sets the message */
    
    tcp_client_send(conn_socket, reply, strlen(reply), MSG_DONTWAIT);
    
    Run Code Online (Sandbox Code Playgroud)
  5. (第 170 行)它等待消息(一段时间)

    DECLARE_WAIT_QUEUE_HEAD(recv_wait);
    
    /* wait for a response or for a timetout */
    wait_event_timeout(recv_wait,\
                    !skb_queue_empty(&conn_socket->sk->sk_receive_queue),\
                                                                    5*HZ);
    
    Run Code Online (Sandbox Code Playgroud)
  6. (第 180 到 190 行)它获得响应。

    int len = 49;
    char response[len+1];
    
    /* if something has arrived */
    if(!skb_queue_empty(&conn_socket->sk->sk_receive_queue))
    {
        memset(&response, 0, len+1);
        tcp_client_receive(conn_socket, response, MSG_DONTWAIT);
    }
    
    Run Code Online (Sandbox Code Playgroud)

network_client_exit函数中,

  1. (第 239 到 240 行)它关闭了连接。

    /* if the socket has been created */
    if(conn_socket != NULL)
    {
            /* relase the socket */
            sock_release(conn_socket);
    }
    
    Run Code Online (Sandbox Code Playgroud)