%% %% %CopyrightBegin% %% %% Copyright Ericsson AB 1997-2020. All Rights Reserved. %% %% Licensed under the Apache License, Version 2.0 (the "License"); %% you may not use this file except in compliance with the License. %% You may obtain a copy of the License at %% %% http://www.apache.org/licenses/LICENSE-2.0 %% %% Unless required by applicable law or agreed to in writing, software %% distributed under the License is distributed on an "AS IS" BASIS, %% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %% See the License for the specific language governing permissions and %% limitations under the License. %% %% %CopyrightEnd% %% -module(inet6_udp). -export([open/1, open/2, close/1]). -export([send/2, send/4, recv/2, recv/3, connect/3]). -export([controlling_process/2]). -export([fdopen/2]). -export([getserv/1, getaddr/1, getaddr/2, translate_ip/1]). -include("inet_int.hrl"). -define(FAMILY, inet6). -define(PROTO, udp). -define(TYPE, dgram). %% inet_udp port lookup getserv(Port) when is_integer(Port) -> {ok, Port}; getserv(Name) when is_atom(Name) -> inet:getservbyname(Name, ?PROTO). %% inet_udp address lookup getaddr(Address) -> inet:getaddr(Address, ?FAMILY). getaddr(Address, Timer) -> inet:getaddr(Address, ?FAMILY, Timer). %% inet_udp special this side addresses translate_ip(IP) -> inet:translate_ip(IP, ?FAMILY). -spec open(_) -> {ok, inet:socket()} | {error, atom()}. open(Port) -> open(Port, []). -spec open(_, _) -> {ok, inet:socket()} | {error, atom()}. open(Port, Opts) -> case inet:udp_options( [{port,Port} | Opts], ?MODULE) of {error, Reason} -> exit(Reason); {ok, #udp_opts{ fd = Fd, ifaddr = BAddr = {A,B,C,D,E,F,G,H}, port = BPort, opts = SockOpts}} when ?ip6(A,B,C,D,E,F,G,H), ?port(BPort) -> inet:open( Fd, BAddr, BPort, SockOpts, ?PROTO, ?FAMILY, ?TYPE, ?MODULE); {ok, _} -> exit(badarg) end. send(S, {A,B,C,D,E,F,G,H} = IP, Port, Data) when ?ip6(A,B,C,D,E,F,G,H), ?port(Port) -> prim_inet:sendto(S, {IP, Port}, [], Data); send(S, {{A,B,C,D,E,F,G,H}, Port} = Addr, AncData, Data) when ?ip6(A,B,C,D,E,F,G,H), ?port(Port), is_list(AncData) -> prim_inet:sendto(S, Addr, AncData, Data); send(S, {?FAMILY, {{A,B,C,D,E,F,G,H}, Port}} = Address, AncData, Data) when ?ip6(A,B,C,D,E,F,G,H), ?port(Port), is_list(AncData) -> prim_inet:sendto(S, Address, AncData, Data); send(S, {?FAMILY, {loopback, Port}} = Address, AncData, Data) when ?port(Port), is_list(AncData) -> prim_inet:sendto(S, Address, AncData, Data). send(S, Data) -> prim_inet:sendto(S, {any, 0}, [], Data). connect(S, Addr = {A,B,C,D,E,F,G,H}, Port) when ?ip6(A,B,C,D,E,F,G,H), ?port(Port) -> prim_inet:connect(S, Addr, Port). recv(S, Len) -> prim_inet:recvfrom(S, Len). recv(S, Len, Time) -> prim_inet:recvfrom(S, Len, Time). -spec close(inet:socket()) -> ok. close(S) -> inet:udp_close(S). %% %% Set controlling process: %% 1) First sync socket into a known state %% 2) Move all messages onto the new owners message queue %% 3) Commit the owner %% 4) Wait for ack of new Owner (since socket does some link and unlink) %% controlling_process(Socket, NewOwner) -> inet:udp_controlling_process(Socket, NewOwner). %% %% Create a port/socket from a file descriptor %% fdopen(Fd, Opts) -> inet:fdopen(Fd, Opts, ?PROTO, ?FAMILY, ?TYPE, ?MODULE).