%% %% %CopyrightBegin% %% %% Copyright Ericsson AB 1997-2016. 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(local_tcp). %% Socket server for TCP/IP -export([connect/3, connect/4, listen/2, accept/1, accept/2, close/1]). -export([send/2, send/3, recv/2, recv/3, unrecv/2]). -export([shutdown/2]). -export([controlling_process/2]). -export([fdopen/2]). -export([getserv/1, getaddr/1, getaddr/2, getaddrs/1, getaddrs/2]). -export([translate_ip/1]). -include("inet_int.hrl"). -define(FAMILY, local). -define(PROTO, tcp). -define(TYPE, stream). %% port lookup getserv(0) -> {ok, 0}. %% no address lookup getaddr({?FAMILY, _} = Address) -> {ok, Address}. getaddr({?FAMILY, _} = Address, _Timer) -> {ok, Address}. %% no address lookup getaddrs({?FAMILY, _} = Address) -> {ok, [Address]}. getaddrs({?FAMILY, _} = Address, _Timer) -> {ok, [Address]}. %% special this side addresses translate_ip(IP) -> IP. %% %% Send data on a socket %% send(Socket, Packet, Opts) -> prim_inet:send(Socket, Packet, Opts). send(Socket, Packet) -> prim_inet:send(Socket, Packet, []). %% %% Receive data from a socket (inactive only) %% recv(Socket, Length) -> prim_inet:recv(Socket, Length). recv(Socket, Length, Timeout) -> prim_inet:recv(Socket, Length, Timeout). unrecv(Socket, Data) -> prim_inet:unrecv(Socket, Data). %% %% Shutdown one end of a socket %% shutdown(Socket, How) -> prim_inet:shutdown(Socket, How). %% %% Close a socket (async) %% close(Socket) -> inet:tcp_close(Socket). %% %% Set controlling process %% FIXME: move messages to new owner!!! %% controlling_process(Socket, NewOwner) -> inet:tcp_controlling_process(Socket, NewOwner). %% %% Connect %% connect(Address, Port, Opts) -> do_connect(Address, Port, Opts, infinity). %% connect(Address, Port, Opts, infinity) -> do_connect(Address, Port, Opts, infinity); connect(Address, Port, Opts, Timeout) when is_integer(Timeout), Timeout >= 0 -> do_connect(Address, Port, Opts, Timeout). do_connect(Addr = {?FAMILY, _}, 0, Opts, Time) -> case inet:connect_options(Opts, ?MODULE) of {error, Reason} -> exit(Reason); {ok, #connect_opts{ fd = Fd, ifaddr = BAddr, port = 0, opts = SockOpts}} when tuple_size(BAddr) =:= 2, element(1, BAddr) =:= ?FAMILY; BAddr =:= any -> case inet:open( Fd, case BAddr of any -> undefined; _ -> BAddr end, 0, SockOpts, ?PROTO, ?FAMILY, ?TYPE, ?MODULE) of {ok, S} -> case prim_inet:connect(S, Addr, 0, Time) of ok -> {ok,S}; Error -> prim_inet:close(S), Error end; Error -> Error end; {ok, _} -> exit(badarg) end. %% %% Listen %% listen(0, Opts) -> case inet:listen_options([{port,0} | Opts], ?MODULE) of {error, Reason} -> exit(Reason); {ok, #listen_opts{ fd = Fd, ifaddr = BAddr, port = 0, opts = SockOpts} = R} when tuple_size(BAddr) =:= 2, element(1, BAddr) =:= ?FAMILY; BAddr =:= any -> case inet:open( Fd, BAddr, 0, SockOpts, ?PROTO, ?FAMILY, ?TYPE, ?MODULE) of {ok, S} -> case prim_inet:listen(S, R#listen_opts.backlog) of ok -> {ok, S}; Error -> prim_inet:close(S), Error end; Error -> Error end; {ok, _} -> exit(badarg) end. %% %% Accept %% accept(L) -> case prim_inet:accept(L) of {ok, S} -> inet_db:register_socket(S, ?MODULE), {ok,S}; Error -> Error end. %% accept(L, Timeout) -> case prim_inet:accept(L, Timeout) of {ok, S} -> inet_db:register_socket(S, ?MODULE), {ok,S}; Error -> Error end. %% %% Create a port/socket from a file descriptor %% fdopen(Fd, Opts) -> inet:open(Fd, undefined, 0, Opts, ?PROTO, ?FAMILY, ?TYPE, ?MODULE).