aboutsummaryrefslogblamecommitdiffstats
path: root/lib/crypto/src/crypto_server.erl
blob: 0b1e5c9b0298b010135ef3b0c777cc44bd52eae6 (plain) (tree)









































































































































                                                                                                                                                    
%%
%% %CopyrightBegin%
%% 
%% Copyright Ericsson AB 1999-2009. All Rights Reserved.
%% 
%% The contents of this file are subject to the Erlang Public License,
%% Version 1.1, (the "License"); you may not use this file except in
%% compliance with the License. You should have received a copy of the
%% Erlang Public License along with this software. If not, it can be
%% retrieved online at http://www.erlang.org/.
%% 
%% Software distributed under the License is distributed on an "AS IS"
%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
%% the License for the specific language governing rights and limitations
%% under the License.
%% 
%% %CopyrightEnd%
%%

%% Purpose: Provide cryptographic algorithms.

-module(crypto_server).

-behaviour(gen_server).

-export([start_link/0,client_port/0]).

%% Internal exports, call-back functions.
-export([init/1,handle_call/3,handle_cast/2,handle_info/2,code_change/3,
	 terminate/2]).

%% Measurements shows that inlining port_names/0 is worth doing.
-compile({inline,[{port_names,0}]}).

%%% --------------------------------------------------------
%%% Interface Functions. 
%%% --------------------------------------------------------

start_link() ->
    gen_server:start_link({local, crypto_server}, crypto_server, [], []).

init([]) ->
    process_flag(trap_exit, true),
    erl_ddll:start(),
    PrivDir = code:priv_dir(crypto),
    LibDir1 = filename:join([PrivDir, "lib"]),
    {Status, LibDir} =
	case erl_ddll:load_driver(LibDir1, crypto_drv) of
	    ok -> {ok,LibDir1};
	    {error,Error1} ->
		LibDir2 = 
		    filename:join(LibDir1, 
				  erlang:system_info(system_architecture)),
		Candidate =
		    filelib:wildcard(filename:join([LibDir2,"crypto_drv*"])),
		case Candidate of
		    [] ->
			{{error,Error1},LibDir1};
		    _ ->
			case erl_ddll:load_driver(LibDir2, crypto_drv) of
			    ok ->
				{ok,LibDir2};
			    {error, Error2} ->
				{{error,Error2},LibDir2}
			end
		end
	end,
    case Status of
	ok ->
	    Cmd = "crypto_drv elibcrypto " ++ 
		filename:join([LibDir, "elibcrypto"]),
	    open_ports(Cmd,size(port_names()));
	{error, E} ->
	    Str = erl_ddll:format_error(E),
	    error_logger:error_msg("Unable to load crypto_drv. Failed with error:~n\"~s\"~nOpenSSL might not be installed on this system.~n",[Str]),
	    {stop,nodriver}
    end.

open_ports(_,0) ->
    {ok, []};
open_ports(Cmd,N) ->   
    Port = open_port({spawn, Cmd}, []),
    %% check that driver is loaded, linked and working
    %% since crypto_drv links towards libcrypto, this is a good thing
    %% since libcrypto is known to be bad with backwards compatibility
    case catch port_control(Port, 0, []) of
	{'EXIT', _} ->
	    {stop, nodriver};
	_ ->
	    register(element(N,port_names()), Port),
	    open_ports(Cmd,N-1)
    end.

port_names() -> 
    { crypto_drv01, crypto_drv02, crypto_drv03, crypto_drv04,
      crypto_drv05, crypto_drv06, crypto_drv07, crypto_drv08,
      crypto_drv09, crypto_drv10, crypto_drv11, crypto_drv12,
      crypto_drv13, crypto_drv14, crypto_drv15, crypto_drv16 }.

client_port() ->
    element(erlang:system_info(scheduler_id) rem size(port_names()) + 1,
	    port_names()).


%%% --------------------------------------------------------
%%% The call-back functions.
%%% --------------------------------------------------------

handle_call(_, _, State) ->
    {noreply, State}.

handle_cast(_, State) ->
    {noreply, State}.

handle_info({'EXIT', Pid, _Reason}, State) when is_pid(Pid) ->
    {noreply, State};

handle_info({'EXIT', Port, Reason}, State) when is_port(Port) ->
    {stop, {port_died, Reason}, State};
handle_info(_, State) ->
    {noreply, State}.

code_change(_OldVsn, State, _Extra) ->
    {ok, State}.

terminate(_Reason, _State) ->
    close_ports(size(port_names())).

close_ports(0) ->
    ok;
close_ports(N) ->   
    element(N,port_names()) ! {self(), close},
    close_ports(N-1).