123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186 |
- %% -*- erlang-indent-level: 4; indent-tabs-mode: nil -*-
- %% --------------------------------------------------
- %% This file is provided to you 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.
- %% --------------------------------------------------
- %%
- %% @author Ulf Wiger <ulf@wiger.net>
- %%
- %% @doc Extended process registry
- %% This module implements an extended process registry
- %%
- %% For a detailed description, see
- %% <a href="erlang07-wiger.pdf">erlang07-wiger.pdf</a>.
- %%
- %% <b>NOTE:</b> The functions in the Gproc API expect the Gproc application
- %% to be running.
- %%
- %% <h2>Tuning Gproc performance</h2>
- %%
- %% Gproc relies on a central server and an ordered-set ets table.
- %% Effort is made to perform as much work as possible in the client without
- %% sacrificing consistency. A few things can be tuned by setting the following
- %% application environment variables in the top application of `gproc'
- %% (usually `gproc'):
- %%
- %% * `{ets_options, list()}' - Currently, the options `{write_concurrency, F}'
- %% and `{read_concurrency, F}' are allowed. The default is
- %% `[{write_concurrency, true}, {read_concurrency, true}]'
- %% * `{server_options, list()}' - These will be passed as spawn options when
- %% starting the `gproc' and `gproc_dist' servers. Default is `[]'. It is
- %% likely that `{priority, high | max}' and/or increasing `min_heap_size'
- %% will improve performance.
- %%
- %% @end
- -module(gproc).
- -behaviour(gen_server).
- -export([start_link/0,
- reg/1, reg/2, reg/3, unreg/1, set_attributes/2,
- reg_other/2, reg_other/3, reg_other/4, unreg_other/2,
- reg_or_locate/1, reg_or_locate/2, reg_or_locate/3,
- reg_shared/1, reg_shared/2, reg_shared/3, unreg_shared/1,
- set_attributes_shared/2, set_value_shared/2,
- ensure_reg/1, ensure_reg/2, ensure_reg/3,
- ensure_reg_other/2, ensure_reg_other/3, ensure_reg_other/4,
- mreg/3,
- munreg/3,
- set_value/2,
- get_value/1, get_value/2, get_value_shared/1,
- get_attribute/2, get_attribute/3, get_attribute_shared/2,
- get_attributes/1, get_attributes/2,
- where/1,
- await/1, await/2, await/3,
- wide_await/3,
- nb_wait/1, nb_wait/2,
- cancel_wait/2, cancel_wait/3,
- cancel_wait_or_monitor/1,
- monitor/1, monitor/2,
- demonitor/2,
- lookup_pid/1,
- lookup_pids/1,
- lookup_value/1,
- lookup_values/1,
- update_counter/2, update_counter/3,
- update_counters/2,
- reset_counter/1,
- update_shared_counter/2,
- give_away/2,
- goodbye/0,
- send/2,
- bcast/2, bcast/3,
- info/1, info/2,
- i/0,
- select/1, select/2, select/3,
- select_count/1, select_count/2,
- first/1,
- next/2,
- prev/2,
- last/1,
- table/0, table/1, table/2]).
- %% Environment handling
- -export([get_env/3, get_env/4,
- get_set_env/3, get_set_env/4,
- set_env/5]).
- %% Convenience functions
- -export([add_local_name/1,
- add_global_name/1,
- add_local_property/2,
- add_global_property/2,
- add_local_counter/2,
- add_global_counter/2,
- add_local_aggr_counter/1,
- add_global_aggr_counter/1,
- add_shared_local_counter/2,
- lookup_local_name/1,
- lookup_global_name/1,
- lookup_local_properties/1,
- lookup_global_properties/1,
- lookup_local_counters/1,
- lookup_global_counters/1,
- lookup_local_resources/1,
- lookup_global_resources/1,
- lookup_local_aggr_counter/1,
- lookup_global_aggr_counter/1]).
- %% Callbacks for behaviour support
- -export([whereis_name/1,
- register_name/2,
- unregister_name/1]).
- -export([default/1]).
- %%% internal exports
- -export([init/1,
- handle_cast/2,
- handle_call/3,
- handle_info/2,
- code_change/3,
- terminate/2]).
- %% this shouldn't be necessary
- -export([audit_process/1]).
- -include("gproc_int.hrl").
- -include("gproc.hrl").
- -export_type([scope/0, type/0, key/0,
- context/0, sel_pattern/0, sel_scope/0, sel_context/0,
- reg_id/0, unique_id/0, monitor_type/0]).
- -type type() :: n | p | c | a | r | rc.
- -type scope() :: l | g.
- -type context() :: {scope(),type()} | type().
- -type sel_type() :: type()
- | names | props | counters | aggr_counters
- | resources | resource_counters.
- -type sel_var() :: '_' | atom().
- -type keypat() :: {sel_type() | sel_var(), l | g | sel_var(), any()}.
- -type pidpat() :: pid() | sel_var().
- -type headpat() :: {keypat(), pidpat(), any()}.
- -type key() :: {type(), scope(), any()}.
- -type value() :: any().
- -type attr() :: {atom(), any()}.
- -type attrs() :: [attr()].
- -type sel_pattern() :: [{headpat(), list(), list()}].
- -type reg_id() :: {type(), scope(), any()}.
- -type unique_id() :: {n | a, scope(), any()}.
- -type monitor_type() :: info | standby | follow.
- -type sel_scope() :: scope | all | global | local.
- -type sel_context() :: {scope(), type()} | type().
- %% update_counter increment
- -type ctr_incr() :: integer().
- -type ctr_thr() :: integer().
- -type ctr_setval() :: integer().
- -type ctr_update() :: ctr_incr()
- | {ctr_incr(), ctr_thr(), ctr_setval()}.
- -type increment() :: ctr_incr() | ctr_update() | [ctr_update()].
- -define(SERVER, ?MODULE).
- %%-define(l, l(?LINE)). % when activated, calls a traceable empty function
- -define(l, ignore).
- -define(CHK_DIST,
- case whereis(gproc_dist) of
- undefined ->
- ?THROW_GPROC_ERROR(local_only);
- _ ->
- ok
- end).
- -define(PID_IS_DEAD(Pid),
- (node(Pid) == node() andalso is_process_alive(Pid) == false)).
- -record(state, {}).
- %% @spec () -> {ok, pid()}
- %%
- %% @doc Starts the gproc server.
- %%
- %% This function is intended to be called from gproc_sup, as part of
- %% starting the gproc application.
- %% @end
- start_link() ->
- _ = create_tabs(),
- SpawnOpts = gproc_lib:valid_opts(server_options, []),
- gen_server:start_link({local, ?SERVER}, ?MODULE, [],
- [{spawn_opt, SpawnOpts}]).
- %% spec(Name::any()) -> true
- %%
- %% @doc Registers a local (unique) name. @equiv reg({n,l,Name})
- %% @end
- %%
- add_local_name(Name) ->
- ?CATCH_GPROC_ERROR(reg1({n,l,Name}, undefined, [], reg), [Name]).
- %% spec(Name::any()) -> true
- %%
- %% @doc Registers a global (unique) name. @equiv reg({n,g,Name})
- %% @end
- %%
- add_global_name(Name) ->
- ?CATCH_GPROC_ERROR(reg1({n,g,Name}, undefined, [], reg), [Name]).
- %% spec(Name::any(), Value::any()) -> true
- %%
- %% @doc Registers a local (non-unique) property. @equiv reg({p,l,Name},Value)
- %% @end
- %%
- add_local_property(Name , Value) ->
- ?CATCH_GPROC_ERROR(reg1({p,l,Name}, Value, [], reg), [Name, Value]).
- %% spec(Name::any(), Value::any()) -> true
- %%
- %% @doc Registers a global (non-unique) property. @equiv reg({p,g,Name},Value)
- %% @end
- %%
- add_global_property(Name, Value) ->
- ?CATCH_GPROC_ERROR(reg1({p,g,Name}, Value, [], reg), [Name, Value]).
- %% spec(Name::any(), Initial::integer()) -> true
- %%
- %% @doc Registers a local (non-unique) counter. @equiv reg({c,l,Name},Value)
- %% @end
- %%
- add_local_counter(Name, Initial) when is_integer(Initial) ->
- ?CATCH_GPROC_ERROR(reg1({c,l,Name}, Initial, [], reg), [Name, Initial]).
- %% spec(Name::any(), Initial::integer()) -> true
- %%
- %% @doc Registers a local shared (unique) counter.
- %% @equiv reg_shared({c,l,Name},Value)
- %% @end
- %%
- add_shared_local_counter(Name, Initial) when is_integer(Initial) ->
- reg_shared({c,l,Name}, Initial).
- %% spec(Name::any(), Initial::integer()) -> true
- %%
- %% @doc Registers a global (non-unique) counter. @equiv reg({c,g,Name},Value)
- %% @end
- %%
- add_global_counter(Name, Initial) when is_integer(Initial) ->
- ?CATCH_GPROC_ERROR(reg1({c,g,Name}, Initial, [], reg), [Name, Initial]).
- %% spec(Name::any()) -> true
- %%
- %% @doc Registers a local (unique) aggregated counter.
- %% @equiv reg({a,l,Name})
- %% @end
- %%
- add_local_aggr_counter(Name) -> ?CATCH_GPROC_ERROR(reg1({a,l,Name}), [Name]).
- %% spec(Name::any()) -> true
- %%
- %% @doc Registers a global (unique) aggregated counter.
- %% @equiv reg({a,g,Name})
- %% @end
- %%
- add_global_aggr_counter(Name) ->
- ?CATCH_GPROC_ERROR(reg1({a,g,Name}), [Name]).
- %% @spec (Name::any()) -> pid()
- %%
- %% @doc Lookup a local unique name. Fails if there is no such name.
- %% @equiv where({n,l,Name})
- %% @end
- %%
- lookup_local_name(Name) -> where({n,l,Name}).
- %% @spec (Name::any()) -> pid()
- %%
- %% @doc Lookup a global unique name. Fails if there is no such name.
- %% @equiv where({n,g,Name})
- %% @end
- %%
- lookup_global_name(Name) -> where({n,g,Name}).
- %% @spec (Name::any()) -> integer()
- %%
- %% @doc Lookup a local (unique) aggregated counter and returns its value.
- %% Fails if there is no such object.
- %% @equiv where({a,l,Name})
- %% @end
- %%
- lookup_local_aggr_counter(Name) -> lookup_value({a,l,Name}).
- %% @spec (Name::any()) -> integer()
- %%
- %% @doc Lookup a global (unique) aggregated counter and returns its value.
- %% Fails if there is no such object.
- %% @equiv lookup_value({a,g,Name})
- %% @end
- %%
- lookup_global_aggr_counter(Name) -> lookup_value({a,g,Name}).
- %% @spec (Property::any()) -> [{pid(), Value}]
- %%
- %% @doc Look up all local (non-unique) instances of a given Property.
- %% Returns a list of {Pid, Value} tuples for all matching objects.
- %% @equiv lookup_values({p, l, Property})
- %% @end
- %%
- lookup_local_properties(P) -> lookup_values({p,l,P}).
- %% @spec (Property::any()) -> [{pid(), Value}]
- %%
- %% @doc Look up all global (non-unique) instances of a given Property.
- %% Returns a list of {Pid, Value} tuples for all matching objects.
- %% @equiv lookup_values({p, g, Property})
- %% @end
- %%
- lookup_global_properties(P) -> lookup_values({p,g,P}).
- %% @spec (Counter::any()) -> [{pid(), Value::integer()}]
- %%
- %% @doc Look up all local (non-unique) instances of a given Counter.
- %% Returns a list of {Pid, Value} tuples for all matching objects.
- %% @equiv lookup_values({c, l, Counter})
- %% @end
- %%
- lookup_local_counters(P) -> lookup_values({c,l,P}).
- %% @spec (Counter::any()) -> [{pid(), Value::integer()}]
- %%
- %% @doc Look up all global (non-unique) instances of a given Counter.
- %% Returns a list of {Pid, Value} tuples for all matching objects.
- %% @equiv lookup_values({c, g, Counter})
- %% @end
- %%
- lookup_global_counters(P) -> lookup_values({c,g,P}).
- %% @spec (Resource::any()) -> [{pid(), Value::integer()}]
- %%
- %% @doc Look up all local (non-unique) instances of a given Resource.
- %% Returns a list of {Pid, Value} tuples for all matching objects.
- %% @equiv lookup_values({r, l, Resource})
- %% @end
- %%
- lookup_local_resources(P) -> lookup_values({r,l,P}).
- %% @spec (Resource::any()) -> [{pid(), Value::integer()}]
- %%
- %% @doc Look up all global (non-unique) instances of a given Resource.
- %% Returns a list of {Pid, Value} tuples for all matching objects.
- %% @equiv lookup_values({r, g, Resource})
- %% @end
- %%
- lookup_global_resources(P) -> lookup_values({r,g,P}).
- %% @spec get_env(Scope::scope(), App::atom(), Key::atom()) -> term()
- %% @equiv get_env(Scope, App, Key, [app_env])
- get_env(Scope, App, Key) ->
- get_env(Scope, App, Key, [app_env]).
- %% @spec (Scope::scope(), App::atom(), Key::atom(), Strategy) -> term()
- %% Strategy = [Alternative]
- %% Alternative = app_env
- %% | os_env
- %% | inherit | {inherit, pid()} | {inherit, unique_id()}
- %% | init_arg
- %% | {mnesia, ActivityType, Oid, Pos}
- %% | {default, term()}
- %% | error
- %% @doc Read an environment value, potentially cached as a `gproc_env' property.
- %%
- %% This function first tries to read the value of a cached property,
- %% `{p, Scope, {gproc_env, App, Key}}'. If this fails, it will try the provided
- %% alternative strategy. `Strategy' is a list of alternatives, tried in order.
- %% Each alternative can be one of:
- %%
- %% * `app_env' - try `application:get_env(App, Key)'
- %% * `os_env' - try `os:getenv(ENV)', where `ENV' is `Key' converted into an
- %% uppercase string
- %% * `{os_env, ENV}' - try `os:getenv(ENV)'
- %% * `inherit' - inherit the cached value, if any, held by the parent process.
- %% * `{inherit, Pid}' - inherit the cached value, if any, held by `Pid'.
- %% * `{inherit, Id}' - inherit the cached value, if any, held by the process
- %% registered in `gproc' as `Id'.
- %% * `init_arg' - try `init:get_argument(Key)'; expects a single value, if any.
- %% * `{mnesia, ActivityType, Oid, Pos}' - try
- %% `mnesia:activity(ActivityType, fun() -> mnesia:read(Oid) end)'; retrieve
- %% the value in position `Pos' if object found.
- %% * `{default, Value}' - set a default value to return once alternatives have
- %% been exhausted; if not set, `undefined' will be returned.
- %% * `error' - raise an exception, `erlang:error(gproc_env, [App, Key, Scope])'.
- %%
- %% While any alternative can occur more than once, the only one that might make
- %% sense to use multiple times is `{default, Value}'.
- %%
- %% The return value will be one of:
- %%
- %% * The value of the first matching alternative, or `error' eception,
- %% whichever comes first
- %% * The last instance of `{default, Value}', or `undefined', if there is no
- %% matching alternative, default or `error' entry in the list.
- %%
- %% The `error' option can be used to assert that a value has been previously
- %% cached. Alternatively, it can be used to assert that a value is either cached
- %% or at least defined somewhere,
- %% e.g. `get_env(l, mnesia, dir, [app_env, error])'.
- %% @end
- get_env(Scope, App, Key, Strategy)
- when Scope==l, is_atom(App), is_atom(Key);
- Scope==g, is_atom(App), is_atom(Key) ->
- do_get_env(Scope, App, Key, Strategy, false).
- %% @spec get_set_env(Scope::scope(), App::atom(), Key::atom()) -> term()
- %% @equiv get_set_env(Scope, App, Key, [app_env])
- get_set_env(Scope, App, Key) ->
- get_set_env(Scope, App, Key, [app_env]).
- %% @spec get_set_env(Scope::scope(), App::atom(), Key::atom(), Strategy) ->
- %% Value
- %% @doc Fetch and cache an environment value, if not already cached.
- %%
- %% This function does the same thing as {@link get_env/4}, but also updates the
- %% cache. Note that the cache will be updated even if the result of the lookup
- %% is `undefined'.
- %%
- %% @see get_env/4.
- %% @end
- %%
- get_set_env(Scope, App, Key, Strategy)
- when Scope==l, is_atom(App), is_atom(Key);
- Scope==g, is_atom(App), is_atom(Key) ->
- do_get_env(Scope, App, Key, Strategy, true).
- do_get_env(Context, App, Key, Alternatives, Set) ->
- case lookup_env(Context, App, Key, self()) of
- undefined ->
- check_alternatives(Alternatives, Context, App, Key, undefined, Set);
- {ok, Value} ->
- Value
- end.
- %% @spec set_env(Scope::scope(), App::atom(),
- %% Key::atom(), Value::term(), Strategy) -> Value
- %% Strategy = [Alternative]
- %% Alternative = app_env | os_env | {os_env, VAR}
- %% | {mnesia, ActivityType, Oid, Pos}
- %%
- %% @doc Updates the cached value as well as underlying environment.
- %%
- %% This function should be exercised with caution, as it affects the larger
- %% environment outside gproc. This function modifies the cached value, and then
- %% proceeds to update the underlying environment (OS environment variable or
- %% application environment variable).
- %%
- %% When the `mnesia' alternative is used, gproc will try to update any existing
- %% object, changing only the `Pos' position. If no such object exists, it will
- %% create a new object, setting any other attributes (except `Pos' and the key)
- %% to `undefined'.
- %% @end
- %%
- set_env(Scope, App, Key, Value, Strategy)
- when Scope==l, is_atom(App), is_atom(Key);
- Scope==g, is_atom(App), is_atom(Key) ->
- case is_valid_set_strategy(Strategy, Value) of
- true ->
- update_cached_env(Scope, App, Key, Value),
- set_strategy(Strategy, App, Key, Value);
- false ->
- erlang:error(badarg)
- end.
- check_alternatives([{default, Val}|Alts], Scope, App, Key, _, Set) ->
- check_alternatives(Alts, Scope, App, Key, Val, Set);
- check_alternatives([H|T], Scope, App, Key, Def, Set) ->
- case try_alternative(H, App, Key, Scope) of
- undefined ->
- check_alternatives(T, Scope, App, Key, Def, Set);
- {ok, Value} ->
- if Set ->
- cache_env(Scope, App, Key, Value),
- Value;
- true ->
- Value
- end
- end;
- check_alternatives([], Scope, App, Key, Def, Set) ->
- if Set ->
- cache_env(Scope, App, Key, Def);
- true ->
- ok
- end,
- Def.
- try_alternative(error, App, Key, Scope) ->
- erlang:error(gproc_env, [App, Key, Scope]);
- try_alternative(inherit, App, Key, Scope) ->
- case get('$ancestors') of
- [P|_] ->
- lookup_env(Scope, App, Key, P);
- _ ->
- undefined
- end;
- try_alternative({inherit, P}, App, Key, Scope) when is_pid(P) ->
- lookup_env(Scope, App, Key, P);
- try_alternative({inherit, P}, App, Key, Scope) ->
- case where(P) of
- undefined -> undefined;
- Pid when is_pid(Pid) ->
- lookup_env(Scope, App, Key, Pid)
- end;
- try_alternative(app_env, App, Key, _Scope) ->
- case application:get_env(App, Key) of
- undefined -> undefined;
- {ok, undefined} -> undefined;
- {ok, Value} -> {ok, Value}
- end;
- try_alternative(os_env, _App, Key, _) ->
- case os:getenv(os_env_key(Key)) of
- false -> undefined;
- Val -> {ok, Val}
- end;
- try_alternative({os_env, Key}, _, _, _) ->
- case os:getenv(Key) of
- false -> undefined;
- Val -> {ok, Val}
- end;
- try_alternative(init_arg, _, Key, _) ->
- case init:get_argument(Key) of
- {ok, [[Value]]} ->
- {ok, Value};
- error ->
- undefined
- end;
- try_alternative({mnesia,Type,Key,Pos}, _, _, _) ->
- case mnesia:activity(Type, fun() -> mnesia:read(Key) end) of
- [] -> undefined;
- [Found] ->
- {ok, element(Pos, Found)}
- end.
- os_env_key(Key) ->
- string:to_upper(atom_to_list(Key)).
- lookup_env(Scope, App, Key, P) ->
- case ets:lookup(?TAB, {{p, Scope, {gproc_env, App, Key}}, P}) of
- [] ->
- undefined;
- [{_, _, Value}] ->
- {ok, Value}
- end.
- cache_env(Scope, App, Key, Value) ->
- ?CATCH_GPROC_ERROR(
- reg1({p, Scope, {gproc_env, App, Key}}, Value, [], reg),
- [Scope,App,Key,Value]).
- update_cached_env(Scope, App, Key, Value) ->
- case lookup_env(Scope, App, Key, self()) of
- undefined ->
- cache_env(Scope, App, Key, Value);
- {ok, _} ->
- set_value({p, Scope, {gproc_env, App, Key}}, Value)
- end.
- is_valid_set_strategy([os_env|T], Value) ->
- is_string(Value) andalso is_valid_set_strategy(T, Value);
- is_valid_set_strategy([{os_env, _}|T], Value) ->
- is_string(Value) andalso is_valid_set_strategy(T, Value);
- is_valid_set_strategy([app_env|T], Value) ->
- is_valid_set_strategy(T, Value);
- is_valid_set_strategy([{mnesia,_Type,_Oid,_Pos}|T], Value) ->
- is_valid_set_strategy(T, Value);
- is_valid_set_strategy([], _) ->
- true;
- is_valid_set_strategy(_, _) ->
- false.
- set_strategy([H|T], App, Key, Value) ->
- case H of
- app_env ->
- application:set_env(App, Key, Value);
- os_env ->
- os:putenv(os_env_key(Key), Value);
- {os_env, ENV} ->
- os:putenv(ENV, Value);
- {mnesia,Type,Oid,Pos} ->
- mnesia:activity(
- Type,
- fun() ->
- Rec = case mnesia:read(Oid) of
- [] ->
- {Tab,K} = Oid,
- Tag = mnesia:table_info(Tab, record_name),
- Attrs = mnesia:table_info(Tab, attributes),
- list_to_tuple(
- [Tag,K |
- [undefined || _ <- tl(Attrs)]]);
- [Old] ->
- Old
- end,
- mnesia:write(setelement(Pos, Rec, Value))
- end)
- end,
- set_strategy(T, App, Key, Value);
- set_strategy([], _, _, Value) ->
- Value.
- is_string(S) ->
- try begin _ = iolist_to_binary(S),
- true
- end
- catch
- error:_ ->
- false
- end.
- %% @spec reg(Key::key()) -> true
- %%
- %% @doc
- %% @equiv reg(Key, default(Key), [])
- %% @end
- reg(Key) ->
- ?CATCH_GPROC_ERROR(reg1(valid_key(Key)), [Key]).
- reg1(Key) ->
- reg1(Key, default(Key), [], reg).
- %% @spec reg_or_locate(Key::key()) -> {pid(), NewValue}
- %%
- %% @doc
- %% @equiv reg_or_locate(Key, default(Key))
- %% @end
- reg_or_locate(Key) ->
- ?CATCH_GPROC_ERROR(reg_or_locate1(valid_key(Key)), [Key]).
- reg_or_locate1(Key) ->
- reg_or_locate1(Key, default(Key), self()).
- valid_key(Key) ->
- gproc_lib:valid_key(Key).
- default({T,_,_}) when T==c -> 0;
- default(_) -> undefined.
- %% @spec await(Key::key()) -> {pid(),Value}
- %% @equiv await(Key,infinity)
- %%
- await(Key) ->
- ?CATCH_GPROC_ERROR(await1(Key, infinity), [Key]).
- %% @spec await(Key::key(), Timeout) -> {pid(),Value}
- %% Timeout = integer() | infinity
- %%
- %% @doc Wait for a name or aggregated counter to be registered.
- %% The function raises an exception if the timeout expires. Timeout must be
- %% either an interger > 0 or 'infinity'.
- %% A small optimization: we first perform a lookup, to see if the name
- %% is already registered. This way, the cost of the operation will be
- %% roughly the same as of where/1 in the case where the name is already
- %% registered (the difference: await/2 also returns the value).
- %% @end
- %%
- await(Key, Timeout) ->
- ?CATCH_GPROC_ERROR(await1(Key, Timeout), [Key, Timeout]).
- %% @spec await(Node::node(), Key::key(), Timeout) -> {pid(),Value}
- %% Timeout = integer() | infinity
- %%
- %% @doc Wait for a name or aggregated counter to be registered on `Node'.
- %% This function works exactly like {@link await/2}, but queries a remote
- %% node instead. An exception is thrown if `Node' cannot be reached. If gproc
- %% is not running on a given node, this is treated the same as the node being
- %% down.
- %% @end
- %%
- await(Node, Key, Timeout) when Node == node() ->
- await(Key, Timeout);
- await(Node, Key, Timeout) when is_atom(Node) ->
- ?CATCH_GPROC_ERROR(await1(Node, Key, Timeout), [Node, Key, Timeout]).
- await1({T,g,_} = Key, Timeout) when T=:=n; T=:=a; T=:=rc ->
- ?CHK_DIST,
- request_wait(Key, Timeout);
- await1({T,l,_} = Key, Timeout) when T=:=n; T=:=a; T=:=rc ->
- case ets:lookup(?TAB, {Key, T}) of
- [{_, Pid, Value}] ->
- case is_process_alive(Pid) of
- true ->
- {Pid, Value};
- false ->
- %% we can send an asynchronous audit request, since the purpose is
- %% only to ensure that the server handles the audit before it serves
- %% our 'await' request. Strictly speaking, we could allow the bad Pid
- %% to be returned, as there are no guarantees that whatever Pid we return
- %% will still be alive when addressed. Still, we don't want to knowingly
- %% serve bad data.
- nb_audit_process(Pid),
- request_wait(Key, Timeout)
- end;
- _ ->
- request_wait(Key, Timeout)
- end;
- await1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- await1(N, {n,l,_} = Key, Timeout) when is_atom(N) ->
- request_wait(N, Key, Timeout);
- await1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- request_wait({_,g,_} = Key, Timeout) ->
- request_wait(undefined, Key, Timeout);
- request_wait(Key, Timeout) ->
- request_wait(node(), Key, Timeout).
- request_wait(N, {_,C,_} = Key, Timeout) when C==l; C==g ->
- TRef = case Timeout of
- infinity -> no_timer;
- T when is_integer(T), T > 0 ->
- erlang:start_timer(T, self(), gproc_timeout);
- _ ->
- ?THROW_GPROC_ERROR(badarg)
- end,
- WRef = case {call(N, {await,Key,self()}, C), C} of
- {{R, {Kg,Pg,Vg}}, g} ->
- self() ! {gproc, R, registered, {Kg,Pg,Vg}},
- R;
- {R,_} ->
- R
- end,
- receive
- {gproc, WRef, registered, {_K, Pid, V}} ->
- _ = case TRef of
- no_timer -> ignore;
- _ -> erlang:cancel_timer(TRef)
- end,
- {Pid, V};
- {timeout, TRef, gproc_timeout} ->
- cancel_wait(N, Key, WRef),
- ?THROW_GPROC_ERROR(timeout)
- end.
- %% @spec wide_await(Nodes::[node()], Key::key(), Timeout) -> {pid(),Value}
- %% Timeout = integer() | infinity
- %%
- %% @doc Wait for a local name to be registered on any of `Nodes'.
- %% This function works rather like {@link await/2}, but queries all nodes in
- %% the `Nodes' list at the same time. The first node to respond with a
- %% process registered as `Key' will provide the result. Other results are
- %% ignored. `Key' must be a unique name with local scope, i.e. `{n,l,Name}'.
- %%
- %% An exception is thrown upon timeout, or if no node can be reached (if gproc is
- %% not running on a given node, this is treated the same as the node being down).
- %% @end
- %%
- wide_await(Nodes, Key, Timeout) ->
- ?CATCH_GPROC_ERROR(wide_await1(Nodes, Key, Timeout), [Nodes, Key, Timeout]).
- wide_await1(Nodes, {T,l,_} = Key, Timeout) when T=:=n; T=:=a ->
- {_, Ref} = spawn_monitor(fun() ->
- wide_request_wait(Nodes, Key, Timeout)
- end),
- receive
- {'DOWN', Ref, _, _, Reason} ->
- case Reason of
- {ok, {gproc,_,registered,{_,Pid,V}}} ->
- {Pid, V};
- Other ->
- ?THROW_GPROC_ERROR(Other)
- end
- end;
- wide_await1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- wide_request_wait(Nodes, {Tk,l,_} = Key, Timeout) when Tk=:=n; Tk=:=a ->
- TRef = case Timeout of
- infinity -> no_timer;
- T when is_integer(T), T > 0 ->
- erlang:start_timer(T, self(), gproc_timeout);
- _ ->
- exit(badarg)
- end,
- Req = {await, Key, self()},
- Refs = lists:map(
- fun(Node) ->
- S = {?MODULE, Node},
- Ref = erlang:monitor(process, S),
- ?MAY_FAIL(erlang:send(S, {'$gen_call', {self(), Ref}, Req},
- [noconnect])),
- {Node, Ref}
- end, Nodes),
- collect_replies(Refs, Key, TRef).
- collect_replies(Refs, Key, TRef) ->
- receive
- {gproc, _Ref, registered, {_, _, _}} = Result ->
- exit({ok, Result});
- {'DOWN', Ref, _, _, _} ->
- case lists:keydelete(Ref, 2, Refs) of
- [] ->
- exit(nodedown);
- Refs1 ->
- collect_replies(Refs1, Key, TRef)
- end;
- {timeout, TRef, gproc_timeout} ->
- exit(timeout);
- {Ref, Ref} ->
- %% ignore
- collect_replies(Refs, Key, TRef)
- end.
- %% @spec nb_wait(Key::key()) -> Ref
- %%
- %% @doc Wait for a name or aggregated counter to be registered.
- %% The caller can expect to receive a message,
- %% {gproc, Ref, registered, {Key, Pid, Value}}, once the name is registered.
- %% @end
- %%
- nb_wait(Key) ->
- ?CATCH_GPROC_ERROR(nb_wait1(Key), [Key]).
- %% @spec nb_wait(Node::node(), Key::key()) -> Ref
- %%
- %% @doc Wait for a name or aggregated counter to be registered on `Node'.
- %% The caller can expect to receive a message,
- %% {gproc, Ref, registered, {Key, Pid, Value}}, once the name is registered.
- %% @end
- %%
- nb_wait(Node, Key) ->
- ?CATCH_GPROC_ERROR(nb_wait1(Node, Key), [Node, Key]).
- nb_wait1({T,g,_} = Key) when T=:=n; T=:=a; T=:=rc ->
- ?CHK_DIST,
- call({await, Key, self()}, g);
- nb_wait1({T,l,_} = Key) when T=:=n; T=:=a; T=:=rc ->
- call({await, Key, self()}, l);
- nb_wait1(_) ->
- ?THROW_GPROC_ERROR(badarg).
- nb_wait1(Node, {T,l,_} = Key) when is_atom(Node), T=:=n;
- is_atom(Node), T=:=a;
- is_atom(Node), T=:=rc ->
- call(Node, {await, Key, self()}, l).
- %% @spec cancel_wait(Key::key(), Ref) -> ok
- %% Ref = all | reference()
- %%
- %% @doc Cancels a previous call to nb_wait/1
- %%
- %% If `Ref = all', all wait requests on `Key' from the calling process
- %% are canceled.
- %% @end
- %%
- cancel_wait(Key, Ref) ->
- ?CATCH_GPROC_ERROR(cancel_wait1(Key, Ref), [Key, Ref]).
- %% @spec cancel_wait(Node::node(), Key::key(), Ref) -> ok
- %% Ref = all | reference()
- %%
- %% @doc Cancels a previous call to nb_wait/2
- %%
- %% This function works just like {@link cancel_wait/2}, but talks to a remote
- %% node.
- %% @end
- %%
- cancel_wait(N, Key, Ref) when N == node() ->
- cancel_wait(Key, Ref);
- cancel_wait(N, Key, Ref) ->
- ?CATCH_GPROC_ERROR(cancel_wait1(N, Key, Ref), [N, Key, Ref]).
- cancel_wait1({_,g,_} = Key, Ref) ->
- ?CHK_DIST,
- cast({cancel_wait, self(), Key, Ref}, g),
- ok;
- cancel_wait1({_,l,_} = Key, Ref) ->
- cast({cancel_wait, self(), Key, Ref}, l),
- ok.
- cancel_wait1(undefined, {_,g,_} = Key, Ref) ->
- cast({cancel_wait, self(), Key, Ref}, g);
- cancel_wait1(N, {_,l,_} = Key, Ref) ->
- cast(N, {cancel_wait, self(), Key, Ref}, l).
- cancel_wait_or_monitor(Key) ->
- ?CATCH_GPROC_ERROR(cancel_wait_or_monitor1(Key), [Key]).
- cancel_wait_or_monitor1({_,g,_} = Key) ->
- ?CHK_DIST,
- cast({cancel_wait_or_monitor, self(), Key}, g),
- ok;
- cancel_wait_or_monitor1({_,l,_} = Key) ->
- cast({cancel_wait_or_monitor, self(), Key}, l),
- ok.
- %% @equiv monitor(Key, info)
- monitor(Key) ->
- ?CATCH_GPROC_ERROR(monitor1(Key, info), [Key]).
- %% @spec monitor(key(), monitor_type()) -> reference()
- %%
- %% @doc monitor a registered name
- %% `monitor(Key, info)' works much like erlang:monitor(process, Pid), but monitors
- %% a unique name registered via gproc. A message, `{gproc, unreg, Ref, Key}'
- %% will be sent to the requesting process, if the name is unregistered or
- %% the registered process dies. If there is a standby monitor (see below), a
- %% message `{gproc, {failover, ToPid}, Ref, Key}' is sent to all monitors.
- %% If the name is passed to another process using {@link give_away/2}, the event
- %% `{gproc, {migrated, ToPid}, Ref, Key}' is sent to all monitors.
- %%
- %% `monitor(Key, standby)' sets up the monitoring process as a standby for the
- %% registered name. If the registered process dies, the first standby process
- %% inherits the name, and a message `{gproc, {failover, ToPid}, Ref, Key}' is
- %% sent to all monitors, including the one that inherited the name.
- %%
- %% If the name is not yet registered, the unreg event is sent immediately.
- %% If the calling process in this case tried to start a `standby' monitoring,
- %% it receives the registered name and the failover event immediately.
- %%
- %% `monitor(Key, follow)' keeps monitoring the registered name even if it is
- %% temporarily unregistered. The messages received are the same as for the other
- %% monitor types, but `{gproc, registered, Ref, Key}' is also sent when a new
- %% process registers the name.
- %% @end
- monitor(Key, Type) when Type==info;
- Type==follow;
- Type==standby ->
- ?CATCH_GPROC_ERROR(monitor1(Key, Type), [Key, Type]).
- monitor1({T,g,_} = Key, Type) when T==n; T==a; T==rc ->
- ?CHK_DIST,
- gproc_dist:monitor(Key, Type);
- monitor1({T,l,_} = Key, Type) when T==n; T==a; T==rc ->
- call({monitor, Key, self(), Type}, l);
- monitor1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec demonitor(key(), reference()) -> ok
- %%
- %% @doc Remove a monitor on a registered name
- %% This function is the reverse of monitor/1. It removes a monitor previously
- %% set on a unique name. This function always succeeds given legal input.
- %% @end
- demonitor(Key, Ref) ->
- ?CATCH_GPROC_ERROR(demonitor1(Key, Ref), [Key, Ref]).
- demonitor1({T,g,_} = Key, Ref) when T==n; T==a; T==rc ->
- ?CHK_DIST,
- gproc_dist:demonitor(Key, Ref);
- demonitor1({T,l,_} = Key, Ref) when T==n; T==a; T==rc ->
- call({demonitor, Key, Ref, self()}, l);
- demonitor1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec reg(Key::key(), Value::value()) -> true
- %%
- %% @doc Register a name or property for the current process
- %%
- %%
- reg(Key, Value) ->
- ?CATCH_GPROC_ERROR(reg1(valid_key(Key), Value, [], reg), [Key, Value]).
- %% @spec reg(Key::key(), Value::value(), Attrs::attrs()) -> true
- %%
- %% @doc Register a name or property for the current process
- %% `Attrs' (default: `[]') can be inspected using {@link get_attribute/2}.
- %%
- %% The structure of `Key' is `{Type, Context, Name}', where:
- %%
- %% * `Context :: l | g' - `l' means 'local' context; `g' means 'global'
- %% * `Type :: p | n | c | a | r | rc' specifies the type of entry
- %%
- %% The semantics of the different types:
- %%
- %% * `p' - 'property', is non-unique, i.e. different processes can each
- %% register a property with the same name.
- %% * `n' - 'name, is unique within the given context (local or global).
- %% * `c' - 'counter', is similar to a property, but has a numeric value
- %% and behaves roughly as an ets counter (see {@link update_counter/2}.)
- %% * `a' - 'aggregated counter', is automatically updated by gproc, and
- %% reflects the sum of all counter objects with the same name in the given
- %% scope. The initial value for an aggregated counter must be `undefined'.
- %% * `r' - 'resource property', behaves like a property, but can be tracked
- %% with a 'resource counter'. Note that using an `rc' wildcard name
- %% pattern (see below) for a resource property is not allowed.
- %% * `rc' - 'resource counter', tracks the number of resource properties
- %% with the same name. When the resource count reaches `0', any triggers
- %% specified using an `on_zero' attribute may be executed (see below).
- %% If `Name' is a tuple, the last element of the name can contain a
- %% wildcard, using the symbol <code>'\\_'</code>. This will make the resource
- %% counter keep track of any resources where all elements match except
- %% the last position. For example, <code>{rc,l,{a,b,'\\_'}}</code> would keep
- %% track of both `{r,l,{a,b,1}}' and `{r,l,{a,b,2}}'.
- %%
- %% On-zero triggers:
- %%
- %% `Msg = {gproc, resource_on_zero, Context, Name, Pid}'
- %%
- %% * `{send, Key}' - run `gproc:send(Key, Msg)'
- %% * `{bcast, Key}' - run `gproc:bcast(Key, Msg)'
- %% * `publish' - run
- %% `gproc_ps:publish(Context, gproc_resource_on_zero, {Context, Name, Pid})'
- %% * `{unreg_shared, Type, Name}' - unregister the shared key
- %% `{Type, Context, Name}'
- %% @end
- reg(Key, Value, Attrs) ->
- ?CATCH_GPROC_ERROR(reg1(valid_key(Key), Value, Attrs, reg), [Key, Value, Attrs]).
- %% @equiv ensure_reg(Key, default(Key), [])
- ensure_reg(Key) ->
- ?CATCH_GPROC_ERROR(reg1(valid_key(Key), ensure), [Key]).
- %% @equiv ensure_reg(Key, Value, [])
- -spec ensure_reg(key(), value()) -> new | updated.
- ensure_reg(Key, Value) ->
- ?CATCH_GPROC_ERROR(reg1(valid_key(Key), Value, ensure), [Key, Value]).
- %% @spec ensure_reg(Key::key(), Value::value(), Attrs::attrs()) ->
- %% new | updated
- %%
- %% @doc Registers a new name or property unless such and entry (by key) has
- %% already been registered by the current process. If `Key' already exists,
- %% the entry will be updated with the given `Value' and `Attrs'.
- %%
- %% This function allows the caller to efficiently register an entry without
- %% first checking whether it has already been registered. An exception is
- %% raised if the name or property is already registered by someone else.
- %% @end
- -spec ensure_reg(key(), value(), attrs()) -> new | updated.
- ensure_reg(Key, Value, Attrs) ->
- ?CATCH_GPROC_ERROR(reg1(valid_key(Key), Value, Attrs, ensure), [Key, Value, Attrs]).
- reg1(Key, Op) ->
- reg1(Key, default(Key), [], Op).
- reg1(Key, Value, Op) ->
- reg1(Key, Value, [], Op).
- reg1({T,g,_} = Key, Value, As, Op) when T==p; T==a; T==c; T==n; T==r; T==rc ->
- %% anything global
- ?CHK_DIST,
- gproc_dist:reg(Key, Value, As, Op);
- reg1({p,l,_} = Key, Value, As, Op) ->
- local_reg(Key, Value, As, Op);
- reg1({a,l,_} = Key, undefined, As, Op) ->
- call({reg, Key, undefined, As, Op});
- reg1({c,l,_} = Key, Value, As, Op) when is_integer(Value) ->
- call({reg, Key, Value, As, Op});
- reg1({n,l,_} = Key, Value, As, Op) ->
- call({reg, Key, Value, As, Op});
- reg1({r,l,_} = Key, Value, As, Op) ->
- call({reg, Key, Value, As, Op});
- reg1({rc,l,_} = Key, Value, As, Op) ->
- call({reg, Key, Value, As, Op});
- reg1(_, _, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @equiv reg_other(Key, Pid, default(Key), [])
- reg_other(Key, Pid) ->
- ?CATCH_GPROC_ERROR(reg_other1(valid_key(Key), Pid, reg), [Key, Pid]).
- %% @equiv reg_other(Key, Pid, Value, [])
- reg_other(Key, Pid, Value) ->
- ?CATCH_GPROC_ERROR(reg_other1(valid_key(Key), Pid, Value, [], reg), [Key, Pid, Value]).
- %% @spec reg_other(Key, Pid, Value, Attrs) -> true
- %% @doc Register name or property to another process.
- %%
- %% Equivalent to {@link reg/3}, but allows for registration of another process
- %% instead of the current process.
- %%
- %% Note that registering other processes introduces the possibility of
- %% confusing race conditions in user code. Letting each process register
- %% its own resources is highly recommended.
- %%
- %% Only the following resource types can be registered through this function:
- %%
- %% * `n' - unique names
- %% * `a' - aggregated counters
- %% * `r' - resource properties
- %% * `rc' - resource counters
- %% @end
- reg_other(Key, Pid, Value, Attrs) ->
- ?CATCH_GPROC_ERROR(reg_other1(valid_key(Key), Pid, Value, Attrs, reg),
- [Key, Pid, Value, Attrs]).
- ensure_reg_other(Key, Pid) ->
- ?CATCH_GPROC_ERROR(reg_other1(valid_key(Key), Pid, ensure), [Key, Pid]).
- %% @equiv ensure_reg_other(Key, Pid, Value, [])
- ensure_reg_other(Key, Pid, Value) ->
- ?CATCH_GPROC_ERROR(reg_other1(valid_key(Key), Pid, Value, [], ensure),
- [Key, Pid, Value]).
- %% @spec ensure_reg_other(Key::key(), Pid::pid(),
- %% Value::value(), Attrs::attrs()) ->
- %% new | updated
- %%
- %% @doc Register or update name or property to another process.
- %%
- %% Equivalent to {@link reg_other/3}, but allows for registration of another
- %% process instead of the current process. Also see {@link ensure_reg/3}.
- %% @end
- ensure_reg_other(Key, Pid, Value, Attrs) ->
- ?CATCH_GPROC_ERROR(reg_other1(valid_key(Key), Pid, Value, Attrs, ensure),
- [Key, Pid, Value, Attrs]).
- reg_other1(Key, Pid, Op) ->
- reg_other1(Key, Pid, default(Key), [], Op).
- reg_other1({_,g,_} = Key, Pid, Value, As, Op) when is_pid(Pid) ->
- ?CHK_DIST,
- gproc_dist:reg_other(Key, Pid, Value, As, Op);
- reg_other1({T,l,_} = Key, Pid, Value, As, Op) when is_pid(Pid) ->
- if T==n; T==a; T==r; T==rc ->
- call({reg_other, Key, Pid, Value, As, Op});
- true ->
- ?THROW_GPROC_ERROR(badarg)
- end.
- %% @spec reg_or_locate(Key::key(), Value) -> {pid(), NewValue}
- %%
- %% @doc Try registering a unique name, or return existing registration.
- %%
- %% This function tries to register the name `Key', if available.
- %% If such a registration object already exists, the pid and value of
- %% the current registration is returned instead.
- %% @end
- reg_or_locate(Key, Value) ->
- ?CATCH_GPROC_ERROR(reg_or_locate1(valid_key(Key), Value, self()), [Key, Value]).
- %% @spec reg_or_locate(Key::key(), Value, Fun::fun()) -> {pid(), NewValue}
- %%
- %% @doc Spawn a process with a registered name, or return existing registration.
- %%
- %% This function checks whether a local name is registered; if not, it spawns
- %% a new process (with `spawn(Fun)') and gives it the name.
- %% The pid and value of the resulting registration is returned.
- %%
- %% When a global name is registered in this fashion, the process is
- %% spawned on the caller's node, and the group_leader of the spawned
- %% process is set to the group_leader of the calling process.
- %% @end
- reg_or_locate({n,_,_} = Key, Value, F) when is_function(F, 0) ->
- ?CATCH_GPROC_ERROR(reg_or_locate1(valid_key(Key), Value, F), [Key, Value, F]).
- reg_or_locate1({_,g,_} = Key, Value, P) ->
- ?CHK_DIST,
- gproc_dist:reg_or_locate(Key, Value, P);
- reg_or_locate1({T,l,_} = Key, Value, P) when T==n; T==a; T==rc ->
- call({reg_or_locate, Key, Value, P});
- reg_or_locate1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec reg_shared(Key::key()) -> true
- %%
- %% @doc Register a resource, but don't tie it to a particular process.
- %%
- %% `reg_shared({c,l,C}) -> reg_shared({c,l,C}, 0).'
- %% `reg_shared({a,l,A}) -> reg_shared({a,l,A}, undefined).'
- %% @end
- reg_shared(Key) ->
- ?CATCH_GPROC_ERROR(reg_shared1(valid_key(Key)), [Key]).
- %% @private
- reg_shared1({T,_,_} = Key) when T==a; T==p; T==c; T==r ->
- reg_shared(Key, default(Key)).
- %% @spec reg_shared(Key::key(), Value) -> true
- %%
- %% @doc Register a resource, but don't tie it to a particular process.
- %%
- %% Shared resources are all unique. They remain until explicitly unregistered
- %% (using {@link unreg_shared/1}). The types of shared resources currently
- %% supported are `counter' and `aggregated counter'. In listings and query
- %% results, shared resources appear as other similar resources, except that
- %% `Pid == shared'. To wit, update_counter({c,l,myCounter}, shared, 1) would
- %% increment the shared counter `myCounter' with 1, provided it exists.
- %%
- %% A shared aggregated counter will track updates in exactly the same way as
- %% an aggregated counter which is owned by a process.
- %% @end
- %%
- reg_shared(Key, Value) ->
- ?CATCH_GPROC_ERROR(reg_shared1(valid_key(Key), Value, []), [Key, Value]).
- reg_shared(Key, Value, Attrs) when is_list(Attrs) ->
- ?CATCH_GPROC_ERROR(reg_shared1(valid_key(Key), Value, Attrs), [Key, Value, Attrs]).
- %% @private
- reg_shared1({_,g,_} = Key, Value, As) ->
- %% anything global
- ?CHK_DIST,
- gproc_dist:reg_shared(Key, Value, As);
- reg_shared1({a,l,_} = Key, undefined, As) ->
- call({reg_shared, Key, undefined, As, reg});
- reg_shared1({c,l,_} = Key, Value, As) when is_integer(Value) ->
- call({reg_shared, Key, Value, As, reg});
- reg_shared1({p,l,_} = Key, Value, As) ->
- call({reg_shared, Key, Value, As, reg});
- reg_shared1({rc,l,_} = Key, undefined, As) ->
- call({reg_shared, Key, undefined, As, reg});
- reg_shared1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec mreg(type(), scope(), [{Key::any(), Value::any()}]) -> true
- %%
- %% @doc Register multiple {Key,Value} pairs of a given type and scope.
- %%
- %% This function is more efficient than calling {@link reg/2} repeatedly.
- %% It is also atomic in regard to unique names; either all names are registered
- %% or none are.
- %% @end
- mreg(T, C, KVL) ->
- ?CATCH_GPROC_ERROR(mreg1(T, C, KVL), [T, C, KVL]).
- mreg1(T, g, KVL) ->
- ?CHK_DIST,
- gproc_dist:mreg(T, KVL);
- mreg1(T, l, KVL) when T==p; T==n; T==a; T==r ->
- if is_list(KVL) ->
- call({mreg, T, l, KVL});
- true ->
- erlang:error(badarg)
- end;
- mreg1(p, l, KVL) ->
- local_mreg(p, KVL);
- mreg1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec munreg(type(), scope(), [Key::any()]) -> true
- %%
- %% @doc Unregister multiple Key items of a given type and scope.
- %%
- %% This function is usually more efficient than calling {@link unreg/1}
- %% repeatedly.
- %% @end
- munreg(T, C, L) ->
- ?CATCH_GPROC_ERROR(munreg1(T, C, L), [T, C, L]).
- munreg1(T, g, L) ->
- ?CHK_DIST,
- gproc_dist:munreg(T, existing(T,g,L));
- munreg1(T, l, L) when T==p; T==a; T==n; T==rc; T==r ->
- if is_list(L) ->
- call({munreg, T, l, existing(T,l,L)});
- true ->
- erlang:error(badarg)
- end;
- munreg1(p, l, L) ->
- local_munreg(p, existing(p,l,L));
- munreg1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- existing(T,Scope,L) ->
- Keys = if T==p; T==c; T==r ->
- [{{T,Scope,K}, self()} || K <- L];
- T==a; T==n; T==rc ->
- [{{T,Scope,K}, T} || K <- L]
- end,
- _ = [case ets:member(?TAB, K) of
- false -> erlang:error(badarg);
- true -> true
- end || K <- Keys],
- L.
- %% @spec (Key:: key()) -> true
- %%
- %% @doc Unregister a name or property.
- %% @end
- unreg(Key) ->
- ?CATCH_GPROC_ERROR(unreg1(Key), [Key]).
- unreg1(Key) ->
- case Key of
- {_, g, _} ->
- ?CHK_DIST,
- gproc_dist:unreg(Key);
- {T, l, _} when T == n; T == a; T == r; T == rc ->
- call({unreg, Key});
- {_, l, _} ->
- case ets:member(?TAB, {Key,self()}) of
- true ->
- _ = gproc_lib:remove_reg(Key, self(), unreg),
- true;
- false ->
- ?THROW_GPROC_ERROR(badarg)
- end
- end.
- %% @spec unreg_other(key(), pid()) -> true
- %% @doc Unregister a name registered to another process.
- %%
- %% This function is equivalent to {@link unreg/1}, but specifies another
- %% process as the holder of the registration. An exception is raised if the
- %% name or property is not registered to the given process.
- %% @end
- unreg_other(Key, Pid) ->
- ?CATCH_GPROC_ERROR(unreg_other1(Key, Pid), [Key, Pid]).
- unreg_other1({_,g,_} = Key, Pid) ->
- ?CHK_DIST,
- gproc_dist:unreg_other(Key, Pid);
- unreg_other1({T,l,_} = Key, Pid) when is_pid(Pid) ->
- if T==n; T==a; T==r; T==rc ->
- call({unreg_other, Key, Pid});
- true ->
- ?THROW_GPROC_ERROR(badarg)
- end.
- %% @spec (Key::key(), Props::[{atom(), any()}]) -> true
- %%
- %% @doc Add/modify `{Key, Value}' attributes associated with a registration.
- %%
- %% Gproc registration objects can have `{Key, Value}' attributes associated with
- %% them. These are stored in a way that doesn't affect the cost of name lookup.
- %%
- %% Attributs can be retrieved using `gproc:get_attribute/3' or
- %% `gproc:get_attributes/2'.
- %% @end
- set_attributes(Key, Props) ->
- ?CATCH_GPROC_ERROR(set_attributes1(Key, Props), [Key, Props]).
- set_attributes1(Key, Props) ->
- case Key of
- {_, g, _} ->
- ?CHK_DIST,
- gproc_dist:set_attributes(Key, Props);
- {_, l, _} ->
- call({set_attributes, Key, Props})
- end.
- %% @spec (Key:: key()) -> true
- %%
- %% @doc Unregister a shared resource.
- %% @end
- unreg_shared(Key) ->
- ?CATCH_GPROC_ERROR(unreg_shared1(Key), [Key]).
- %% @private
- unreg_shared1(Key) ->
- case Key of
- {_, g, _} ->
- ?CHK_DIST,
- gproc_dist:unreg_shared(Key);
- {T, l, _} when T == c;
- T == a;
- T == p;
- T == rc -> call({unreg_shared, Key});
- _ ->
- ?THROW_GPROC_ERROR(badarg)
- end.
- %% @spec (Key::key(), Props::[{K,V}]) -> true
- %% @doc Add/modify `{Key, Value}' attributes associated with a shared registration.
- %%
- %% Gproc registration objects can have `{Key, Value}' attributes associated with
- %% them. These are stored in a way that doesn't affect the cost of name lookup.
- %%
- %% Attributes can be retrieved using `gproc:get_attribute/3' or
- %% `gproc:get_attributes/2'.
- %% @end
- %%
- set_attributes_shared(Key, Attrs) ->
- ?CATCH_GPROC_ERROR(set_attributes_shared1(Key, Attrs), [Key, Attrs]).
- set_attributes_shared1(Key, Attrs) ->
- case Key of
- {_, g, _} ->
- ?CHK_DIST,
- gproc_dist:set_attributes_shared(Key, Attrs);
- {_, l, _} ->
- call({set_attributes_shared, Key, Attrs})
- end.
- %% @spec (key(), pid()) -> yes | no
- %%
- %% @doc Behaviour support callback
- %% @end
- register_name({n,_,_} = Name, Pid) when Pid == self() ->
- try reg(Name), yes
- catch
- error:_ ->
- no
- end.
- %% @equiv unreg/1
- unregister_name(Key) ->
- unreg(Key).
- %% @spec select(Arg) -> [Match] | {[Match], Continuation} | '$end_of_table'
- %% where Arg = Continuation
- %% | sel_pattern()
- %% Match = {Key, Pid, Value}
- %% @doc Perform a select operation on the process registry
- %%
- %% When Arg = Contination, resume a gproc:select/1 operation
- %% (see {@link //stdlib/ets:select/1. ets:select/1}
- %%
- %% When Arg = {@type sel_pattern()}, this function executes a select operation,
- %% emulating ets:select/1
- %%
- %% {@link select/2} offers the opportunity to narrow the search
- %% (by limiting to only global or local scope, or a single type of object).
- %% When only a pattern as single argument is given, both global and local scope,
- %% as well as all types of object can be searched. Note that the pattern may
- %% still limit the select operation so that scanning the entire table is avoided.
- %%
- %% The physical representation in the registry may differ from the above,
- %% but the select patterns are transformed appropriately. The logical
- %% representation for the gproc select operations is given by
- %% {@type headpat()}.
- %% @end
- select({?TAB, _, _, _, _, _, _, _} = Continuation) ->
- ets:select(Continuation);
- select(Pat) ->
- select(all, Pat).
- %% @spec (Context::sel_context(), Pat::sel_pattern()) -> [{Key, Pid, Value}]
- %%
- %% @doc Perform a select operation with limited context on the process registry
- %%
- %% The physical representation in the registry may differ from the above,
- %% but the select patterns are transformed appropriately.
- %%
- %% Note that limiting the context is just a convenience function, allowing you
- %% to write a simpler select pattern and still avoid searching the entire
- %% registry. Whenever variables are used in the head pattern, this will result
- %% in a wider scan, even if the values are restricted through a guard (e.g.
- %% <code>select([{'$1','$2','$3'}, [{'==', {element,1,'$1'}, p}], ...])</code>
- %% will count as a wild pattern on the key and result in a full scan).
- %% In this case, specifying a Context will allow gproc to perform some
- %% variable substitution and ensure that the scan is limited.
- %% @end
- select(Context, Pat) ->
- ets:select(?TAB, pattern(Pat, Context)).
- %% @spec (Context::context(), Pat::sel_patten(), Limit::integer()) ->
- %% {[Match],Continuation} | '$end_of_table'
- %% @doc Like {@link select/2} but returns Limit objects at a time.
- %%
- %% See [http://www.erlang.org/doc/man/ets.html#select-3].
- %% @end
- select(Context, Pat, Limit) ->
- ets:select(?TAB, pattern(Pat, Context), Limit).
- %% @spec (sel_pattern()) -> list(sel_object())
- %% @doc
- %% @equiv select_count(all, Pat)
- %% @end
- select_count(Pat) ->
- select_count(all, Pat).
- %% @spec (Context::context(), Pat::sel_pattern()) -> [{Key, Pid, Value}]
- %%
- %% @doc Perform a select_count operation on the process registry.
- %%
- %% The physical representation in the registry may differ from the above,
- %% but the select patterns are transformed appropriately.
- %% @end
- select_count(Context, Pat) ->
- ets:select_count(?TAB, pattern(Pat, Context)).
- %%% Local properties can be registered in the local process, since
- %%% no other process can interfere.
- %%%
- local_reg({_,Scope,_} = Key, Value, As, Op) ->
- case gproc_lib:insert_reg(Key, Value, self(), l) of
- false ->
- case ets:member(?TAB, {Key, self()}) of
- true when Op == ensure ->
- gproc_lib:do_set_value(Key, Value, self()),
- set_attrs(As, Key, self()),
- updated;
- _ ->
- ?THROW_GPROC_ERROR(badarg)
- end;
- true ->
- monitor_me(),
- if As =/= [] ->
- gproc_lib:insert_attr(Key, As, self(), Scope),
- regged_new(Op);
- true ->
- regged_new(Op)
- end
- end.
- regged_new(reg ) -> true;
- regged_new(ensure) -> new.
- local_mreg(_, []) -> true;
- local_mreg(T, [_|_] = KVL) ->
- case gproc_lib:insert_many(T, l, KVL, self()) of
- false -> ?THROW_GPROC_ERROR(badarg);
- {true,_} -> monitor_me()
- end.
- local_munreg(T, L) when T==p; T==c ->
- _ = [gproc_lib:remove_reg({T,l,K}, self(), unreg) || K <- L],
- true.
- %% @spec (Key :: key(), Value) -> true
- %% @doc Sets the value of the registration given by Key
- %%
- %% Key is assumed to exist and belong to the calling process.
- %% If it doesn't, this function will exit.
- %%
- %% Value can be any term, unless the object is a counter, in which case
- %% it must be an integer.
- %% @end
- %%
- set_value(Key, Value) ->
- ?CATCH_GPROC_ERROR(set_value1(Key, Value), [Key, Value]).
- %% @spec (Key :: key(), Value) -> true
- %% @doc Sets the value of the shared registration given by Key
- %%
- %% Key is assumed to exist as a shared entity.
- %% If it doesn't, this function will exit.
- %%
- %% Value can be any term, unless the object is a counter, in which case
- %% it must be an integer.
- %% @end
- %%
- set_value_shared({T,_,_} = Key, Value) when T == c;
- T == a;
- T == p;
- T == r ->
- ?CATCH_GPROC_ERROR(set_value_shared1(Key, Value), [Key, Value]).
- set_value1({_,g,_} = Key, Value) ->
- ?CHK_DIST,
- gproc_dist:set_value(Key, Value);
- set_value1({a,l,_} = Key, Value) when is_integer(Value) ->
- call({set, Key, Value});
- set_value1({n,l,_} = Key, Value) ->
- %% we cannot do this locally, since we have to check that the object
- %% exists first - not an atomic update.
- call({set, Key, Value});
- set_value1({p,l,_} = Key, Value) ->
- %% we _can_ to this locally, since there is no race condition - no
- %% other process can update our properties.
- case gproc_lib:do_set_value(Key, Value, self()) of
- true -> true;
- false ->
- erlang:error(badarg)
- end;
- set_value1({c,l,_} = Key, Value) when is_integer(Value) ->
- gproc_lib:do_set_counter_value(Key, Value, self());
- set_value1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- set_value_shared1({_,g,_} = Key, Value) ->
- ?CHK_DIST,
- gproc_dist:set_value_shared(Key, Value);
- set_value_shared1({_,l,_} = Key, Value) ->
- call({set_shared, Key, Value}).
- %% @spec (Key) -> Value
- %% @doc Reads the value stored with a key registered to the current process.
- %%
- %% If no such key is registered to the current process, this function exits.
- %% @end
- get_value(Key) ->
- ?CATCH_GPROC_ERROR(get_value1(Key, self()), [Key]).
- %% @spec (Key) -> Value
- %% @doc Reads the value stored with a shared key.
- %%
- %% If no such shared key is registered, this function exits.
- %% @end
- get_value_shared(Key) ->
- ?CATCH_GPROC_ERROR(get_value1(Key, shared), [Key]).
- %% @spec (Key, Pid) -> Value
- %% @doc Reads the value stored with a key registered to the process Pid.
- %%
- %% If `Pid == shared', the value of a shared key (see {@link reg_shared/1})
- %% will be read.
- %% @end
- %%
- get_value(Key, Pid) ->
- ?CATCH_GPROC_ERROR(get_value1(Key, Pid), [Key, Pid]).
- get_value1({T,_,_} = Key, Pid) when is_pid(Pid) ->
- if T==n; T==a; T==rc ->
- case ets:lookup(?TAB, {Key, T}) of
- [{_, P, Value}] when P == Pid -> Value;
- _ -> ?THROW_GPROC_ERROR(badarg)
- end;
- true ->
- ets:lookup_element(?TAB, {Key, Pid}, 3)
- end;
- get_value1({T,_,_} = K, shared) when T==c; T==a; T==p; T==r ->
- Key = case T of
- c -> {K, shared};
- p -> {K, shared};
- r -> {K, shared};
- a -> {K, a}
- end,
- case ets:lookup(?TAB, Key) of
- [{_, shared, Value}] -> Value;
- _ -> ?THROW_GPROC_ERROR(badarg)
- end;
- get_value1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec (Key, Attribute::atom()) -> Value
- %% @doc Get attribute value of `Attr' associated with `Key' for most likely Pid.
- %%
- %% The most likely Pid in this case is `self()' for properties and counters,
- %% and the current registration holder in case of names or aggregated counters.
- %% An exception is raised if `Key' is not registered for the given process.
- %% @end
- get_attribute(Key, A) ->
- Pid = case Key of
- {T,_,_} when T==n; T==a; T==rc ->
- where(Key);
- {T,_,_} when T==p; T==c; T==r ->
- self()
- end,
- ?CATCH_GPROC_ERROR(get_attribute1(Key, Pid, A), [Key, A]).
- %% @spec (Key, Pid::pid() | shared, Attr::atom()) -> Value
- %% @doc Get the attribute value of `Attr' associated with `Key' for process Pid.
- %%
- %% If `Pid == shared', the attribute of a shared key (see {@link reg_shared/1})
- %% will be read.
- %% @end
- %%
- get_attribute(Key, Pid, A) ->
- ?CATCH_GPROC_ERROR(get_attribute1(Key, Pid, A), [Key, Pid, A]).
- %% @spec (Key, Attr::atom()) -> Value
- %% @doc Get the attribute value of `Attr' associated with the shared `Key'.
- %%
- %% Equivalent to `get_attribute(Key, shared, Attr)'
- %% (see {@link get_attribute/3}).
- %% @end
- get_attribute_shared(Key, Attr) ->
- ?CATCH_GPROC_ERROR(get_attribute1(Key, shared, Attr), [Key, Attr]).
- %% @private
- get_attribute1({_,_,_} = Key, Pid, A) when is_pid(Pid); Pid==shared ->
- case ets:lookup(?TAB, {Pid, Key}) of
- [{_, Attrs}] ->
- case lists:keyfind(attrs, 1, Attrs) of
- false -> undefined;
- {_, As} ->
- case lists:keyfind(A, 1, As) of
- false -> undefined;
- {_, V} -> V
- end
- end;
- _ -> ?THROW_GPROC_ERROR(badarg)
- end;
- get_attribute1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec get_attributes(Key::key()) -> [{K, V}]
- %% @doc Get attributes associated with registration.
- %% @equiv get_attributes(Key, self())
- %%
- get_attributes(Key) ->
- ?CATCH_GPROC_ERROR(get_attributes1(Key, self()), [Key]).
- %% @spec (Key::key(), Pid::pid() | shared) -> [{K, V}]
- %%
- %% @doc Returns the list of attributes associated with the registration.
- %%
- %% This function raises a `badarg' exception if there is no corresponding
- %% registration.
- %%
- get_attributes(Key, Pid) ->
- ?CATCH_GPROC_ERROR(get_attributes1(Key, Pid), [Key, Pid]).
- get_attributes1({_,_,_} = Key, Pid) when is_pid(Pid); Pid==shared ->
- case ets:lookup(?TAB, {Pid, Key}) of
- [{_, Attrs}] ->
- case lists:keyfind(attrs, 1, Attrs) of
- false -> [];
- {_, As} -> As
- end;
- _ -> ?THROW_GPROC_ERROR(badarg)
- end;
- get_attributes1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec (Key) -> Pid
- %% @doc Lookup the Pid stored with a key.
- %%
- %% This function raises a `badarg' exception if `Key' is not registered.
- %% @end
- lookup_pid({_T,_,_} = Key) ->
- case where(Key) of
- undefined -> erlang:error(badarg);
- P -> P
- end.
- %% @spec (Key) -> Value
- %% @doc Lookup the value stored with a key.
- %%
- %% This function raises a `badarg' exception if `Key' is not registered.
- %% @end
- lookup_value({T,_,_} = Key) ->
- if T==n orelse T==a orelse T==rc ->
- ets:lookup_element(?TAB, {Key,T}, 3);
- true ->
- erlang:error(badarg)
- end.
- %% @spec (Key::key()) -> pid() | undefined
- %%
- %% @doc Returns the pid registered as Key
- %%
- %% The type of registration must be either name or aggregated counter.
- %% Otherwise this function will raise a `badarg' exception.
- %% Use {@link lookup_pids/1} in these cases.
- %% @end
- %%
- where(Key) ->
- ?CATCH_GPROC_ERROR(where1(Key), [Key]).
- where1({T,_,_}=Key) ->
- if T==n orelse T==a orelse T==rc ->
- case ets:lookup(?TAB, {Key,T}) of
- [{_, P, _Value}] ->
- case my_is_process_alive(P) of
- true -> P;
- false ->
- undefined
- end;
- _ -> % may be [] or [{Key,Waiters}]
- undefined
- end;
- true ->
- ?THROW_GPROC_ERROR(badarg)
- end.
- %% @equiv where/1
- whereis_name(Key) ->
- ?CATCH_GPROC_ERROR(where1(Key), [Key]).
- %% @spec (Key::key()) -> [pid()]
- %%
- %% @doc Returns a list of pids with the published key Key
- %%
- %% If the type of registration is either name or aggregated counter,
- %% this function will return either an empty list, or a list of one pid.
- %% For non-unique types, the return value can be a list of any length.
- %%
- %% Note: shared resources are not associated with any pid, and will
- %% therefore be excluded.
- %% @end
- %%
- lookup_pids({T,_,_} = Key) ->
- L = if T==n orelse T==a orelse T==rc ->
- ets:select(?TAB, [{{{Key,T}, '$1', '_'},
- [{is_pid, '$1'}], ['$1']}]);
- true ->
- ets:select(?TAB, [{{{Key,'_'}, '$1', '_'},
- [{is_pid, '$1'}], ['$1']}])
- end,
- [P || P <- L, my_is_process_alive(P)].
- %% @spec (pid()) -> boolean()
- %%
- my_is_process_alive(P) when node(P) =:= node() ->
- is_process_alive(P);
- my_is_process_alive(_) ->
- %% remote pid - assume true (too costly to find out)
- true.
- %% @spec (Key::key()) -> [{pid(), Value}]
- %%
- %% @doc Retrieve the `{Pid,Value}' pairs corresponding to Key.
- %%
- %% Key refer to any type of registry object. If it refers to a unique
- %% object, the list will be of length 0 or 1. If it refers to a non-unique
- %% object, the return value can be a list of any length.
- %% @end
- %%
- lookup_values({T,_,_} = Key) ->
- L = if T==n orelse T==a orelse T==rc ->
- ets:select(?TAB, [{{{Key,T}, '$1', '$2'},[],[{{'$1','$2'}}]}]);
- true ->
- ets:select(?TAB, [{{{Key,'_'}, '$1', '$2'},[],[{{'$1','$2'}}]}])
- end,
- [Pair || {P,_} = Pair <- L, my_is_process_alive(P)].
- %% @ spec (Key::key(), Incr) -> integer() | [integer()]
- %% Incr = IncrVal | UpdateOp | [UpdateOp]
- %% UpdateOp = IncrVal | {IncrVal, Threshold, SetValue}
- %% IncrVal = integer()
- %%
- %% @doc Updates the counter registered as Key for the current process.
- %%
- %% This function works almost exactly like ets:update_counter/3
- %% (see [http://www.erlang.org/doc/man/ets.html#update_counter-3]), but
- %% will fail if the type of object referred to by Key is not a counter or
- %% a unique name (update_counter/2 can be performed on names as well, but they
- %% do not count as counter objects, and do not affect aggregated counters).
- %%
- %% Aggregated counters with the same name will be updated automatically.
- %% The `UpdateOp' patterns are the same as for `ets:update_counter/3', except
- %% that the position is omitted; in gproc, the value position is always `3'.
- %%
- %% If `Key' refers to a unique name, the operation will depend on the value
- %% part of the registration being an integer(). While non-integer values are
- %% not permitted at all for counter objects, it is the user's responsibility to
- %% ensure that a name, on which `update_counter/2' is to be performed, has the
- %% appropriate value type.
- %% @end
- %%
- -spec update_counter(key(), increment()) -> integer() | [integer()].
- update_counter(Key, Incr) ->
- Pid = case Key of
- {n,_,_} -> n;
- {T,_,_} when T==c; T==r ->
- self()
- end,
- ?CATCH_GPROC_ERROR(update_counter1(Key, Pid, Incr), [Key, Incr]).
- update_counter(Key, Pid, Incr) when is_pid(Pid);
- Pid == shared; Pid == n ->
- ?CATCH_GPROC_ERROR(update_counter1(Key, Pid, Incr), [Key, Pid, Incr]).
- update_counter1({T,l,_} = Key, Pid, Incr) when T==c; T==r; T==n ->
- gproc_lib:update_counter(Key, Incr, Pid);
- update_counter1({T,g,_} = Key, Pid, Incr) when T==c; T==r; T==n ->
- ?CHK_DIST,
- gproc_dist:update_counter(Key, Pid, Incr);
- update_counter1(_, _, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @doc Update a list of counters
- %%
- %% This function is not atomic, except (in a sense) for global counters. For local counters,
- %% it is more of a convenience function. For global counters, it is much more efficient
- %% than calling `gproc:update_counter/2' for each individual counter.
- %%
- %% The return value is the corresponding list of `[{Counter, Pid, NewValue}]'.
- %% @end
- -spec update_counters(scope(), [{key(), pid(), increment()}]) ->
- [{key(), pid(), integer()}].
- update_counters(_, []) ->
- [];
- update_counters(l, [_|_] = Cs) ->
- ?CATCH_GPROC_ERROR(update_counters1(Cs), [Cs]);
- update_counters(g, [_|_] = Cs) ->
- ?CHK_DIST,
- gproc_dist:update_counters(Cs).
- update_counters1([{{c,l,_} = Key, Pid, Incr}|T]) ->
- [{Key, Pid, gproc_lib:update_counter(Key, Incr, Pid)}|update_counters1(T)];
- update_counters1([]) ->
- [];
- update_counters1(_) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec (Key) -> {ValueBefore, ValueAfter}
- %% Key = {c, Scope, Name} | {n, Scope, Name}
- %% Scope = l | g
- %% ValueBefore = integer()
- %% ValueAfter = integer()
- %%
- %% @doc Reads and resets a counter in a "thread-safe" way
- %%
- %% This function reads the current value of a counter and then resets it to its
- %% initial value. The reset operation is done using {@link update_counter/2},
- %% which allows for concurrent calls to {@link update_counter/2} without losing
- %% updates. Aggregated counters are updated accordingly.
- %%
- %% If `Key' refers to a unique name, the operation will depend on the value
- %% part of the registration being an integer(). While non-integer values are
- %% not permitted at all for counter objects, it is the user's responsibility to
- %% ensure that a name, on which `reset_counter/1' is to be performed, has the
- %% appropriate value type.
- %% @end
- %%
- reset_counter(Key) ->
- ?CATCH_GPROC_ERROR(reset_counter1(Key), [Key]).
- reset_counter1({T,g,_} = Key) when T==c; T==n ->
- ?CHK_DIST,
- gproc_dist:reset_counter(Key);
- reset_counter1({n,l,_} = Key) ->
- [{_, Pid, Current}] = ets:lookup(?TAB, {Key, n}),
- {Current, update_counter(Key, get_initial(Pid, Key) - Current)};
- reset_counter1({c,l,_} = Key) ->
- Current = ets:lookup_element(?TAB, {Key, self()}, 3),
- {Current, update_counter(Key, get_initial(self(), Key) - Current)}.
- get_initial(Pid, Key) ->
- case ets:lookup(?TAB, {Pid, Key}) of
- [{_, r}] -> 0;
- [{_, Opts}] ->
- proplists:get_value(initial, Opts, 0)
- end.
- %% @spec (Key::key(), Incr) -> integer() | [integer()]
- %% Incr = IncrVal | UpdateOp | [UpdateOp]
- %% UpdateOp = IncrVal | {IncrVal, Threshold, SetValue}
- %% IncrVal = integer()
- %%
- %% @doc Updates the shared counter registered as Key.
- %%
- %% This function works almost exactly like ets:update_counter/3
- %% (see [http://www.erlang.org/doc/man/ets.html#update_counter-3]), but
- %% will fail if the type of object referred to by Key is not a counter.
- %%
- %% Aggregated counters with the same name will be updated automatically.
- %% The `UpdateOp' patterns are the same as for `ets:update_counter/3', except
- %% that the position is omitted; in gproc, the value position is always `3'.
- %% @end
- %%
- update_shared_counter(Key, Incr) ->
- ?CATCH_GPROC_ERROR(update_shared_counter1(Key, Incr), [Key, Incr]).
- update_shared_counter1({c,g,_} = Key, Incr) ->
- ?CHK_DIST,
- gproc_dist:update_shared_counter(Key, Incr);
- update_shared_counter1({c,l,_} = Key, Incr) ->
- gproc_lib:update_counter(Key, Incr, shared).
- %% @spec (From::key(), To::pid() | key()) -> undefined | pid()
- %%
- %% @doc Atomically transfers the key `From' to the process identified by `To'.
- %%
- %% This function transfers any gproc key (name, property, counter, aggr counter)
- %% from one process to another, and returns the pid of the new owner.
- %%
- %% `To' must be either a pid or a unique name (name or aggregated counter), but
- %% does not necessarily have to resolve to an existing process. If there is
- %% no process registered with the `To' key, `give_away/2' returns `undefined',
- %% and the `From' key is effectively unregistered.
- %%
- %% It is allowed to give away a key to oneself, but of course, this operation
- %% will have no effect.
- %%
- %% Fails with `badarg' if the calling process does not have a `From' key
- %% registered.
- %% @end
- give_away(Key, ToPid) ->
- ?CATCH_GPROC_ERROR(give_away1(Key, ToPid), [Key, ToPid]).
- give_away1({_,l,_} = Key, ToPid) when is_pid(ToPid), node(ToPid) == node() ->
- call({give_away, Key, ToPid});
- give_away1({_,l,_} = Key, {n,l,_} = ToKey) ->
- call({give_away, Key, ToKey});
- give_away1({_,g,_} = Key, To) ->
- ?CHK_DIST,
- gproc_dist:give_away(Key, To).
- %% @spec () -> ok
- %%
- %% @doc Unregister all items of the calling process and inform gproc
- %% to forget about the calling process.
- %%
- %% This function is more efficient than letting gproc perform these
- %% cleanup operations.
- %% @end
- goodbye() ->
- process_is_down(self()).
- %% @spec (Key::process() | key(), Msg::any()) -> Msg
- %%
- %% @doc Sends a message to the process, or processes, corresponding to Key.
- %%
- %% If Key belongs to a unique object (name or aggregated counter), this
- %% function will send a message to the corresponding process, or fail if there
- %% is no such process. If Key is for a non-unique object type (counter or
- %% property), Msg will be send to all processes that have such an object.
- %%
- %% Key can also be anything that the erlang:send/2, or '!' operator accepts as a process
- %% identifier, namely a pid(), an atom(), or `{Name::atom(), Node::atom()}'.
- %% @end
- %%
- send(P, Msg) when is_pid(P); is_atom(P) ->
- P ! Msg;
- send({Name, Node} = P, Msg) when is_atom(Node), is_atom(Name) ->
- P ! Msg;
- send(Key, Msg) ->
- ?CATCH_GPROC_ERROR(send1(Key, Msg), [Key, Msg]).
- send1({T,C,_} = Key, Msg) when C==l; C==g ->
- if T == n orelse T == a orelse T == rc ->
- case ets:lookup(?TAB, {Key, T}) of
- [{_, Pid, _}] ->
- Pid ! Msg;
- _ ->
- ?THROW_GPROC_ERROR(badarg)
- end;
- T==p orelse T==c orelse T==r ->
- %% BUG - if the key part contains select wildcards, we may end up
- %% sending multiple messages to the same pid
- lists:foreach(fun(Pid) ->
- Pid ! Msg
- end, lookup_pids(Key)),
- Msg;
- true ->
- erlang:error(badarg)
- end;
- send1(_, _) ->
- ?THROW_GPROC_ERROR(badarg).
- %% @spec (Key::key(), Msg::any()) -> Msg
- %%
- %% @equiv bcast(nodes(), Key, Msg)
- %% @end
- %%
- bcast(Key, Msg) ->
- bcast(nodes(), Key, Msg).
- %% @spec (Nodes::[atom()], Key::key(), Msg::any()) -> Msg
- %%
- %% @doc Sends a message to processes corresponding to Key on Nodes.
- %%
- %% This function complements `send/2' and works on locally registered resources
- %% that `send/2' supports. Messages are routed via a special broadcast server
- %% on each node to ensure that ordering is preserved. Distributed delivery
- %% is asynchronous and carries the same guarantees as normal message passing
- %% (with the added proviso that the broadcast server also needs to be available).
- %% @see send/2
- %% @end
- %%
- bcast(Ns, Key, Msg) ->
- ?CATCH_GPROC_ERROR(bcast1(Ns, Key, Msg), [Key, Msg]).
- bcast1(Ns, {T,l,_} = Key, Msg) when T==p; T==a; T==c; T==n; T==r; T==rc ->
- send1(Key, Msg),
- gen_server:abcast(Ns -- [node()], gproc_bcast, {send, Key, Msg}),
- Msg.
- %% @spec (Context :: context()) -> key() | '$end_of_table'
- %%
- %% @doc Behaves as ets:first(Tab) for a given type of registration.
- %%
- %% See [http://www.erlang.org/doc/man/ets.html#first-1].
- %% The registry behaves as an ordered_set table.
- %% @end
- %%
- first(Context) ->
- {S, T} = get_s_t(Context),
- {HeadPat,_} = headpat({S, T}, '_', '_', '_'),
- case ets:select(?TAB, [{HeadPat,[],[{element,1,'$_'}]}], 1) of
- {[First], _} ->
- First;
- _ ->
- '$end_of_table'
- end.
- %% @spec (Context :: context()) -> key() | '$end_of_table'
- %%
- %% @doc Behaves as ets:last(Tab) for a given type of registration.
- %%
- %% See [http://www.erlang.org/doc/man/ets.html#last-1].
- %% The registry behaves as an ordered_set table.
- %% @end
- %%
- last(Context) ->
- {S, T} = get_s_t(Context),
- S1 = if S == '_'; S == l -> m; % 'm' comes after 'l'
- S == g -> h % 'h' comes between 'g' & 'l'
- end,
- Beyond = {{T,S1,[]},[]},
- step(ets:prev(?TAB, Beyond), S, T).
- %% @spec (Context::context(), Key::key()) -> key() | '$end_of_table'
- %%
- %% @doc Behaves as ets:next(Tab,Key) for a given type of registration.
- %%
- %% See [http://www.erlang.org/doc/man/ets.html#next-2].
- %% The registry behaves as an ordered_set table.
- %% @end
- %%
- next(Context, K) ->
- {S,T} = get_s_t(Context),
- {Prev,Unwrap} =
- case K of
- {{_,_,_},_} ->
- {K, false};
- {_,_,_} ->
- {{K,[]}, true} % [] is higher than pid(), shared, p, c...
- end,
- unwrap(Unwrap, step(ets:next(?TAB,Prev), S, T)).
- unwrap(true, {{_,_,_} = R,_}) ->
- R;
- unwrap(_, R) ->
- R.
- %% @spec (Context::context(), Key::key()) -> key() | '$end_of_table'
- %%
- %% @doc Behaves as ets:prev(Tab,Key) for a given type of registration.
- %%
- %% See [http://www.erlang.org/doc/man/ets.html#prev-2].
- %% The registry behaves as an ordered_set table.
- %% @end
- %%
- prev(Context, K) ->
- {S, T} = get_s_t(Context),
- {Prev, Unwrap} =
- case K of
- {{_,_,_},_} -> {K, false};
- {_,_,_} ->
- {{K,1}, true}
- end,
- unwrap(Unwrap, step(ets:prev(?TAB, Prev), S, T)).
- step(Key, '_', '_') ->
- case Key of
- {{_,_,_},_} -> Key;
- _ -> '$end_of_table'
- end;
- step(Key, '_', T) ->
- case Key of
- {{T,_,_},_} -> Key;
- _ -> '$end_of_table'
- end;
- step(Key, S, '_') ->
- case Key of
- {{_, S, _}, _} -> Key;
- _ -> '$end_of_table'
- end;
- step(Key, S, T) ->
- case Key of
- {{T, S, _}, _} -> Key;
- _ -> '$end_of_table'
- end.
- %% @spec (Pid::pid()) -> ProcessInfo
- %% ProcessInfo = [{gproc, [{Key,Value}]} | ProcessInfo]
- %%
- %% @doc Similar to `process_info(Pid)' but with additional gproc info.
- %%
- %% Returns the same information as process_info(Pid), but with the
- %% addition of a `gproc' information item, containing the `{Key,Value}'
- %% pairs registered to the process.
- %% @end
- info(Pid) when is_pid(Pid) ->
- Items = [?MODULE | [ I || {I,_} <- process_info(self())]],
- [info(Pid,I) || I <- Items].
- %% @spec (Pid::pid(), Item::atom()) -> {Item, Info}
- %%
- %% @doc Similar to process_info(Pid, Item), but with additional gproc info.
- %%
- %% For `Item = gproc', this function returns a list of `{Key, Value}' pairs
- %% registered to the process Pid. For other values of Item, it returns the
- %% same as [http://www.erlang.org/doc/man/erlang.html#process_info-2].
- %% @end
- info(Pid, gproc) ->
- gproc_info(Pid, '_');
- info(Pid, {gproc, Pat}) ->
- gproc_info(Pid, Pat);
- info(Pid, current_function) ->
- {_, T} = process_info(Pid, backtrace),
- info_cur_f(T, process_info(Pid, current_function));
- info(Pid, I) ->
- process_info(Pid, I).
- %% We don't want to return the internal gproc:info() function as current
- %% function, so we grab the 'backtrace' and extract the call stack from it,
- %% filtering out the functions gproc:info/_ and gproc:'-info/1-lc...' entries.
- %%
- %% This is really an indication that wrapping the process_info() BIF was a
- %% bad idea to begin with... :P
- %%
- info_cur_f(T, Default) ->
- {match, Matches} = re:run(T,<<"\\(([^\\)]+):(.+)/([0-9]+)">>,
- [global,{capture,[1,2,3],list}]),
- case lists:dropwhile(fun(["gproc","info",_]) -> true;
- (["gproc","'-info/1-lc" ++ _, _]) -> true;
- (["erlang","process_info", _]) -> true;
- (_) -> false
- end, Matches) of
- [] ->
- Default;
- [[M,F,A]|_] ->
- {current_function,
- {to_atom(M), to_atom(F), list_to_integer(A)}}
- end.
- to_atom(S) ->
- case erl_scan:string(S) of
- {ok, [{atom,_,A}|_],_} ->
- A;
- _ ->
- list_to_atom(S)
- end.
- gproc_info(Pid, Pat) ->
- Keys = ets:select(?TAB, [{ {{Pid,Pat}, '_'}, [], [{element,2,
- {element,1,'$_'}}] }]),
- {?MODULE, lists:zf(
- fun(K) ->
- try V = get_value(K, Pid),
- {true, {K,V}}
- catch
- error:_ ->
- false
- end
- end, Keys)}.
- %% @spec () -> ok
- %%
- %% @doc Similar to the built-in shell command `i()' but inserts information
- %% about names and properties registered in Gproc, where applicable.
- %% @end
- i() ->
- gproc_info:i().
- %%% ==========================================================
- %% @hidden
- handle_cast({monitor_me, Pid}, S) ->
- erlang:monitor(process, Pid),
- {noreply, S};
- handle_cast({audit_process, Pid}, S) ->
- case is_process_alive(Pid) of
- false ->
- process_is_down(Pid);
- true ->
- ignore
- end,
- {noreply, S};
- handle_cast({cancel_wait, Pid, {T,_,_} = Key, Ref}, S) ->
- _ = case ets:lookup(?TAB, {Key,T}) of
- [{_, Waiters}] ->
- gproc_lib:remove_wait(Key, Pid, Ref, Waiters);
- _ ->
- ignore
- end,
- {noreply, S};
- handle_cast({cancel_wait_or_monitor, Pid, {T,_,_} = Key}, S) ->
- _ = case ets:lookup(?TAB, {Key, T}) of
- [{_, Waiters}] ->
- gproc_lib:remove_wait(Key, Pid, all, Waiters);
- [{_, OtherPid, _}] ->
- gproc_lib:remove_monitors(Key, OtherPid, Pid);
- _ ->
- ok
- end,
- {noreply, S}.
- %% @hidden
- handle_call({reg, {_T,l,_} = Key, Val, Attrs, Op}, {Pid,_}, S) ->
- handle_reg_call(Key, Pid, Val, Attrs, Op, S);
- handle_call({reg_other, {_T,l,_} = Key, Pid, Val, Attrs, Op}, _, S) ->
- handle_reg_call(Key, Pid, Val, Attrs, Op, S);
- handle_call({set_attributes, {_,l,_} = Key, Attrs}, {Pid,_}, S) ->
- case gproc_lib:insert_attr(Key, Attrs, Pid, l) of
- false -> {reply, badarg, S};
- L when is_list(L) ->
- {reply, true, S}
- end;
- handle_call({set_attributes_shared, {_,l,_} = Key, Attrs}, _, S) ->
- case gproc_lib:insert_attr(Key, Attrs, shared, l) of
- false ->
- {reply, badarg, S};
- L when is_list(L) ->
- {reply, true, S}
- end;
- handle_call({reg_or_locate, {T,l,_} = Key, Val, P}, _, S) ->
- Reg = fun() ->
- Pid = if is_function(P, 0) ->
- spawn(P);
- is_pid(P) ->
- P
- end,
- true = gproc_lib:insert_reg(Key, Val, Pid, l),
- _ = gproc_lib:ensure_monitor(Pid, l),
- {reply, {Pid, Val}, S}
- end,
- case ets:lookup(?TAB, {Key, T}) of
- [] ->
- Reg();
- [{_, _Waiters}] ->
- Reg();
- [{_, OtherPid, OtherValue}] ->
- {reply, {OtherPid, OtherValue}, S}
- end;
- handle_call({monitor, {T,l,_} = Key, Pid, Type}, _From, S)
- when T==n; T==a; T==rc ->
- Ref = make_ref(),
- Lookup = ets:lookup(?TAB, {Key, T}),
- IsRegged = is_regged(Lookup),
- _ = case {IsRegged, Type} of
- {false, info} ->
- Pid ! {gproc, unreg, Ref, Key};
- {false, follow} ->
- Pid ! {gproc, unreg, Ref, Key},
- _ = gproc_lib:ensure_monitor(Pid, l),
- case Lookup of
- [{K, Waiters}] ->
- NewWaiters = [{Pid,Ref,follow}|Waiters],
- ets:insert(?TAB, {K, NewWaiters}),
- ets:insert_new(?TAB, {{Pid,Key}, []});
- [] ->
- ets:insert(?TAB, {{Key,T}, [{Pid,Ref,follow}]}),
- ets:insert_new(?TAB, {{Pid,Key}, []})
- end;
- {false, standby} ->
- Evt = {failover, Pid},
- true = gproc_lib:insert_reg(Key, undefined, Pid, l, Evt),
- Pid ! {gproc, Evt, Ref, Key},
- _ = gproc_lib:ensure_monitor(Pid, l);
- {true, _} ->
- [{_, RegPid, _}] = Lookup,
- _ = gproc_lib:ensure_monitor(Pid, l),
- case ets:lookup(?TAB, {RegPid, Key}) of
- [{K,r}] ->
- ets:insert(?TAB, {K, [{monitor, [{Pid,Ref,Type}]}]}),
- ets:insert_new(?TAB, {{Pid,Key}, []});
- [{K, Opts}] ->
- ets:insert(?TAB, {K, gproc_lib:add_monitor(
- Opts, Pid, Ref, Type)}),
- ets:insert_new(?TAB, {{Pid,Key}, []})
- end
- end,
- {reply, Ref, S};
- handle_call({demonitor, {T,l,_} = Key, Ref, Pid}, _From, S)
- when T==n; T==a; T==rc ->
- _ = case ets:lookup(?TAB, {Key, T}) of
- [] ->
- ok; % be nice
- [{_, Waiters}] ->
- case lists:filter(fun({P, R, _}) ->
- P =/= Pid orelse R =/= Ref
- end, Waiters) of
- [] ->
- ets:delete(?TAB, {Pid, Key}),
- ets:delete(?TAB, {Key, T});
- NewWaiters ->
- case lists:keymember(Pid, 1, NewWaiters) of
- true ->
- ok;
- false ->
- ets:delete(?TAB, {Pid, Key})
- end,
- ets:insert(?TAB, {{Key, T}, Waiters})
- end;
- [{_, RegPid, _}] ->
- case ets:lookup(?TAB, {RegPid, Key}) of
- [{_K,r}] ->
- ok; % be nice
- [{K, Opts}] ->
- Opts1 = gproc_lib:remove_monitor(Opts, Pid, Ref),
- ets:insert(?TAB, {K, Opts1}),
- case gproc_lib:does_pid_monitor(Pid, Opts1) of
- true ->
- ok;
- false ->
- ets:delete(?TAB, {Pid, Key})
- end
- end
- end,
- {reply, ok, S};
- handle_call({reg_shared, {_T,l,_} = Key, Val, Attrs, Op}, _From, S) ->
- case try_insert_reg(Key, Val, shared) of
- true ->
- _ = if Attrs =/= [] ->
- gproc_lib:insert_attr(Key, Attrs, shared, l);
- true -> true
- end,
- {reply, true, S};
- already_registered when Op == ensure ->
- case gproc_lib:do_set_value(Key, Val, shared) of
- true ->
- {reply, updated, S};
- false ->
- {reply, badarg, S}
- end;
- _ ->
- {reply, badarg, S}
- end;
- handle_call({unreg, {_,l,_} = Key}, {Pid,_}, S) ->
- handle_unreg_call(Key, Pid, S);
- handle_call({unreg_other, {_,l,_} = Key, Pid}, _, S) ->
- handle_unreg_call(Key, Pid, S);
- handle_call({unreg_shared, {_,l,_} = Key}, _, S) ->
- _ = case ets:lookup(?TAB, {shared, Key}) of
- [{_, r}] ->
- _ = gproc_lib:remove_reg(Key, shared, unreg, []);
- [{_, Opts}] ->
- _ = gproc_lib:remove_reg(Key, shared, unreg, Opts);
- [] ->
- %% don't crash if shared key already unregged.
- ok
- end,
- {reply, true, S};
- handle_call({await, {_,l,_} = Key, Pid}, From, S) ->
- %% Passing the pid explicitly is needed when leader_call is used,
- %% since the Pid given as From in the leader is the local gen_leader
- %% instance on the calling node.
- case gproc_lib:await(Key, Pid, From) of
- noreply ->
- {noreply, S};
- {reply, Reply, _} ->
- {reply, Reply, S}
- end;
- handle_call({mreg, T, l, L}, {Pid,_}, S) ->
- try gproc_lib:insert_many(T, l, L, Pid) of
- {true,_} -> {reply, true, S};
- false -> {reply, badarg, S}
- catch
- throw:?GPROC_THROW(_) ->
- {reply, badarg, S};
- error:_ ->
- {reply, badarg, S}
- end;
- handle_call({munreg, T, l, L}, {Pid,_}, S) ->
- _ = gproc_lib:remove_many(T, l, L, Pid),
- {reply, true, S};
- handle_call({set, {_,l,_} = Key, Value}, {Pid,_}, S) ->
- case gproc_lib:do_set_value(Key, Value, Pid) of
- true ->
- {reply, true, S};
- false ->
- {reply, badarg, S}
- end;
- handle_call({set_shared, {_,l,_} = Key, Value}, {_,_}, S) ->
- case gproc_lib:do_set_value(Key, Value, shared) of
- true ->
- {reply, true, S};
- false ->
- {reply, badarg, S}
- end;
- handle_call({audit_process, Pid}, _, S) ->
- _ = case is_process_alive(Pid) of
- false ->
- process_is_down(Pid);
- true ->
- ignore
- end,
- {reply, ok, S};
- handle_call({give_away, Key, To}, {Pid,_}, S) ->
- Reply = do_give_away(Key, To, Pid),
- {reply, Reply, S};
- handle_call(_, _, S) ->
- {reply, badarg, S}.
- %% @hidden
- handle_info({'DOWN', _MRef, process, Pid, _}, S) ->
- _ = process_is_down(Pid),
- {noreply, S};
- handle_info(_, S) ->
- {noreply, S}.
- %% @hidden
- code_change(_FromVsn, S, _Extra) ->
- %% We have changed local monitor markers from {Pid} to {Pid,l}.
- _ = case ets:select(?TAB, [{{'$1'},[],['$1']}]) of
- [] ->
- ok;
- Pids ->
- ets:insert(?TAB, [{P,l} || P <- Pids]),
- ets:select_delete(?TAB, [{{'_'},[],[true]}])
- end,
- {ok, S}.
- %% @hidden
- terminate(_Reason, _S) ->
- ok.
- %% handle_call body common to reg and reg_other.
- %%
- handle_reg_call(Key, Pid, Val, Attrs, Op, S) ->
- case try_insert_reg(Key, Val, Pid) of
- true ->
- _ = gproc_lib:ensure_monitor(Pid,l),
- _ = set_attrs(Attrs, Key, Pid),
- {reply, regged_new(Op), S};
- already_registered when Op == ensure ->
- case gproc_lib:do_set_value(Key, Val, Pid) of
- true ->
- _ = set_attrs(Attrs, Key, Pid),
- {reply, updated, S};
- false ->
- %% actually pretty bad, if it ever happens
- {reply, badarg, S}
- end;
- already_registered ->
- {reply, badarg, S};
- false ->
- {reply, badarg, S}
- end.
- set_attrs([], _, _) ->
- true;
- set_attrs([_|_] = Attrs, Key, Pid) ->
- gproc_lib:insert_attr(Key, Attrs, Pid, l).
- handle_unreg_call(Key, Pid, S) ->
- case ets:lookup(?TAB, {Pid,Key}) of
- [{_, r}] ->
- _ = gproc_lib:remove_reg(Key, Pid, unreg, []),
- {reply, true, S};
- [{_, Opts}] when is_list(Opts) ->
- _ = gproc_lib:remove_reg(Key, Pid, unreg, Opts),
- {reply, true, S};
- [] ->
- {reply, badarg, S}
- end.
- call(Req) ->
- call(Req, l).
- call(Req, l) ->
- chk_reply(gen_server:call(?MODULE, Req));
- call(Req, g) ->
- chk_reply(gproc_dist:leader_call(Req)).
- call(N, Req, l) ->
- chk_reply(gen_server:call({?MODULE, N}, Req));
- call(undefined, Req, g) ->
- %% we always call the leader
- chk_reply(gproc_dist:leader_call(Req)).
- chk_reply(Reply) ->
- case Reply of
- badarg -> ?THROW_GPROC_ERROR(badarg);
- _ -> Reply
- end.
- cast(Msg) ->
- cast(Msg, l).
- cast(Msg, l) ->
- gen_server:cast(?MODULE, Msg);
- cast(Msg, g) ->
- gproc_dist:leader_cast(Msg).
- cast(N, Msg, l) ->
- gen_server:cast({?MODULE, N}, Msg).
- try_insert_reg({T,l,_} = Key, Val, Pid) ->
- case gproc_lib:insert_reg(Key, Val, Pid, l) of
- false ->
- case ets:lookup(?TAB, {Key,T}) of
- %% In this particular case, the lookup cannot result in
- %% [{_, Waiters}], since the insert_reg/4 function would
- %% have succeeded then.
- [{_, Pid, _}] ->
- already_registered;
- [{_, OtherPid, _}] ->
- case is_process_alive(OtherPid) of
- true ->
- false;
- false ->
- process_is_down(OtherPid), % may result in failover
- try_insert_reg(Key, Val, Pid)
- end;
- [] ->
- false
- end;
- true ->
- true
- end.
- %% try_insert_shared({c,l,_} = Key, Val) ->
- %% ets:insert_new(?TAB, [{{Key,shared}, shared, Val}, {{shared, Key}, []}]);
- %% try_insert_shared({a,l,_} = Key, Val) ->
- %% ets:insert_new(?TAB, [{{Key, a}, shared, Val}, {{shared, Key}, []}]).
- -spec audit_process(pid()) -> ok.
- audit_process(Pid) when is_pid(Pid) ->
- ok = gen_server:call(gproc, {audit_process, Pid}, infinity).
- nb_audit_process(Pid) when is_pid(Pid) ->
- ok = gen_server:cast(gproc, {audit_process, Pid}).
- -spec process_is_down(pid()) -> ok.
- process_is_down(Pid) when is_pid(Pid) ->
- %% delete the monitor marker
- %% io:fwrite(user, "process_is_down(~p) - ~p~n", [Pid,ets:tab2list(?TAB)]),
- Marker = {Pid,l},
- case ets:member(?TAB, Marker) of
- false ->
- ok;
- true ->
- Revs = ets:select(?TAB, [{{{Pid,'$1'}, '$2'},
- [{'==',{element,2,'$1'},l}],
- [{{'$1','$2'}}]}]),
- lists:foreach(
- fun({{n,l,_}=K, R}) ->
- Key = {K,n},
- case ets:lookup(?TAB, Key) of
- [{_, Pid, V}] ->
- ets:delete(?TAB, Key),
- opt_notify(R, K, Pid, V);
- [{_, Waiters}] ->
- case [W || W <- Waiters,
- element(1,W) =/= Pid] of
- [] ->
- ets:delete(?TAB, Key);
- Waiters1 ->
- ets:insert(?TAB, {Key, Waiters1})
- end;
- [{_, OtherPid, _}] when Pid =/= OtherPid ->
- case ets:lookup(?TAB, {OtherPid, K}) of
- [{RK, Opts}] when is_list(Opts) ->
- Opts1 = gproc_lib:remove_monitor_pid(
- Opts, Pid),
- ets:insert(?TAB, {RK, Opts1});
- _ ->
- true
- end;
- [] ->
- true
- end;
- ({{c,l,C} = K, _}) ->
- Key = {K, Pid},
- [{_, _, Value}] = ets:lookup(?TAB, Key),
- ets:delete(?TAB, Key),
- gproc_lib:update_aggr_counter(l, C, -Value);
- ({{r,l,Rsrc} = K, _}) ->
- Key = {K, Pid},
- ets:delete(?TAB, Key),
- gproc_lib:decrement_resource_count(l, Rsrc);
- ({{rc,l,_} = K, R}) ->
- remove_aggregate(rc, K, R, Pid);
- ({{a,l,_} = K, R}) ->
- remove_aggregate(a, K, R, Pid);
- ({{p,_,_} = K, _}) ->
- ets:delete(?TAB, {K, Pid})
- end, Revs),
- ets:select_delete(?TAB, [{{{Pid,{'_',l,'_'}},'_'}, [], [true]}]),
- ets:delete(?TAB, Marker),
- ok
- end.
- remove_aggregate(T, K, R, Pid) ->
- case ets:lookup(?TAB, {K,T}) of
- [{_, Pid, V}] ->
- ets:delete(?TAB, {K,T}),
- opt_notify(R, K, Pid, V);
- [{_, OtherPid, _}] when Pid =/= OtherPid ->
- case ets:lookup(?TAB, {OtherPid, K}) of
- [{RK, Opts}] when is_list(Opts) ->
- Opts1 = gproc_lib:remove_monitor_pid(
- Opts, Pid),
- ets:insert(?TAB, {RK, Opts1});
- _ ->
- true
- end;
- [] ->
- opt_notify(R, K, Pid, undefined)
- end.
- opt_notify(r, _, _, _) ->
- ok;
- opt_notify(Opts, {T,_,_} = Key, Pid, Value) ->
- case gproc_lib:standbys(Opts) of
- [] ->
- keep_followers(Opts, Key),
- gproc_lib:notify(unreg, Key, Opts);
- SBs ->
- case pick_standby(SBs) of
- false ->
- keep_followers(Opts, Key),
- gproc_lib:notify(unreg, Key, Opts),
- ok;
- {ToPid, Ref} ->
- ets:insert(?TAB, [{{Key,T}, ToPid, Value},
- {{ToPid, Key},
- gproc_lib:remove_monitor(
- Opts, ToPid, Ref)}]),
- _ = gproc_lib:remove_reverse_mapping(
- {failover,ToPid}, Pid, Key),
- _ = gproc_lib:ensure_monitor(ToPid, l),
- ok
- end
- end.
- keep_followers(Opts, {T,_,_} = Key) ->
- case gproc_lib:followers(Opts) of
- [] ->
- ok;
- [_|_] = F ->
- ets:insert(?TAB, {{Key,T}, F})
- end.
- pick_standby([{Pid, Ref, standby}|T]) when node(Pid) =:= node() ->
- case is_process_alive(Pid) of
- true ->
- {Pid, Ref};
- false ->
- pick_standby(T)
- end;
- pick_standby([_|T]) ->
- pick_standby(T);
- pick_standby([]) ->
- false.
- do_give_away({T,l,_} = K, To, Pid) when T==n; T==a; T==rc ->
- Key = {K, T},
- case ets:lookup(?TAB, Key) of
- [{_, Pid, Value}] ->
- %% Pid owns the reg; allowed to give_away
- case pid_to_give_away_to(To) of
- Pid ->
- %% Give away to ourselves? Why not? We'll allow it,
- %% but nothing needs to be done.
- Pid;
- ToPid when is_pid(ToPid) ->
- ets:insert(?TAB, [{Key, ToPid, Value},
- {{ToPid, K}, []}]),
- _ = gproc_lib:remove_reverse_mapping({migrated,ToPid}, Pid, K),
- _ = gproc_lib:ensure_monitor(ToPid, l),
- ToPid;
- undefined ->
- _ = gproc_lib:remove_reg(K, Pid, unreg),
- undefined
- end;
- _ ->
- badarg
- end;
- do_give_away({T,l,_} = K, To, Pid) when T==c; T==p; T==r ->
- Key = {K, Pid},
- case ets:lookup(?TAB, Key) of
- [{_, Pid, Value}] ->
- case pid_to_give_away_to(To) of
- ToPid when is_pid(ToPid) ->
- ToKey = {K, ToPid},
- case ets:member(?TAB, ToKey) of
- true ->
- badarg;
- false ->
- ets:insert(?TAB, [{ToKey, ToPid, Value},
- {{ToPid, K}, []}]),
- ets:delete(?TAB, {Pid, K}),
- ets:delete(?TAB, Key),
- _ = gproc_lib:ensure_monitor(ToPid, l),
- ToPid
- end;
- undefined ->
- _ = gproc_lib:remove_reg(K, Pid, {migrated, undefined}),
- undefined
- end;
- _ ->
- badarg
- end.
- pid_to_give_away_to(P) when is_pid(P), node(P) == node() ->
- P;
- pid_to_give_away_to({T,l,_} = Key) when T==n; T==a; T==rc ->
- case ets:lookup(?TAB, {Key, T}) of
- [{_, Pid, _}] ->
- Pid;
- _ ->
- undefined
- end.
- create_tabs() ->
- Opts = gproc_lib:valid_opts(ets_options, [{write_concurrency,true},
- {read_concurrency, true}]),
- case ets:info(?TAB, name) of
- undefined ->
- ets:new(?TAB, [ordered_set, public, named_table | Opts]);
- _ ->
- ok
- end.
- %% @hidden
- init([]) ->
- set_monitors(),
- {ok, #state{}}.
- set_monitors() ->
- set_monitors(ets:select(?TAB, [{{{'$1',l}},[],['$1']}], 100)).
- set_monitors('$end_of_table') ->
- ok;
- set_monitors({Pids, Cont}) ->
- _ = [erlang:monitor(process,Pid) || Pid <- Pids],
- set_monitors(ets:select(Cont)).
- monitor_me() ->
- case ets:insert_new(?TAB, {{self(),l}}) of
- false -> true;
- true ->
- cast({monitor_me,self()}),
- true
- end.
- pattern([{'_', Gs, As}], T) ->
- ?l,
- {HeadPat, Vs} = headpat(T, '$1', '$2', '$3'),
- [{HeadPat, rewrite(Gs,Vs), rewrite(As,Vs)}];
- pattern([{{A,B,C},Gs,As}], Scope) ->
- ?l,
- {HeadPat, Vars} = headpat(Scope, A,B,C),
- [{HeadPat, rewrite(Gs,Vars), rewrite(As,Vars)}];
- pattern([{Head, Gs, As}], Scope) ->
- ?l,
- {S, T} = get_s_t(Scope),
- case is_var(Head) of
- {true,_N} ->
- HeadPat = {{{T,S,'_'},'_'},'_','_'},
- Vs = [{Head, obj_prod()}],
- %% the headpat function should somehow verify that Head is
- %% consistent with Scope (or should we add a guard?)
- [{HeadPat, rewrite(Gs, Vs), rewrite(As, Vs)}];
- false ->
- erlang:error(badarg)
- end.
- %% This is the expression to use in guards and the RHS to address the whole
- %% object, in its logical representation.
- obj_prod() ->
- {{ {element,1,{element,1,'$_'}},
- {element,2,'$_'},
- {element,3,'$_'} }}.
- obj_prod_l() ->
- [ {element,1,{element,1,'$_'}},
- {element,2,'$_'},
- {element,3,'$_'} ].
- headpat({S, T}, V1,V2,V3) ->
- headpat(type(T), scope(S), V1,V2,V3);
- headpat(T, V1, V2, V3) when is_atom(T) ->
- headpat(type(T), l, V1, V2, V3);
- headpat(_, _, _, _) -> erlang:error(badarg).
- headpat(T, C, V1,V2,V3) ->
- Rf = fun(Pos) ->
- {element,Pos,{element,1,{element,1,'$_'}}}
- end,
- K2 = if T==n orelse T==a orelse T==rc -> T;
- true -> '_'
- end,
- {Kp,Vars} = case V1 of
- {Vt,Vc,Vn} ->
- ?l,
- {T1,Vs1} = subst(T,Vt,fun() -> Rf(1) end,[]),
- {C1,Vs2} = subst(C,Vc,fun() -> Rf(2) end,Vs1),
- {{T1,C1,Vn}, Vs2};
- '_' ->
- ?l,
- {{T,C,'_'}, []};
- _ ->
- ?l,
- case is_var(V1) of
- {true,_} ->
- {{T,C,V1}, [{V1, {element,1,
- {element,1,'$_'}}}]};
- false ->
- erlang:error(badarg)
- end
- end,
- {{{Kp,K2},V2,V3}, Vars}.
- %% l(L) -> L.
- subst(X, '_', _F, Vs) ->
- {X, Vs};
- subst(X, V, F, Vs) ->
- case is_var(V) of
- {true,_} ->
- {X, [{V,F()}|Vs]};
- false ->
- {V, Vs}
- end.
- scope('_') -> '_';
- scope(all) -> '_';
- scope(global) -> g;
- scope(local) -> l;
- scope(S) when S==l; S==g -> S.
- type('_') -> '_';
- type(all) -> '_';
- type(T) when T==n; T==p; T==c; T==a; T==r; T==rc -> T;
- type(names) -> n;
- type(props) -> p;
- type(resources) -> r;
- type(counters) -> c;
- type(aggr_counters) -> a;
- type(resource_counters) -> rc.
- rev_keypat(Context) ->
- {S,T} = get_s_t(Context),
- {T,S,'_'}.
- get_s_t({S,T}) -> {scope(S), type(T)};
- get_s_t(T) when is_atom(T) ->
- {scope(all), type(T)}.
- is_var('$1') -> {true,1};
- is_var('$2') -> {true,2};
- is_var('$3') -> {true,3};
- is_var('$4') -> {true,4};
- is_var('$5') -> {true,5};
- is_var('$6') -> {true,6};
- is_var('$7') -> {true,7};
- is_var('$8') -> {true,8};
- is_var('$9') -> {true,9};
- is_var(X) when is_atom(X) ->
- case atom_to_list(X) of
- "\$" ++ Tl ->
- try N = list_to_integer(Tl),
- {true,N}
- catch
- error:_ ->
- false
- end;
- _ ->
- false
- end;
- is_var(_) -> false.
- rewrite(Gs, R) ->
- [rewrite1(G, R) || G <- Gs].
- rewrite1('$_', _) ->
- obj_prod();
- rewrite1('$$', _) ->
- obj_prod_l();
- rewrite1(Guard, R) when is_tuple(Guard) ->
- list_to_tuple([rewrite1(G, R) || G <- tuple_to_list(Guard)]);
- rewrite1(Exprs, R) when is_list(Exprs) ->
- [rewrite1(E, R) || E <- Exprs];
- rewrite1(V, R) when is_atom(V) ->
- case is_var(V) of
- {true,_N} ->
- case lists:keysearch(V, 1, R) of
- {value, {_, V1}} ->
- V1;
- false ->
- V
- end;
- false ->
- V
- end;
- rewrite1(Expr, _) ->
- Expr.
- %% @spec () -> any()
- %%
- %% @doc
- %% @equiv table({all, all})
- %% @end
- table() ->
- table({all, all}).
- %% @spec (Context::context()) -> any()
- %%
- %% @doc
- %% @equiv table(Context, [])
- %% @end
- %%
- table(Context) ->
- table(Context, []).
- %% @spec (Context::context(), Opts) -> any()
- %%
- %% @doc QLC table generator for the gproc registry.
- %% Context specifies which subset of the registry should be queried.
- %% See [http://www.erlang.org/doc/man/qlc.html].
- %%
- %% NOTE: By default, the gproc table generator will not filter out entries
- %% belonging to processes that have just died, but which have yet to be cleared
- %% out of the registry. Use the option `check_pids' (or `{check_pids, true}')
- %% if you want to filter out dead entries already in the query. There will be
- %% some overhead associated with doing so, and given that the process monitoring
- %% is asynchronous, there can never be any guarantee that there are no dead
- %% entries in the list by the time your program processes it.
- %%
- %% @end
- table(Context, Opts) ->
- Ctxt = {_, Type} = get_s_t(Context),
- [Traverse, NObjs] = [proplists:get_value(K,Opts,Def) ||
- {K,Def} <- [{traverse,select}, {n_objects,100}]],
- CheckPids = proplists:get_bool(check_pids, Opts),
- TF = case Traverse of
- first_next ->
- fun() -> qlc_next(Ctxt, first(Ctxt), CheckPids) end;
- last_prev -> fun() -> qlc_prev(Ctxt, last(Ctxt), CheckPids) end;
- select ->
- fun(MS) -> qlc_select(
- CheckPids,
- select(Ctxt, wrap_qlc_ms_prod(CheckPids, MS),
- NObjs))
- end;
- {select,MS} ->
- fun() -> qlc_select(
- CheckPids,
- select(Ctxt, wrap_qlc_ms_prod(CheckPids, MS),
- NObjs))
- end;
- _ ->
- erlang:error(badarg, [Ctxt,Opts])
- end,
- InfoFun = fun(indices) -> [2];
- (is_unique_objects) -> is_unique(Type);
- (keypos) -> 1;
- (is_sorted_key) -> true;
- (num_of_objects) ->
- %% this is just a guesstimate.
- trunc(ets:info(?TAB,size) / 2.5)
- end,
- LookupFun =
- case Traverse of
- {select, _MS} -> undefined;
- _ -> fun(Pos, Ks) -> qlc_lookup(Ctxt, Pos, Ks, CheckPids) end
- end,
- qlc:table(TF, [{info_fun, InfoFun},
- {lookup_fun, LookupFun}] ++ [{K,V} || {K,V} <- Opts,
- K =/= traverse,
- K =/= n_objects]).
- wrap_qlc_ms_prod(false, Pats) ->
- Pats;
- wrap_qlc_ms_prod(true, Pats) ->
- [ wrap_qlc_ms_prod_(P) || P <- Pats ].
- wrap_qlc_ms_prod_({H, Gs, [P]}) ->
- {H, Gs, [{{ {element, 2, '$_'}, P }}]}.
- qlc_lookup(_Scope, 1, Keys, Check) ->
- lists:flatmap(
- fun(Key) ->
- remove_dead(
- Check,
- ets:select(?TAB, [{ {{Key,'_'},'_','_'}, [],
- [{{ {element,1,{element,1,'$_'}},
- {element,2,'$_'},
- {element,3,'$_'} }}] }]))
- end, Keys);
- qlc_lookup(Scope, 2, Pids, Check) ->
- lists:flatmap(fun(Pid) ->
- qlc_lookup_pid(Pid, Scope, Check)
- end, Pids).
- remove_dead(false, Objs) ->
- Objs;
- remove_dead(true, Objs) ->
- [ Reg || {_, Pid, _} = Reg <- Objs,
- not ?PID_IS_DEAD(Pid) ].
- %% While it may seem obsessive not to do the sensible pid liveness check here
- %% every time, we make it optional for consistency; this way, we can devise
- %% a test case that verifies the difference between having the option and not.
- qlc_lookup_pid(Pid, Scope, Check) ->
- case Check andalso ?PID_IS_DEAD(Pid) of
- true ->
- [];
- false ->
- Found =
- ets:select(?TAB, [{{{Pid, rev_keypat(Scope)}, '_'},
- [], ['$_']}]),
- lists:flatmap(
- fun({{_,{T,_,_}=K}, _}) ->
- K2 = if T==n orelse T==a orelse T==rc -> T;
- true -> Pid
- end,
- case ets:lookup(?TAB, {K,K2}) of
- [{{Key,_},_,Value}] ->
- [{Key, Pid, Value}];
- [] ->
- []
- end
- end, Found)
- end.
- qlc_next(_, '$end_of_table', _) -> [];
- qlc_next(Scope, K, Check) ->
- case ets:lookup(?TAB, K) of
- [{{Key,_}, Pid, V}] ->
- case Check andalso ?PID_IS_DEAD(Pid) of
- true ->
- qlc_next(Scope, next(Scope, K), Check);
- false ->
- [{Key,Pid,V}] ++ fun() ->
- qlc_next(Scope, next(Scope, K),
- Check)
- end
- end;
- [] ->
- qlc_next(Scope, next(Scope, K), Check)
- end.
- qlc_prev(_, '$end_of_table', _) -> [];
- qlc_prev(Scope, K, Check) ->
- case ets:lookup(?TAB, K) of
- [{{Key,_},Pid,V}] ->
- case Check andalso ?PID_IS_DEAD(Pid) of
- true ->
- qlc_prev(Scope, prev(Scope, K), Check);
- false ->
- [{Key,Pid,V}] ++ fun() ->
- qlc_prev(Scope, prev(Scope, K),
- Check)
- end
- end;
- [] ->
- qlc_prev(Scope, prev(Scope, K), Check)
- end.
- qlc_select(_, '$end_of_table') ->
- [];
- qlc_select(true, {Objects, Cont}) ->
- case [O || {Pid,O} <- Objects,
- not ?PID_IS_DEAD(Pid)] of
- [] ->
- %% re-run search
- qlc_select(true, ets:select(Cont));
- Found ->
- Found ++ fun() -> qlc_select(true, ets:select(Cont)) end
- end;
- qlc_select(false, {Objects, Cont}) ->
- Objects ++ fun() -> qlc_select(false, ets:select(Cont)) end.
- is_unique(n) -> true;
- is_unique(a) -> true;
- is_unique(_) -> false.
- is_regged([{_, _, _}]) ->
- true;
- is_regged(_) ->
- false.
|