diff options
| -rw-r--r-- | erts/emulator/beam/erl_bif_info.c | 10 | ||||
| -rw-r--r-- | erts/emulator/beam/erl_db_hash.c | 12 | ||||
| -rw-r--r-- | erts/emulator/beam/erl_db_tree.c | 14 | ||||
| -rw-r--r-- | lib/stdlib/test/ets_SUITE.erl | 57 | 
4 files changed, 69 insertions, 24 deletions
| diff --git a/erts/emulator/beam/erl_bif_info.c b/erts/emulator/beam/erl_bif_info.c index 74708b2caa..8c51bdb630 100644 --- a/erts/emulator/beam/erl_bif_info.c +++ b/erts/emulator/beam/erl_bif_info.c @@ -4689,6 +4689,16 @@ BIF_RETTYPE erts_debug_set_internal_state_2(BIF_ALIST_2)                      BIF_RET(am_ok);              }          } +        else if (ERTS_IS_ATOM_STR("mbuf", BIF_ARG_1)) { +            Uint sz = size_object(BIF_ARG_2); +            ErlHeapFragment* frag = new_message_buffer(sz); +            Eterm *hp = frag->mem; +            Eterm copy = copy_struct(BIF_ARG_2, sz, &hp, &frag->off_heap); +            frag->next = BIF_P->mbuf; +            BIF_P->mbuf = frag; +            BIF_P->mbuf_sz += sz; +            BIF_RET(copy); +        }      }      BIF_ERROR(BIF_P, BADARG); diff --git a/erts/emulator/beam/erl_db_hash.c b/erts/emulator/beam/erl_db_hash.c index f05a3b51c9..426c7d2d48 100644 --- a/erts/emulator/beam/erl_db_hash.c +++ b/erts/emulator/beam/erl_db_hash.c @@ -1323,11 +1323,7 @@ static int match_traverse(Process* p, DbTableHash* tb,                  unlock_hash_function(lck);                  break;              } -            if (iterations_left <= 0 || MBUF(p)) { -                /* -                 * We have either reached our limit, or just created some heap fragments. -                 * Since many heap fragments will make the GC slower, trap and GC now. -                 */ +            if (iterations_left <= 0) {                  unlock_hash_function(lck);                  ret_value = ctx->on_trap(ctx, slot_ix, got, &mpi.mp, ret);                  goto done; @@ -1433,11 +1429,7 @@ static int match_traverse_continue(Process* p, DbTableHash* tb,                  unlock_hash_function(lck);                  break;              } -            if (iterations_left <= 0 || MBUF(p)) { -                /* -                 * We have either reached our limit, or just created some heap fragments. -                 * Since many heap fragments will make the GC slower, trap and GC now. -                 */ +            if (iterations_left <= 0) {                  unlock_hash_function(lck);                  ret_value = ctx->on_trap(ctx, slot_ix, got, mpp, ret);                  goto done; diff --git a/erts/emulator/beam/erl_db_tree.c b/erts/emulator/beam/erl_db_tree.c index 02a5934a6e..fe57348700 100644 --- a/erts/emulator/beam/erl_db_tree.c +++ b/erts/emulator/beam/erl_db_tree.c @@ -3743,13 +3743,6 @@ static int doit_select(DbTableCommon *tb, TreeDbTerm *this,      if (is_value(ret)) {  	sc->accum = CONS(hp, ret, sc->accum);      } -    if (MBUF(sc->p)) { -	/* -	 * Force a trap and GC if a heap fragment was created. Many heap fragments -	 * make the GC slow. -	 */ -	sc->max = 0; -    }      if (--(sc->max) <= 0) {  	return 0;      } @@ -3806,13 +3799,6 @@ static int doit_select_chunk(DbTableCommon *tb, TreeDbTerm *this,  	++(sc->got);  	sc->accum = CONS(hp, ret, sc->accum);      } -    if (MBUF(sc->p)) { -	/* -	 * Force a trap and GC if a heap fragment was created. Many heap fragments -	 * make the GC slow. -	 */ -	sc->max = 0; -    }      if (--(sc->max) <= 0 || sc->got == sc->chunk_size) {  	return 0;      } diff --git a/lib/stdlib/test/ets_SUITE.erl b/lib/stdlib/test/ets_SUITE.erl index 22c77aa172..7703198c4c 100644 --- a/lib/stdlib/test/ets_SUITE.erl +++ b/lib/stdlib/test/ets_SUITE.erl @@ -59,6 +59,7 @@  -export([otp_5340/1]).  -export([otp_6338/1]).  -export([otp_6842_select_1000/1]). +-export([select_mbuf_trapping/1]).  -export([otp_7665/1]).  -export([meta_wb/1]).  -export([grow_shrink/1, grow_pseudo_deleted/1, shrink_pseudo_deleted/1]). @@ -132,6 +133,7 @@ all() ->       t_named_select,       select_fail, t_insert_new, t_repair_continuation,       otp_5340, otp_6338, otp_6842_select_1000, otp_7665, +     select_mbuf_trapping,       otp_8732, meta_wb, grow_shrink, grow_pseudo_deleted,       shrink_pseudo_deleted, {group, meta_smp}, smp_insert,       smp_fixed_delete, smp_unfix_fix, smp_select_replace, @@ -5292,6 +5294,61 @@ otp_6338(Config) when is_list(Config) ->        end),      ok. +%% OTP-15660: Verify select not doing excessive trapping +%%            when process have mbuf heap fragments. +select_mbuf_trapping(Config) when is_list(Config) -> +    select_mbuf_trapping_do(set), +    select_mbuf_trapping_do(ordered_set). + +select_mbuf_trapping_do(Type) -> +    T = ets:new(xxx, [Type]), +    NKeys = 50, +    [ets:insert(T, {K, value}) || K <- lists:seq(1,NKeys)], + +    {priority, Prio} = process_info(self(), priority), +    Tracee = self(), +    [SchedTracer] +	= start_loopers(1, Prio, +			fun (SC) -> +				receive +				    {trace, Tracee, out, _} -> +					SC+1; +				    done -> +					Tracee ! {schedule_count, SC}, +                                        exit(normal) +				end +			end, +			0), + +    erlang:garbage_collect(), +    1 = erlang:trace(self(), true, [running,{tracer,SchedTracer}]), + +    %% Artificially create an mbuf heap fragment +    MbufTerm = "Frag me up", +    MbufTerm = erts_debug:set_internal_state(mbuf, MbufTerm), + +    Keys = ets:select(T, [{{'$1', value}, [], ['$1']}]), +    NKeys = length(Keys), + +    1 = erlang:trace(self(), false, [running]), +    Ref = erlang:trace_delivered(Tracee), +    receive +        {trace_delivered, Tracee, Ref} -> +            SchedTracer ! done +    end, +    receive +	{schedule_count, N} -> +	    io:format("~p context switches: ~p", [Type,N]), +	    if +		N < 3 -> ok; +		true -> ct:fail(failed) +	    end +    end, +    true = ets:delete(T), +    ok. + + +  %% Elements could come in the wrong order in a bag if a rehash occurred.  otp_5340(Config) when is_list(Config) ->      repeat_for_opts(fun otp_5340_do/1). | 
