登录
首页 >  Golang >  Go问答

抽象 Unix 套接字的比较:C 和 Go

来源:stackoverflow

时间:2024-03-18 09:06:31 372浏览 收藏

在使用抽象 Unix 套接字在 C 和 Go 程序之间传递数据时,遇到了连接失败的问题。C 程序使用抽象套接字名称,而 Go 程序以 @ 开头。在调查后发现,Go 程序连接到了错误的抽象套接字。这可能是由于 C 程序绑定到了错误的抽象套接字名称,导致 Go 程序无法建立连接。解决方法是确保 C 程序和 Go 程序使用相同的抽象套接字名称,并且名称长度一致。

问题内容

我正在使用抽象 unix 套接字在 c 和 go 程序之间传递数据。 c 程序正在创建套接字,go 程序连接到它。问题是 go 程序无法连接到套接字,我收到以下错误消息:

uds connection failed: dial unixgram @uds-js: connect: connection refused

这是 c 程序:

#include 
#include 
#include 
#include 

/* buffer size for the receive socket */
#define buffer_size 4096

/* the abstract unix domain socket address name */
#define uds_address_name "#uds-js"

int main() {
    int socket_fd;
    int bytes_received;
    char buffer[buffer_size];
    struct sockaddr_un server_address;
    struct sockaddr_un client_address;
    socklen_t address_length = sizeof(struct sockaddr_un);

    /* create local unix socket */
    if ( ( socket_fd = socket ( af_unix, sock_dgram, 0 ) ) < 0 ) {
        printf ( "socket error\n" );
        return 1;
    }

    /* set an abstract socket address */
    memset( &server_address, 0, sizeof(server_address) );
    server_address.sun_family = af_unix;
    strcpy( server_address.sun_path, uds_address_name );
    server_address.sun_path[0] = '\0';

    /* bind socket to address */
    if ( bind ( socket_fd, (const struct sockaddr *) &server_address, address_length ) < 0 ) {
        close ( socket_fd );
        printf ( "socket bind error\n" );
        return 1;
    }
    bytes_received =
            recvfrom(
                socket_fd,
                &buffer,
                buffer_size,
                0,
                (struct sockaddr *) &client_address,
                &address_length );
    printf ( "received: %s\n", buffer );
    return 0;
}

还有 go 程序:

import (
    "fmt"
    "net"
    "os"
)

func main() {
    addr, _ := net.resolveunixaddr("unixgram", "@uds-js")
    udssock, err := net.dialunix("unixgram", nil, addr)
    if err != nil {
        fmt.printf("uds connection failed: %v\n", err)
        os.exit(1)
    }
    defer udssock.close()
    if _, err := udssock.write([]byte("{\"test\":100}")); err != nil {
        fmt.printf("failed to send message on uds: %v\n", err)
    }
}

在 c 程序中,按照规范,我将套接字名称中的第一个字节设置为空字节。根据我在 go 中收集的信息,名称需要以 @ 开头。

运行 netstat 我可以看到套接字已创建:

$ netstat -ax | grep DGRAM
unix  2      [ ]         DGRAM                    12411992 @uds-js@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@

为什么go程序连接socket失败?

编辑:

将名称更改为路径名 /tmp/uds-js,这确实可以按预期工作。

编辑2:

我用 go 语言创建了一个服务器,用 c 语言为抽象套接字创建了一个客户端,这两个 c 程序和两个 go 程序可以很好地协同工作。问题似乎出在使用抽象套接字从 c 转到 go 时。


解决方案


Go 程序连接到错误的抽象套接字(或者,等效地,C 程序绑定到错误的抽象套接字)。

您的 bind() 是针对长度为 sizeof(struct sockaddr_un) 的抽象命名空间 UNIX 套接字地址创建的。但是,如果我是 reading the Go implementation correctly,则您的 connect() 是针对长度为 9 的套接字地址执行的:sa_family 为两个字节,在本例中为 len(name) 为 7 个字节。

这些是不同的套接字。

在抽象命名空间中,NULL并不特殊,因此 "\0uds-js" 是一个有效的套接字地址,而 "\0uds-js\0\0\0\0\0\0\0 \0..." 是一个不同的有效地址。

strace两个进程都会查看 C 程序的 bind() 和 Go 程序的 connect()。我希望您会看到他们使用不同的 socklen_t 参数调用这些函数。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>