【问题标题】:Stream Read Blocking UDP GNAT流读取阻塞 UDP GNAT
【发布时间】:2017-04-07 05:29:33
【问题描述】:

我必须从 C++ 客户端接收 UDP 数据报。客户端正常工作并将数据报发送到我编译此代码的 ip。数据报的大小是 800Bytes。当我发送数据报时,读取过程被阻止。我按照这个例子创建了这个代码:https://en.wikibooks.org/wiki/Ada_Programming/Libraries/GNAT.Sockets_examples,这在我的主机上完美运行。有什么想法吗?

procedure RECEIVE_DATA (
                         DEST_UDP_PORT        : In  WORD_TYPE;
                         SRC_UDP_PORT         : In  WORD_TYPE;
                         WAIT_TIME       : In  DURATION;
                         MESSAGE_ADDRESS : Out System.Address;
                         WAIT_RESULT     : Out Boolean
                         ) is

task Receive is
    entry Start;
end Receive;

task body Receive is
    Address  : Sock_Addr_Type;
    Socket   : Socket_Type;
    Channel  : Stream_Access;
    Receive_Timeout : constant Duration := WAIT_TIME;
    Offset : Ada.Streams.Stream_Element_Count;
    Data   : Ada.Streams.Stream_Element_Array (1 .. 800);
begin
    accept Start;
    Initialize;
    WAIT_RESULT := False;       
    Create_Socket (Socket, Family_Inet, Socket_Datagram);
    Set_Socket_Option (Socket => Socket,
                       Option => (Gnat.Sockets.Receive_Timeout, Timeout => Receive_Timeout));

    Address.Addr := Inet_Addr(DEST_UDP_IP);
    Address.Port := Port_Type(DEST_UDP_PORT);
    Bind_Socket (Socket, Address);

    Address.Addr := Inet_Addr(SRC_UDP_IP);
    Address.Port := Port_Type(SRC_UDP_PORT);
    Channel := Stream (Socket, Address);

    loop
        Ada.Streams.Read (Channel.All, Data, Offset);
        exit when Offset = 0;
    end loop;
    Free (Channel);
    Close_Socket (Socket);
    WAIT_RESULT := True;
    MESSAGE_ADDRESS := Data'Address;

    exception when E : others =>
        Ada.Text_IO.Put_Line
        (Exception_Name (E) & ": " & Exception_Message (E));
         WAIT_RESULT := False;
         Free (Channel);
         Close_Socket (Socket);
end Receive;

begin
    Initialize (Process_Blocking_IO => False);
    Receive.Start;
    Finalize;
end RECEIVE_DATA;

【问题讨论】:

    标签: sockets stream udp ada gnat


    【解决方案1】:

    您是否尝试过使用 GNAT.Sockets.Receive_Socket ? 流与 TCP 一起使用,您不能流式传输数据报。

    例子:

    客户:

    with Ada.Streams;
    with Ada.Text_IO;
    
    with GNAT.Sockets;
    
    procedure UDP_Client is
       use GNAT.Sockets;
       Address : Sock_Addr_Type;
       Socket : Socket_Type;
       Data : constant Ada.Streams.Stream_Element_Array (1 .. 512) := (others => 42);
       Last : Ada.Streams.Stream_Element_Offset;
    begin
       Address.Port := 50001;
       Address.Addr := Inet_Addr ("127.0.0.1");
       Create_Socket (Socket, Family_Inet, Socket_Datagram);
       Send_Socket (Socket, Data, Last, Address);
       Ada.Text_IO.Put_Line ("last :" & Last'Img);
    end UDP_Client;
    

    服务器:

    with Ada.Streams;
    with Ada.Text_IO;
    
    with GNAT.Sockets;
    
    procedure UDP_Server is
       use GNAT.Sockets;
       Server : Socket_Type;
       Address, From : Sock_Addr_Type;
       Data : Ada.Streams.Stream_Element_Array (1 .. 512);
       Last : Ada.Streams.Stream_Element_Offset;
       Watchdog : Natural := 0;
    begin
       Create_Socket (Server, Family_Inet, Socket_Datagram);
       Set_Socket_Option
         (Server,
          Socket_Level,
          (Reuse_Address, True));
       Set_Socket_Option
         (Server,
          Socket_Level,
          (Receive_Timeout,
           Timeout => 1.0));
       Address.Addr := Any_Inet_Addr;
       Address.Port := 50001;
       Bind_Socket (Server, Address);
       loop
          begin
             GNAT.Sockets.Receive_Socket (Server, Data, Last, From);
             Ada.Text_IO.Put_Line ("last : " & Last'Img);
             Ada.Text_IO.Put_Line ("from : " & Image (From.Addr));
          exception
             when Socket_Error =>
                Watchdog := Watchdog + 1;
                exit when Watchdog = 10;
          end;
       end loop;
    end UDP_Server;
    

    【讨论】:

    • GNAT.Sockets.Receive_Socket 像读取过程一样被阻塞。
    • 这是一个简单的 UDP 客户端/服务器示例:
    • 是的。在伯克利套接字行话中,“流”= TCP 和“数据报”= UDP。
    • 我会强调使用 GNAT 套接字以非阻塞方式通过 UDP 发送/接收的代码部分。 IE。 :Set_Socket_Option (My_Socket, Socket_Level, (Receive_Timeout, Timeout => 1.0));@eliot-b 提供的代码也指出,你需要捕获 Socket_Error 异常并处理它,因为达到超时时会引发此异常。如果要处理其他 socket_error 代码,则需要进一步编码。有关 Windows 中的错误代码,请参见例如 msdn.microsoft.com/en-us/library/windows/desktop/…
    猜你喜欢
    • 1970-01-01
    • 2011-10-30
    • 2019-01-25
    • 2015-08-06
    • 1970-01-01
    • 2014-11-17
    • 1970-01-01
    • 1970-01-01
    • 2010-10-11
    相关资源
    最近更新 更多