%%
%% %CopyrightBegin%
%%
%% Copyright Ericsson AB 1998-2017. 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(int_SUITE).
-include_lib("common_test/include/ct.hrl").
%% Test server specific exports
-export([all/0, suite/0,groups/0,init_per_suite/1, end_per_suite/1,
init_per_group/2,end_per_group/2]).
-export([init_per_testcase/2, end_per_testcase/2]).
%% Test cases
-export([interpret/1, guards/1, interpretable/1]).
-export([ append_1/1, append_2/1, member/1, reverse/1]).
init_per_testcase(interpretable, Config) ->
Config;
init_per_testcase(_Case, Config) ->
%% Interpret some existing and non-existing modules
DataDir = proplists:get_value(data_dir, Config),
{module, lists1} = int:i(filename:join([DataDir,lists1])),
{module, guards} = int:i(filename:join([DataDir,guards])),
Config.
end_per_testcase(interpretable, _Config) ->
ok;
end_per_testcase(_Case, Config) ->
%% Quit interpreting
ok = int:n(lists1),
ok = int:n(guards),
ok.
suite() ->
[{ct_hooks,[ts_install_cth]},
{timetrap,{minutes,1}}].
all() ->
[interpret, guards, {group, list_suite}, interpretable].
groups() ->
[{list_suite, [], [{group, append}, reverse, member]},
{append, [], [append_1, append_2]}].
init_per_suite(Config) ->
DataDir = proplists:get_value(data_dir, Config),
{ok,OldCwd} = file:get_cwd(),
try
ok = file:set_cwd(DataDir),
make:all()
after
file:set_cwd(OldCwd)
end,
Config.
end_per_suite(_Config) ->
ok.
init_per_group(_GroupName, Config) ->
Config.
end_per_group(_GroupName, Config) ->
Config.
%% Interpreting modules.
interpret(Config) when is_list(Config) ->
int:n(int:interpreted()),
%% Interpret some existing and non-existing modules
DataDir = proplists:get_value(data_dir, Config),
{module, lists1} = int:i(filename:join([DataDir,lists1])),
{module, ordsets1} = int:i(filename:join([DataDir,ordsets1])),
error = int:i(non_existent_module),
%% Check that the interpreter has the right view.
ExpectedResult = lists:sort([lists1, ordsets1]),
Result = int:interpreted(),
ExpectedResult = lists:sort(Result),
%% Uniterpret the modules.
ok = int:n(non_existent_module),
ok = int:n(lists1),
[ordsets1] = int:interpreted(),
ok = int:n("ordsets1"),
[] = int:interpreted(),
ok.
%% Evaluate guards.
guards(Config) when is_list(Config) ->
ok = guards:guards().
append_1(Config) when is_list(Config) ->
io:format("In append_1~n"),
io:format("code:which(lists1)=~p~n",
[code:which(lists1)]),
io:format("lists1:append([a],[b])=~p~n",
[spawn_eval(lists1,append,[[a],[b]])]),
"abcdef"=spawn_eval(lists1,append,[["abc","def"]]),
[hej, du,[glade, [bagare]]]=
spawn_eval(lists1,append,[[[hej], [du], [[glade, [bagare]]]]]),
[10, [elem]]=spawn_eval(lists1,append,[[[10], [[elem]]]]),
ok.
append_2(Config) when is_list(Config) ->
io:format("In append_2~n"),
io:format("code:which(lists1)=~p~n",
[code:which(lists1)]),
"abcdef"=spawn_eval(lists1,append,["abc", "def"]),
[hej, du]=spawn_eval(lists1,append,[[hej], [du]]),
[10, [elem]]=spawn_eval(lists1,append,[[10], [[elem]]]),
ok.
reverse(Config) when is_list(Config) ->
ok=reverse_test(0),
ok=reverse_test(1),
ok=reverse_test(2),
ok=reverse_test(537),
ok.
reverse_test(0) ->
case spawn_eval(lists1,reverse,[[]]) of
[] ->
ok;
_Other ->
error
end;
reverse_test(Num) ->
List=spawn_eval(lists1,reverse,
[['The Element'|lists1:duplicate(Num, 'Ele')]]),
case spawn_eval(lists1,reverse,[List]) of
['The Element'|_Rest] ->
ok;
_Other ->
error
end.
%% Tests the lists1:member() implementation. The function
%% is `non-blocking', and only processes 2000 elements
%% at a time.
%% This test case depends on lists1:reverse() to work,
%% which is tested in a separate test case.
member(Config) when list(Config) ->
ok=member_test(0),
ok=member_test(1),
ok=member_test(100),
ok=member_test(537),
ok.
member_test(0) ->
case spawn_eval(lists1,member,['The Element', []]) of
false ->
ok;
true ->
{error, 'Found (!?)'}
end;
member_test(Num) ->
List=spawn_eval(lists1,reverse,
[['The Element'|spawn_eval(lists1,duplicate,
[Num, 'Elem'])]]),
case spawn_eval(lists1,member,['The Element', List]) of
true ->
ok;
false ->
{error, not_found}
end.
spawn_eval(M,F,A) ->
Self = self(),
spawn(fun() -> evaluator(Self, M,F,A) end),
receive
Result ->
Result
end.
evaluator(Pid, M,F,A) ->
Pid ! (catch apply(M,F,A)).
%% Test int:interpretable/1.
interpretable(Config) when is_list(Config) ->
%% First make sure that 'lists1' is not loaded
case code:is_loaded(lists1) of
{file, _Loaded} ->
code:purge(lists1),
code:delete(lists1),
code:purge(lists1);
false -> ignore
end,
%% true
DataDir = filename:dirname(proplists:get_value(data_dir, Config)),
true = code:add_patha(DataDir),
true = int:interpretable(lists1),
true = int:interpretable(filename:join([DataDir,lists1])),
true = code:del_path(DataDir),
%% true (from source)
PrivDir = filename:join(proplists:get_value(priv_dir, Config), ""),
{ok, _} = file:copy(filename:join([DataDir,"lists1.beam"]),
filename:join([PrivDir,"lists1.beam"])),
true = code:add_patha(PrivDir),
true = int:interpretable(lists1),
ok = file:delete(filename:join([PrivDir,"lists1.beam"])),
%% {error, no_beam}
Src = filename:join([PrivDir,"lists1.erl"]),
{ok, _} = file:copy(filename:join([DataDir,"lists1.erl"]),
Src),
{error, no_beam} = int:interpretable(Src),
%% {error, no_debug_info}
{ok, _} = compile:file(Src, [{outdir,PrivDir}]),
{error, no_debug_info} = int:interpretable(Src),
{error, no_debug_info} = int:interpretable(lists1),
ok = file:delete(Src),
true = code:del_path(PrivDir),
%% {error, no_src}
A1 = erl_anno:new(1),
{ok, lists2, Binary} = compile:forms([{attribute,A1,module,lists2}], []),
code:load_binary(lists2, "unknown", Binary),
{error, no_src} = int:interpretable(lists2),
%% {error, badarg}
{error, badarg} = int:interpretable(pride),
{error, badarg} = int:interpretable("prejudice.erl"),
%% {error, {app,App}}
case filename:basename(code:lib_dir(kernel)) of
"kernel" ->
%% Development system (not installed). We are allowed
%% to interpret modules in kernel and stdlib
%% (at our own risk).
ok;
"kernel-" ++ _ ->
%% Installed system. Certain applications (including
%% kernel and stdlib) cannot be interpreted.
{error, {app,_}} = int:interpretable(file),
{error, {app,_}} = int:interpretable(lists),
case int:interpretable(dbg_ieval) of
{error, {app,_}} ->
ok;
{error, badarg} ->
case code:which(dbg_ieval) of
cover_compiled ->
ok;
Other1 ->
ct:fail({unexpected_result, Other1})
end;
Other2 ->
ct:fail({unexpected_result, Other2})
end
end,
ok.