12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325 |
- /**
- * \file
- * COM Interop Support
- *
- *
- * (C) 2002 Ximian, Inc. http://www.ximian.com
- *
- */
- #include "config.h"
- #include <glib.h>
- #ifdef HAVE_ALLOCA_H
- #include <alloca.h>
- #endif
- #include "object.h"
- #include "loader.h"
- #include "cil-coff.h"
- #include "metadata/abi-details.h"
- #include "metadata/cominterop.h"
- #include "metadata/marshal.h"
- #include "metadata/method-builder.h"
- #include "metadata/tabledefs.h"
- #include "metadata/exception.h"
- #include "metadata/appdomain.h"
- #include "metadata/reflection-internals.h"
- #include "mono/metadata/class-init.h"
- #include "mono/metadata/class-internals.h"
- #include "mono/metadata/debug-helpers.h"
- #include "mono/metadata/threads.h"
- #include "mono/metadata/monitor.h"
- #include "mono/metadata/metadata-internals.h"
- #include "mono/metadata/method-builder-ilgen-internals.h"
- #include "mono/metadata/domain-internals.h"
- #include "mono/metadata/gc-internals.h"
- #include "mono/metadata/threads-types.h"
- #include "mono/metadata/string-icalls.h"
- #include "mono/metadata/attrdefs.h"
- #include "mono/utils/mono-counters.h"
- #include "mono/utils/strenc.h"
- #include "mono/utils/atomic.h"
- #include "mono/utils/mono-error.h"
- #include "mono/utils/mono-error-internals.h"
- #include <string.h>
- #include <errno.h>
- #include <mono/utils/w32api.h>
- #if defined (HOST_WIN32)
- MONO_PRAGMA_WARNING_PUSH()
- MONO_PRAGMA_WARNING_DISABLE (4115) // warning C4115: 'IRpcStubBuffer': named type definition in parentheses
- #include <oleauto.h>
- MONO_PRAGMA_WARNING_POP()
- #include <mono/utils/w32subset.h>
- #endif
- #include "icall-decl.h"
- #include "icall-signatures.h"
- static void
- mono_System_ComObject_ReleaseInterfaces (MonoComObjectHandle obj);
- #if !defined (DISABLE_COM) || defined (HOST_WIN32)
- static int
- mono_IUnknown_QueryInterface (MonoIUnknown *pUnk, gconstpointer riid, gpointer* ppv)
- {
- g_assert (pUnk);
- return pUnk->vtable->QueryInterface (pUnk, riid, ppv);
- }
- static int
- mono_IUnknown_AddRef (MonoIUnknown *pUnk)
- {
- // The return value is a reference count, generally transient, generally not to be used, except for debugging,
- // or to assert that it is > 0.
- g_assert (pUnk);
- return pUnk->vtable->AddRef (pUnk);
- }
- static int
- mono_IUnknown_Release (MonoIUnknown *pUnk)
- {
- // Release is like free -- null is silently ignored.
- // Also, the return value is a reference count, generally transient, generally not to be used, except for debugging.
- return pUnk ? pUnk->vtable->Release (pUnk) : 0;
- }
- #endif
- /*
- Code shared between the DISABLE_COM and !DISABLE_COM
- */
- // func is an identifier, that names a function, and is also in jit-icall-reg.h,
- // and therefore a field in mono_jit_icall_info and can be token pasted into an enum value.
- //
- // The name of func must be linkable for AOT, for example g_free does not work (monoeg_g_free instead),
- // nor does the C++ overload fmod (mono_fmod instead). These functions therefore
- // must be extern "C".
- #define register_icall(func, sig, save) \
- (mono_register_jit_icall_info (&mono_get_jit_icall_info ()->func, func, #func, (sig), (save), #func))
- mono_bstr
- mono_string_to_bstr_impl (MonoStringHandle s, MonoError *error)
- {
- if (MONO_HANDLE_IS_NULL (s))
- return NULL;
- MonoGCHandle gchandle = NULL;
- mono_bstr const res = mono_ptr_to_bstr (mono_string_handle_pin_chars (s, &gchandle), mono_string_handle_length (s));
- mono_gchandle_free_internal (gchandle);
- return res;
- }
- static void*
- mono_cominterop_get_com_interface_internal (gboolean icall, MonoObjectHandle object, MonoClass *ic, MonoError *error);
- #ifndef DISABLE_COM
- #define OPDEF(a,b,c,d,e,f,g,h,i,j) \
- a = i,
- typedef enum {
- MONO_MARSHAL_NONE, /* No marshalling needed */
- MONO_MARSHAL_COPY, /* Can be copied by value to the new domain */
- MONO_MARSHAL_COPY_OUT, /* out parameter that needs to be copied back to the original instance */
- MONO_MARSHAL_SERIALIZE /* Value needs to be serialized into the new domain */
- } MonoXDomainMarshalType;
- typedef enum {
- MONO_COM_DEFAULT,
- MONO_COM_MS
- } MonoCOMProvider;
- static MonoCOMProvider com_provider = MONO_COM_DEFAULT;
- enum {
- #include "mono/cil/opcode.def"
- LAST = 0xff
- };
- #undef OPDEF
- /* This mutex protects the various cominterop related caches in MonoImage */
- #define mono_cominterop_lock() mono_os_mutex_lock (&cominterop_mutex)
- #define mono_cominterop_unlock() mono_os_mutex_unlock (&cominterop_mutex)
- static mono_mutex_t cominterop_mutex;
- GENERATE_GET_CLASS_WITH_CACHE (interop_proxy, "Mono.Interop", "ComInteropProxy")
- GENERATE_GET_CLASS_WITH_CACHE (idispatch, "Mono.Interop", "IDispatch")
- GENERATE_GET_CLASS_WITH_CACHE (iunknown, "Mono.Interop", "IUnknown")
- GENERATE_GET_CLASS_WITH_CACHE (com_object, "System", "__ComObject")
- GENERATE_GET_CLASS_WITH_CACHE (variant, "System", "Variant")
- static GENERATE_GET_CLASS_WITH_CACHE (interface_type_attribute, "System.Runtime.InteropServices", "InterfaceTypeAttribute")
- static GENERATE_GET_CLASS_WITH_CACHE (com_visible_attribute, "System.Runtime.InteropServices", "ComVisibleAttribute")
- static GENERATE_GET_CLASS_WITH_CACHE (com_default_interface_attribute, "System.Runtime.InteropServices", "ComDefaultInterfaceAttribute")
- static GENERATE_GET_CLASS_WITH_CACHE (class_interface_attribute, "System.Runtime.InteropServices", "ClassInterfaceAttribute")
- /* Upon creation of a CCW, only allocate a weak handle and set the
- * reference count to 0. If the unmanaged client code decides to addref and
- * hold onto the CCW, I then allocate a strong handle. Once the reference count
- * goes back to 0, convert back to a weak handle.
- */
- typedef struct {
- guint32 ref_count;
- MonoGCHandle gc_handle;
- GHashTable* vtable_hash;
- #ifdef HOST_WIN32
- MonoIUnknown *free_marshaler; // actually IMarshal
- #endif
- } MonoCCW;
- /* This type is the actual pointer passed to unmanaged code
- * to represent a COM interface.
- */
- typedef struct {
- gpointer vtable;
- MonoCCW* ccw;
- } MonoCCWInterface;
- /* IUnknown */
- static int STDCALL cominterop_ccw_addref (MonoCCWInterface* ccwe);
- static int STDCALL cominterop_ccw_release (MonoCCWInterface* ccwe);
- static int STDCALL cominterop_ccw_queryinterface (MonoCCWInterface* ccwe, const guint8* riid, gpointer* ppv);
- /* IDispatch */
- static int STDCALL cominterop_ccw_get_type_info_count (MonoCCWInterface* ccwe, guint32 *pctinfo);
- static int STDCALL cominterop_ccw_get_type_info (MonoCCWInterface* ccwe, guint32 iTInfo, guint32 lcid, gpointer *ppTInfo);
- static int STDCALL cominterop_ccw_get_ids_of_names (MonoCCWInterface* ccwe, gpointer riid,
- gunichar2** rgszNames, guint32 cNames,
- guint32 lcid, gint32 *rgDispId);
- static int STDCALL cominterop_ccw_invoke (MonoCCWInterface* ccwe, guint32 dispIdMember,
- gpointer riid, guint32 lcid,
- guint16 wFlags, gpointer pDispParams,
- gpointer pVarResult, gpointer pExcepInfo,
- guint32 *puArgErr);
- static MonoMethod *
- cominterop_get_managed_wrapper_adjusted (MonoMethod *method);
- static gpointer
- cominterop_get_ccw (MonoObject* object, MonoClass* itf);
- static gpointer
- cominterop_get_ccw_checked (MonoObjectHandle object, MonoClass *itf, MonoError *error);
- static MonoObject*
- cominterop_get_ccw_object (MonoCCWInterface* ccw_entry, gboolean verify);
- static MonoObjectHandle
- cominterop_get_ccw_handle (MonoCCWInterface* ccw_entry, gboolean verify);
- static MonoObject*
- cominterop_set_ccw_object_domain (MonoObject *object, MonoDomain **prev_domain);
- static void
- cominterop_restore_domain (MonoDomain *domain);
- /* SAFEARRAY marshalling */
- static gboolean
- mono_marshal_safearray_begin (gpointer safearray, MonoArray **result, gpointer *indices, gpointer empty, gpointer parameter, gboolean allocateNewArray);
- static gpointer
- mono_marshal_safearray_get_value (gpointer safearray, gpointer indices);
- static gboolean
- mono_marshal_safearray_next (gpointer safearray, gpointer indices);
- static void
- mono_marshal_safearray_end (gpointer safearray, gpointer indices);
- static gboolean
- mono_marshal_safearray_create (MonoArray *input, gpointer *newsafearray, gpointer *indices, gpointer empty);
- static void
- mono_marshal_safearray_set_value (gpointer safearray, gpointer indices, gpointer value);
- static void
- mono_marshal_safearray_free_indices (gpointer indices);
- MonoClass*
- mono_class_try_get_com_object_class (void)
- {
- static MonoClass *tmp_class;
- static gboolean inited;
- MonoClass *klass;
- if (!inited) {
- klass = mono_class_load_from_name (mono_defaults.corlib, "System", "__ComObject");
- mono_memory_barrier ();
- tmp_class = klass;
- mono_memory_barrier ();
- inited = TRUE;
- }
- return tmp_class;
- }
- /**
- * cominterop_method_signature:
- * @method: a method
- *
- * Returns: the corresponding unmanaged method signature for a managed COM
- * method.
- */
- static MonoMethodSignature*
- cominterop_method_signature (MonoMethod* method)
- {
- MonoMethodSignature *res;
- MonoImage *image = m_class_get_image (method->klass);
- MonoMethodSignature *sig = mono_method_signature_internal (method);
- gboolean const preserve_sig = (method->iflags & METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG) != 0;
- int sigsize;
- int i;
- int param_count = sig->param_count + 1; // convert this arg into IntPtr arg
- if (!preserve_sig &&!MONO_TYPE_IS_VOID (sig->ret))
- param_count++;
- res = mono_metadata_signature_alloc (image, param_count);
- sigsize = MONO_SIZEOF_METHOD_SIGNATURE + sig->param_count * sizeof (MonoType *);
- memcpy (res, sig, sigsize);
- // now move args forward one
- for (i = sig->param_count-1; i >= 0; i--)
- res->params[i+1] = sig->params[i];
- // first arg is interface pointer
- res->params[0] = mono_get_int_type ();
- if (preserve_sig) {
- res->ret = sig->ret;
- }
- else {
- // last arg is return type
- if (!MONO_TYPE_IS_VOID (sig->ret)) {
- res->params[param_count-1] = mono_metadata_type_dup (image, sig->ret);
- res->params[param_count-1]->byref = 1;
- res->params[param_count-1]->attrs = PARAM_ATTRIBUTE_OUT;
- }
- // return type is always int32 (HRESULT)
- res->ret = mono_get_int32_type ();
- }
- // no pinvoke
- res->pinvoke = FALSE;
- // no hasthis
- res->hasthis = 0;
- // set param_count
- res->param_count = param_count;
- // STDCALL on windows, CDECL everywhere else to work with XPCOM and MainWin COM
- #ifdef HOST_WIN32
- res->call_convention = MONO_CALL_STDCALL;
- #else
- res->call_convention = MONO_CALL_C;
- #endif
- return res;
- }
- /**
- * cominterop_get_function_pointer:
- * @itf: a pointer to the COM interface
- * @slot: the vtable slot of the method pointer to return
- *
- * Returns: the unmanaged vtable function pointer from the interface
- */
- static gpointer
- cominterop_get_function_pointer (gpointer itf, int slot)
- {
- gpointer func;
- func = *((*(gpointer**)itf)+slot);
- return func;
- }
- /**
- * cominterop_object_is_com_object:
- * @obj: a pointer to the object
- *
- * Returns: a value indicating if the object is a
- * Runtime Callable Wrapper (RCW) for a COM object
- */
- static gboolean
- cominterop_object_is_rcw_handle (MonoObjectHandle obj, MonoRealProxyHandle *real_proxy)
- {
- MonoClass *klass;
- return !MONO_HANDLE_IS_NULL (obj)
- && (klass = mono_handle_class (obj))
- && mono_class_is_transparent_proxy (klass)
- && !MONO_HANDLE_IS_NULL (*real_proxy = MONO_HANDLE_NEW_GET (MonoRealProxy, MONO_HANDLE_CAST (MonoTransparentProxy, obj), rp))
- && (klass = mono_handle_class (*real_proxy))
- && klass == mono_class_get_interop_proxy_class ();
- }
- static gboolean
- cominterop_object_is_rcw (MonoObject *obj_raw)
- {
- if (!obj_raw)
- return FALSE;
- HANDLE_FUNCTION_ENTER ();
- MONO_HANDLE_DCL (MonoObject, obj);
- MonoRealProxyHandle real_proxy;
- gboolean const result = cominterop_object_is_rcw_handle (obj, &real_proxy);
- HANDLE_FUNCTION_RETURN_VAL (result);
- }
- static int
- cominterop_get_com_slot_begin (MonoClass* klass)
- {
- ERROR_DECL (error);
- MonoCustomAttrInfo *cinfo = NULL;
- MonoInterfaceTypeAttribute* itf_attr = NULL;
- cinfo = mono_custom_attrs_from_class_checked (klass, error);
- mono_error_assert_ok (error);
- if (cinfo) {
- itf_attr = (MonoInterfaceTypeAttribute*)mono_custom_attrs_get_attr_checked (cinfo, mono_class_get_interface_type_attribute_class (), error);
- mono_error_assert_ok (error); /*FIXME proper error handling*/
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- }
- if (itf_attr && itf_attr->intType == 1)
- return 3; /* 3 methods in IUnknown*/
- else
- return 7; /* 7 methods in IDispatch*/
- }
- /**
- * cominterop_get_method_interface:
- * @method: method being called
- *
- * Returns: the MonoClass* representing the interface on which
- * the method is defined.
- */
- static MonoClass*
- cominterop_get_method_interface (MonoMethod* method)
- {
- ERROR_DECL (error);
- MonoClass *ic = method->klass;
- /* if method is on a class, we need to look up interface method exists on */
- if (!MONO_CLASS_IS_INTERFACE_INTERNAL (method->klass)) {
- GPtrArray *ifaces = mono_class_get_implemented_interfaces (method->klass, error);
- mono_error_assert_ok (error);
- if (ifaces) {
- int i;
- mono_class_setup_vtable (method->klass);
- for (i = 0; i < ifaces->len; ++i) {
- int j, offset;
- gboolean found = FALSE;
- ic = (MonoClass *)g_ptr_array_index (ifaces, i);
- offset = mono_class_interface_offset (method->klass, ic);
- int mcount = mono_class_get_method_count (ic);
- MonoMethod **method_klass_vtable = m_class_get_vtable (method->klass);
- for (j = 0; j < mcount; ++j) {
- if (method_klass_vtable [j + offset] == method) {
- found = TRUE;
- break;
- }
- }
- if (found)
- break;
- ic = NULL;
- }
- g_ptr_array_free (ifaces, TRUE);
- }
- }
- return ic;
- }
- static void
- mono_cominterop_get_interface_missing_error (MonoError* error, MonoMethod* method)
- {
- mono_error_set_invalid_operation (error, "Method '%s' in ComImport class '%s' must implement an interface method.", method->name, m_class_get_name (method->klass));
- }
- /**
- * cominterop_get_com_slot_for_method:
- * @method: a method
- * @error: set on error
- *
- * Returns: the method's slot in the COM interface vtable
- */
- static int
- cominterop_get_com_slot_for_method (MonoMethod* method, MonoError* error)
- {
- guint32 slot = method->slot;
- MonoClass *ic = method->klass;
- error_init (error);
- /* if method is on a class, we need to look up interface method exists on */
- if (!MONO_CLASS_IS_INTERFACE_INTERNAL (ic)) {
- int offset = 0;
- int i = 0;
- ic = cominterop_get_method_interface (method);
- if (!ic || !MONO_CLASS_IS_INTERFACE_INTERNAL (ic)) {
- mono_cominterop_get_interface_missing_error (error, method);
- return -1;
- }
- offset = mono_class_interface_offset (method->klass, ic);
- g_assert(offset >= 0);
- int mcount = mono_class_get_method_count (ic);
- MonoMethod **ic_methods = m_class_get_methods (ic);
- MonoMethod **method_klass_vtable = m_class_get_vtable (method->klass);
- for(i = 0; i < mcount; ++i) {
- if (method_klass_vtable [i + offset] == method)
- {
- slot = ic_methods[i]->slot;
- break;
- }
- }
- }
- g_assert (ic);
- g_assert (MONO_CLASS_IS_INTERFACE_INTERNAL (ic));
- return slot + cominterop_get_com_slot_begin (ic);
- }
- static gboolean
- cominterop_class_guid (MonoClass* klass, guint8* guid)
- {
- ERROR_DECL (error);
- mono_metadata_get_class_guid (klass, guid, error);
- mono_error_assert_ok (error); /*FIXME proper error handling*/
- return TRUE;
- }
- static gboolean
- cominterop_com_visible (MonoClass* klass)
- {
- ERROR_DECL (error);
- MonoCustomAttrInfo *cinfo;
- GPtrArray *ifaces;
- MonoBoolean visible = 1;
- cinfo = mono_custom_attrs_from_class_checked (klass, error);
- mono_error_assert_ok (error);
- if (cinfo) {
- MonoReflectionComVisibleAttribute *attr = (MonoReflectionComVisibleAttribute*)mono_custom_attrs_get_attr_checked (cinfo, mono_class_get_com_visible_attribute_class (), error);
- mono_error_assert_ok (error); /*FIXME proper error handling*/
- if (attr)
- visible = attr->visible;
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- if (visible)
- return TRUE;
- }
- ifaces = mono_class_get_implemented_interfaces (klass, error);
- mono_error_assert_ok (error);
- if (ifaces) {
- int i;
- for (i = 0; i < ifaces->len; ++i) {
- MonoClass *ic = NULL;
- ic = (MonoClass *)g_ptr_array_index (ifaces, i);
- if (MONO_CLASS_IS_IMPORT (ic))
- visible = TRUE;
- }
- g_ptr_array_free (ifaces, TRUE);
- }
- return visible;
- }
- gboolean
- mono_cominterop_method_com_visible (MonoMethod *method)
- {
- ERROR_DECL (error);
- MonoCustomAttrInfo *cinfo;
- MonoBoolean visible = 1;
- cinfo = mono_custom_attrs_from_method_checked (method, error);
- mono_error_assert_ok (error);
- if (cinfo) {
- MonoReflectionComVisibleAttribute *attr = (MonoReflectionComVisibleAttribute*)mono_custom_attrs_get_attr_checked (cinfo, mono_class_get_com_visible_attribute_class (), error);
- mono_error_assert_ok (error); /*FIXME proper error handling*/
- if (attr)
- visible = attr->visible;
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- }
- return visible;
- }
- static void
- cominterop_set_hr_error (MonoError *oerror, int hr)
- {
- ERROR_DECL (error);
- MonoException* ex;
- void* params[1] = {&hr};
- MONO_STATIC_POINTER_INIT (MonoMethod, throw_exception_for_hr)
- throw_exception_for_hr = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetExceptionForHR", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, throw_exception_for_hr)
- ex = (MonoException*)mono_runtime_invoke_checked (throw_exception_for_hr, NULL, params, error);
- g_assert (ex);
- mono_error_assert_ok (error);
- mono_error_set_exception_instance (oerror, ex);
- }
- /**
- * cominterop_get_interface_checked:
- * @obj: managed wrapper object containing COM object
- * @ic: interface type to retrieve for COM object
- * @error: set on error
- *
- * Returns: the COM interface requested. On failure returns NULL and sets @error
- */
- static gpointer
- cominterop_get_interface_checked (MonoComObjectHandle obj, MonoClass* ic, MonoError *error)
- {
- gpointer itf = NULL;
- g_assert (ic);
- g_assert (MONO_CLASS_IS_INTERFACE_INTERNAL (ic));
- error_init (error);
- mono_cominterop_lock ();
- if (MONO_HANDLE_GETVAL (obj, itf_hash))
- itf = g_hash_table_lookup (MONO_HANDLE_GETVAL (obj, itf_hash), GUINT_TO_POINTER ((guint)m_class_get_interface_id (ic)));
- mono_cominterop_unlock ();
- if (itf)
- return itf;
- guint8 iid [16];
- gboolean const found = cominterop_class_guid (ic, iid);
- g_assert (found);
- g_assert (MONO_HANDLE_GETVAL (obj, iunknown));
- int const hr = mono_IUnknown_QueryInterface (MONO_HANDLE_GETVAL (obj, iunknown), iid, &itf);
- if (hr < 0) {
- g_assert (!itf);
- cominterop_set_hr_error (error, hr);
- g_assert (!is_ok (error));
- return NULL;
- }
- g_assert (itf);
- mono_cominterop_lock ();
- if (!MONO_HANDLE_GETVAL (obj, itf_hash))
- MONO_HANDLE_SETVAL (obj, itf_hash, GHashTable*, g_hash_table_new (mono_aligned_addr_hash, NULL));
- g_hash_table_insert (MONO_HANDLE_GETVAL (obj, itf_hash), GUINT_TO_POINTER ((guint)m_class_get_interface_id (ic)), itf);
- mono_cominterop_unlock ();
- return itf;
- }
- /**
- * cominterop_get_interface:
- * @obj: managed wrapper object containing COM object
- * @ic: interface type to retrieve for COM object
- *
- * Returns: the COM interface requested
- */
- static gpointer
- cominterop_get_interface (MonoComObject *obj_raw, MonoClass *ic)
- {
- HANDLE_FUNCTION_ENTER ();
- ERROR_DECL (error);
- MONO_HANDLE_DCL (MonoComObject, obj);
- gpointer const itf = cominterop_get_interface_checked (obj, ic, error);
- g_assert (!!itf == is_ok (error)); // two equal success indicators
- mono_error_set_pending_exception (error);
- HANDLE_FUNCTION_RETURN_VAL (itf);
- }
- // This is an icall, it will return NULL and set pending exception (in
- // mono_type_from_handle wrapper) on failure.
- static MonoReflectionType *
- cominterop_type_from_handle (MonoType *handle)
- {
- return mono_type_from_handle (handle);
- }
- #endif // DISABLE_COM
- void
- mono_cominterop_init (void)
- {
- #ifndef DISABLE_COM
- mono_os_mutex_init_recursive (&cominterop_mutex);
- char* const com_provider_env = g_getenv ("MONO_COM");
- if (com_provider_env && !strcmp(com_provider_env, "MS"))
- com_provider = MONO_COM_MS;
- g_free (com_provider_env);
- register_icall (cominterop_get_method_interface, mono_icall_sig_ptr_ptr, FALSE);
- register_icall (cominterop_get_function_pointer, mono_icall_sig_ptr_ptr_int32, FALSE);
- register_icall (cominterop_object_is_rcw, mono_icall_sig_int32_object, FALSE);
- register_icall (cominterop_get_ccw, mono_icall_sig_ptr_object_ptr, FALSE);
- register_icall (cominterop_get_ccw_object, mono_icall_sig_object_ptr_int32, FALSE);
- register_icall (cominterop_get_interface, mono_icall_sig_ptr_object_ptr, FALSE);
- register_icall (cominterop_type_from_handle, mono_icall_sig_object_ptr, FALSE);
- register_icall (cominterop_set_ccw_object_domain, mono_icall_sig_object_object_ptr, FALSE);
- register_icall (cominterop_restore_domain, mono_icall_sig_void_ptr, FALSE);
- /* SAFEARRAY marshalling */
- register_icall (mono_marshal_safearray_begin, mono_icall_sig_int32_ptr_ptr_ptr_ptr_ptr_int32, FALSE);
- register_icall (mono_marshal_safearray_get_value, mono_icall_sig_ptr_ptr_ptr, FALSE);
- register_icall (mono_marshal_safearray_next, mono_icall_sig_int32_ptr_ptr, FALSE);
- register_icall (mono_marshal_safearray_end, mono_icall_sig_void_ptr_ptr, FALSE);
- register_icall (mono_marshal_safearray_create, mono_icall_sig_int32_object_ptr_ptr_ptr, FALSE);
- register_icall (mono_marshal_safearray_set_value, mono_icall_sig_void_ptr_ptr_ptr, FALSE);
- register_icall (mono_marshal_safearray_free_indices, mono_icall_sig_void_ptr, FALSE);
- #endif // DISABLE_COM
- /*FIXME
- This icalls are used by the marshal code when doing PtrToStructure and StructureToPtr and pinvoke.
- If we leave them out and the FullAOT compiler finds the need to emit one of the above 3 wrappers it will
- g_assert.
- The proper fix would be to emit warning, remove them from marshal.c when DISABLE_COM is used and
- emit an exception in the generated IL.
- */
- register_icall (mono_string_to_bstr, mono_icall_sig_ptr_obj, FALSE);
- register_icall (mono_string_from_bstr_icall, mono_icall_sig_obj_ptr, FALSE);
- register_icall (mono_free_bstr, mono_icall_sig_void_ptr, FALSE);
- }
- #ifndef DISABLE_COM
- void
- mono_cominterop_cleanup (void)
- {
- mono_os_mutex_destroy (&cominterop_mutex);
- }
- void
- mono_mb_emit_cominterop_get_function_pointer (MonoMethodBuilder *mb, MonoMethod *method)
- {
- #ifndef DISABLE_JIT
- int slot;
- ERROR_DECL (error);
- // get function pointer from 1st arg, the COM interface pointer
- mono_mb_emit_ldarg (mb, 0);
- slot = cominterop_get_com_slot_for_method (method, error);
- if (is_ok (error)) {
- mono_mb_emit_icon (mb, slot);
- mono_mb_emit_icall (mb, cominterop_get_function_pointer);
- /* Leaves the function pointer on top of the stack */
- }
- else {
- mono_mb_emit_exception_for_error (mb, error);
- }
- mono_error_cleanup (error);
- #endif
- }
- void
- mono_mb_emit_cominterop_call_function_pointer (MonoMethodBuilder *mb, MonoMethodSignature *sig)
- {
- #ifndef DISABLE_JIT
- mono_mb_emit_byte (mb, MONO_CUSTOM_PREFIX);
- mono_mb_emit_byte (mb, CEE_MONO_SAVE_LMF);
- mono_mb_emit_calli (mb, sig);
- mono_mb_emit_byte (mb, MONO_CUSTOM_PREFIX);
- mono_mb_emit_byte (mb, CEE_MONO_RESTORE_LMF);
- #endif /* DISABLE_JIT */
- }
- void
- mono_mb_emit_cominterop_call (MonoMethodBuilder *mb, MonoMethodSignature *sig, MonoMethod* method)
- {
- #ifndef DISABLE_JIT
- mono_mb_emit_cominterop_get_function_pointer (mb, method);
- mono_mb_emit_cominterop_call_function_pointer (mb, sig);
- #endif /* DISABLE_JIT */
- }
- void
- mono_cominterop_emit_ptr_to_object_conv (MonoMethodBuilder *mb, MonoType *type, MonoMarshalConv conv, MonoMarshalSpec *mspec)
- {
- #ifndef DISABLE_JIT
- switch (conv) {
- case MONO_MARSHAL_CONV_OBJECT_INTERFACE:
- case MONO_MARSHAL_CONV_OBJECT_IUNKNOWN:
- case MONO_MARSHAL_CONV_OBJECT_IDISPATCH: {
- guint32 pos_null = 0, pos_ccw = 0, pos_end = 0;
- MonoClass *klass = NULL;
- klass = mono_class_from_mono_type_internal (type);
- mono_mb_emit_ldloc (mb, 1);
- mono_mb_emit_byte (mb, CEE_LDNULL);
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_I);
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- /* load dst to store later */
- mono_mb_emit_ldloc (mb, 1);
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_I);
- mono_mb_emit_icon (mb, TRUE);
- mono_mb_emit_icall (mb, cominterop_get_ccw_object);
- pos_ccw = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- MONO_STATIC_POINTER_INIT (MonoMethod, com_interop_proxy_get_proxy)
- ERROR_DECL (error);
- com_interop_proxy_get_proxy = mono_class_get_method_from_name_checked (mono_class_get_interop_proxy_class (), "GetProxy", 2, METHOD_ATTRIBUTE_PRIVATE, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, com_interop_proxy_get_proxy)
- #ifndef DISABLE_REMOTING
- MONO_STATIC_POINTER_INIT (MonoMethod, get_transparent_proxy)
- ERROR_DECL (error);
- get_transparent_proxy = mono_class_get_method_from_name_checked (mono_defaults.real_proxy_class, "GetTransparentProxy", 0, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_transparent_proxy)
- #else
- static MonoMethod* const get_transparent_proxy = NULL; // FIXME?
- #endif
- mono_mb_add_local (mb, m_class_get_byval_arg (mono_class_get_interop_proxy_class ()));
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_I);
- mono_mb_emit_ptr (mb, m_class_get_byval_arg (mono_class_get_com_object_class ()));
- mono_mb_emit_icall (mb, cominterop_type_from_handle);
- mono_mb_emit_managed_call (mb, com_interop_proxy_get_proxy, NULL);
- mono_mb_emit_managed_call (mb, get_transparent_proxy, NULL);
- if (conv == MONO_MARSHAL_CONV_OBJECT_INTERFACE) {
- g_assert (klass);
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- }
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- pos_end = mono_mb_emit_short_branch (mb, CEE_BR_S);
- /* is already managed object */
- mono_mb_patch_short_branch (mb, pos_ccw);
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_I);
- mono_mb_emit_icon (mb, TRUE);
- mono_mb_emit_icall (mb, cominterop_get_ccw_object);
- if (conv == MONO_MARSHAL_CONV_OBJECT_INTERFACE) {
- g_assert (klass);
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- }
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- mono_mb_patch_short_branch (mb, pos_end);
- /* case if null */
- mono_mb_patch_short_branch (mb, pos_null);
- break;
- }
- default:
- g_assert_not_reached ();
- }
- #endif /* DISABLE_JIT */
- }
- void
- mono_cominterop_emit_object_to_ptr_conv (MonoMethodBuilder *mb, MonoType *type, MonoMarshalConv conv, MonoMarshalSpec *mspec)
- {
- #ifndef DISABLE_JIT
- switch (conv) {
- case MONO_MARSHAL_CONV_OBJECT_INTERFACE:
- case MONO_MARSHAL_CONV_OBJECT_IDISPATCH:
- case MONO_MARSHAL_CONV_OBJECT_IUNKNOWN: {
- guint32 pos_null = 0, pos_rcw = 0, pos_end = 0;
- mono_mb_emit_ldloc (mb, 1);
- mono_mb_emit_icon (mb, 0);
- mono_mb_emit_byte (mb, CEE_CONV_U);
- mono_mb_emit_byte (mb, CEE_STIND_I);
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- // if null just break, dst was already inited to 0
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- mono_mb_emit_icall (mb, cominterop_object_is_rcw);
- pos_rcw = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- // load dst to store later
- mono_mb_emit_ldloc (mb, 1);
- // load src
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- mono_mb_emit_ldflda (mb, MONO_STRUCT_OFFSET (MonoTransparentProxy, rp));
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- /* load the RCW from the ComInteropProxy*/
- mono_mb_emit_ldflda (mb, MONO_STRUCT_OFFSET (MonoComInteropProxy, com_object));
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- if (conv == MONO_MARSHAL_CONV_OBJECT_INTERFACE) {
- mono_mb_emit_ptr (mb, mono_type_get_class_internal (type));
- mono_mb_emit_icall (mb, cominterop_get_interface);
- }
- else if (conv == MONO_MARSHAL_CONV_OBJECT_IUNKNOWN) {
- MONO_STATIC_POINTER_INIT (MonoProperty, iunknown)
- iunknown = mono_class_get_property_from_name_internal (mono_class_get_com_object_class (), "IUnknown");
- MONO_STATIC_POINTER_INIT_END (MonoProperty, iunknown)
- mono_mb_emit_managed_call (mb, iunknown->get, NULL);
- }
- else if (conv == MONO_MARSHAL_CONV_OBJECT_IDISPATCH) {
- MONO_STATIC_POINTER_INIT (MonoProperty, idispatch)
- idispatch = mono_class_get_property_from_name_internal (mono_class_get_com_object_class (), "IDispatch");
- MONO_STATIC_POINTER_INIT_END (MonoProperty, idispatch)
- mono_mb_emit_managed_call (mb, idispatch->get, NULL);
- }
- else {
- g_assert_not_reached ();
- }
- mono_mb_emit_byte (mb, CEE_STIND_I);
- pos_end = mono_mb_emit_short_branch (mb, CEE_BR_S);
-
- // if not rcw
- mono_mb_patch_short_branch (mb, pos_rcw);
- /* load dst to store later */
- mono_mb_emit_ldloc (mb, 1);
- /* load src */
- mono_mb_emit_ldloc (mb, 0);
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
-
- if (conv == MONO_MARSHAL_CONV_OBJECT_INTERFACE)
- mono_mb_emit_ptr (mb, mono_type_get_class_internal (type));
- else if (conv == MONO_MARSHAL_CONV_OBJECT_IUNKNOWN)
- mono_mb_emit_ptr (mb, mono_class_get_iunknown_class ());
- else if (conv == MONO_MARSHAL_CONV_OBJECT_IDISPATCH)
- mono_mb_emit_ptr (mb, mono_class_get_idispatch_class ());
- else
- g_assert_not_reached ();
- mono_mb_emit_icall (mb, cominterop_get_ccw);
- mono_mb_emit_byte (mb, CEE_STIND_I);
- mono_mb_patch_short_branch (mb, pos_end);
- mono_mb_patch_short_branch (mb, pos_null);
- break;
- }
- default:
- g_assert_not_reached ();
- }
- #endif /* DISABLE_JIT */
- }
- /**
- * cominterop_get_native_wrapper_adjusted:
- * @method: managed COM Interop method
- *
- * Returns: the generated method to call with signature matching
- * the unmanaged COM Method signature
- */
- static MonoMethod *
- cominterop_get_native_wrapper_adjusted (MonoMethod *method)
- {
- MonoMethod *res;
- MonoMethodBuilder *mb_native;
- MonoMarshalSpec **mspecs;
- MonoMethodSignature *sig, *sig_native;
- MonoMethodPInvoke *piinfo = (MonoMethodPInvoke *) method;
- int i;
- sig = mono_method_signature_internal (method);
- // create unmanaged wrapper
- mb_native = mono_mb_new (method->klass, method->name, MONO_WRAPPER_MANAGED_TO_NATIVE);
- sig_native = cominterop_method_signature (method);
- mspecs = g_new0 (MonoMarshalSpec*, sig_native->param_count + 1);
- mono_method_get_marshal_info (method, mspecs);
- // move managed args up one
- for (i = sig->param_count; i >= 1; i--)
- mspecs[i+1] = mspecs[i];
- // first arg is IntPtr for interface
- mspecs[1] = NULL;
- if (!(method->iflags & METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG)) {
- // move return spec to last param
- if (!MONO_TYPE_IS_VOID (sig->ret))
- mspecs[sig_native->param_count] = mspecs[0];
- mspecs[0] = NULL;
- }
- for (i = 1; i < sig_native->param_count; i++) {
- int mspec_index = i + 1;
- if (mspecs[mspec_index] == NULL) {
- // default object to VARIANT
- if (sig_native->params[i]->type == MONO_TYPE_OBJECT) {
- mspecs[mspec_index] = g_new0 (MonoMarshalSpec, 1);
- mspecs[mspec_index]->native = MONO_NATIVE_STRUCT;
- }
- else if (sig_native->params[i]->type == MONO_TYPE_STRING) {
- mspecs[mspec_index] = g_new0 (MonoMarshalSpec, 1);
- mspecs[mspec_index]->native = MONO_NATIVE_BSTR;
- }
- else if (sig_native->params[i]->type == MONO_TYPE_CLASS) {
- mspecs[mspec_index] = g_new0 (MonoMarshalSpec, 1);
- mspecs[mspec_index]->native = MONO_NATIVE_INTERFACE;
- }
- else if (sig_native->params[i]->type == MONO_TYPE_BOOLEAN) {
- mspecs[mspec_index] = g_new0 (MonoMarshalSpec, 1);
- mspecs[mspec_index]->native = MONO_NATIVE_VARIANTBOOL;
- }
- }
- }
- if (method->iflags & METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG) {
- // move return spec to last param
- if (!MONO_TYPE_IS_VOID (sig->ret) && mspecs[0] == NULL) {
- // default object to VARIANT
- if (sig->ret->type == MONO_TYPE_OBJECT) {
- mspecs[0] = g_new0 (MonoMarshalSpec, 1);
- mspecs[0]->native = MONO_NATIVE_STRUCT;
- }
- else if (sig->ret->type == MONO_TYPE_STRING) {
- mspecs[0] = g_new0 (MonoMarshalSpec, 1);
- mspecs[0]->native = MONO_NATIVE_BSTR;
- }
- else if (sig->ret->type == MONO_TYPE_CLASS) {
- mspecs[0] = g_new0 (MonoMarshalSpec, 1);
- mspecs[0]->native = MONO_NATIVE_INTERFACE;
- }
- else if (sig->ret->type == MONO_TYPE_BOOLEAN) {
- mspecs[0] = g_new0 (MonoMarshalSpec, 1);
- mspecs[0]->native = MONO_NATIVE_VARIANTBOOL;
- }
- }
- }
- mono_marshal_emit_native_wrapper (m_class_get_image (method->klass), mb_native, sig_native, piinfo, mspecs, piinfo->addr, EMIT_NATIVE_WRAPPER_CHECK_EXCEPTIONS);
- res = mono_mb_create_method (mb_native, sig_native, sig_native->param_count + 16);
- mono_mb_free (mb_native);
- for (i = sig_native->param_count; i >= 0; i--)
- if (mspecs [i])
- mono_metadata_free_marshal_spec (mspecs [i]);
- g_free (mspecs);
- return res;
- }
- /**
- * mono_cominterop_get_native_wrapper:
- * \param method managed method
- * \returns the generated method to call
- */
- MonoMethod *
- mono_cominterop_get_native_wrapper (MonoMethod *method)
- {
- MonoMethod *res;
- GHashTable *cache;
- MonoMethodBuilder *mb;
- MonoMethodSignature *sig, *csig;
- g_assert (method);
- cache = mono_marshal_get_cache (&mono_method_get_wrapper_cache (method)->cominterop_wrapper_cache, mono_aligned_addr_hash, NULL);
- if ((res = mono_marshal_find_in_cache (cache, method)))
- return res;
- if (!m_class_get_vtable (method->klass))
- mono_class_setup_vtable (method->klass);
-
- if (!m_class_get_methods (method->klass))
- mono_class_setup_methods (method->klass);
- g_assert (!mono_class_has_failure (method->klass)); /*FIXME do proper error handling*/
- sig = mono_method_signature_internal (method);
- mb = mono_mb_new (method->klass, method->name, MONO_WRAPPER_COMINTEROP);
- #ifndef DISABLE_JIT
- /* if method klass is import, that means method
- * is really a com call. let interop system emit it.
- */
- if (MONO_CLASS_IS_IMPORT(method->klass)) {
- /* FIXME: we have to call actual class .ctor
- * instead of just __ComObject .ctor.
- */
- if (!strcmp(method->name, ".ctor")) {
- MONO_STATIC_POINTER_INIT (MonoMethod, ctor)
- ERROR_DECL (error);
- ctor = mono_class_get_method_from_name_checked (mono_class_get_com_object_class (), ".ctor", 0, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, ctor)
- mono_mb_emit_ldarg (mb, 0);
- mono_mb_emit_managed_call (mb, ctor, NULL);
- mono_mb_emit_byte (mb, CEE_RET);
- }
- else if (method->flags & METHOD_ATTRIBUTE_STATIC) {
- /*
- * The method's class must implement an interface.
- * However, no interfaces are allowed to have static methods.
- * Thus, calling it should invariably lead to an exception.
- */
- ERROR_DECL (error);
- mono_cominterop_get_interface_missing_error (error, method);
- mono_mb_emit_exception_for_error (mb, error);
- mono_error_cleanup (error);
- }
- else {
- MonoMethod *adjusted_method;
- int retval = 0;
- int ptr_this;
- int i;
- gboolean const preserve_sig = (method->iflags & METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG) != 0;
- // add local variables
- ptr_this = mono_mb_add_local (mb, mono_get_int_type ());
- if (!MONO_TYPE_IS_VOID (sig->ret))
- retval = mono_mb_add_local (mb, sig->ret);
- // get the type for the interface the method is defined on
- // and then get the underlying COM interface for that type
- mono_mb_emit_ldarg (mb, 0);
- mono_mb_emit_ptr (mb, method);
- mono_mb_emit_icall (mb, cominterop_get_method_interface);
- mono_mb_emit_icall (mb, cominterop_get_interface);
- mono_mb_emit_stloc (mb, ptr_this);
- // arg 1 is unmanaged this pointer
- mono_mb_emit_ldloc (mb, ptr_this);
- // load args
- for (i = 1; i <= sig->param_count; i++)
- mono_mb_emit_ldarg (mb, i);
- // push managed return value as byref last argument
- if (!MONO_TYPE_IS_VOID (sig->ret) && !preserve_sig)
- mono_mb_emit_ldloc_addr (mb, retval);
-
- adjusted_method = cominterop_get_native_wrapper_adjusted (method);
- mono_mb_emit_managed_call (mb, adjusted_method, NULL);
- if (!preserve_sig) {
- MONO_STATIC_POINTER_INIT (MonoMethod, ThrowExceptionForHR)
- ERROR_DECL (error);
- ThrowExceptionForHR = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "ThrowExceptionForHR", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, ThrowExceptionForHR)
- mono_mb_emit_managed_call (mb, ThrowExceptionForHR, NULL);
- // load return value managed is expecting
- if (!MONO_TYPE_IS_VOID (sig->ret))
- mono_mb_emit_ldloc (mb, retval);
- }
- mono_mb_emit_byte (mb, CEE_RET);
- }
-
-
- }
- /* Does this case ever get hit? */
- else {
- char *msg = g_strdup ("non imported interfaces on \
- imported classes is not yet implemented.");
- mono_mb_emit_exception (mb, "NotSupportedException", msg);
- }
- #endif /* DISABLE_JIT */
- csig = mono_metadata_signature_dup_full (m_class_get_image (method->klass), sig);
- csig->pinvoke = 0;
- res = mono_mb_create_and_cache (cache, method,
- mb, csig, csig->param_count + 16);
- mono_mb_free (mb);
- return res;
- }
- /**
- * mono_cominterop_get_invoke:
- * \param method managed method
- * \returns the generated method that calls the underlying \c __ComObject
- * rather than the proxy object.
- */
- MonoMethod *
- mono_cominterop_get_invoke (MonoMethod *method)
- {
- MonoMethodSignature *sig;
- MonoMethodBuilder *mb;
- MonoMethod *res;
- int i;
- GHashTable* cache;
-
- cache = mono_marshal_get_cache (&mono_method_get_wrapper_cache (method)->cominterop_invoke_cache, mono_aligned_addr_hash, NULL);
- g_assert (method);
- if ((res = mono_marshal_find_in_cache (cache, method)))
- return res;
- sig = mono_signature_no_pinvoke (method);
- /* we cant remote methods without this pointer */
- if (!sig->hasthis)
- return method;
- mb = mono_mb_new (method->klass, method->name, MONO_WRAPPER_COMINTEROP_INVOKE);
- #ifndef DISABLE_JIT
- /* get real proxy object, which is a ComInteropProxy in this case*/
- mono_mb_add_local (mb, mono_get_object_type ());
- mono_mb_emit_ldarg (mb, 0);
- mono_mb_emit_ldflda (mb, MONO_STRUCT_OFFSET (MonoTransparentProxy, rp));
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- /* load the RCW from the ComInteropProxy*/
- mono_mb_emit_ldflda (mb, MONO_STRUCT_OFFSET (MonoComInteropProxy, com_object));
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- /* load args and make the call on the RCW */
- for (i = 1; i <= sig->param_count; i++)
- mono_mb_emit_ldarg (mb, i);
- if ((method->iflags & METHOD_IMPL_ATTRIBUTE_INTERNAL_CALL) || mono_class_is_interface (method->klass)) {
- MonoMethod * native_wrapper = mono_cominterop_get_native_wrapper(method);
- mono_mb_emit_managed_call (mb, native_wrapper, NULL);
- }
- else {
- if (method->flags & METHOD_ATTRIBUTE_VIRTUAL)
- mono_mb_emit_op (mb, CEE_CALLVIRT, method);
- else
- mono_mb_emit_op (mb, CEE_CALL, method);
- }
- if (!strcmp(method->name, ".ctor")) {
- MONO_STATIC_POINTER_INIT (MonoMethod, cache_proxy)
- ERROR_DECL (error);
- cache_proxy = mono_class_get_method_from_name_checked (mono_class_get_interop_proxy_class (), "CacheProxy", 0, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, cache_proxy)
- mono_mb_emit_ldarg (mb, 0);
- mono_mb_emit_ldflda (mb, MONO_STRUCT_OFFSET (MonoTransparentProxy, rp));
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- mono_mb_emit_managed_call (mb, cache_proxy, NULL);
- }
- mono_marshal_emit_thread_interrupt_checkpoint (mb);
- mono_mb_emit_byte (mb, CEE_RET);
- #endif /* DISABLE_JIT */
- res = mono_mb_create_and_cache (cache, method, mb, sig, sig->param_count + 16);
- mono_mb_free (mb);
- return res;
- }
- /* Maps a managed object to its unmanaged representation
- * i.e. it's COM Callable Wrapper (CCW).
- * Key: MonoObject*
- * Value: MonoCCW*
- */
- static GHashTable* ccw_hash = NULL;
- /* Maps a CCW interface to it's containing CCW.
- * Note that a CCW support many interfaces.
- * Key: MonoCCW*
- * Value: MonoCCWInterface*
- */
- static GHashTable* ccw_interface_hash = NULL;
- /* Maps the IUnknown value of a RCW to
- * it's MonoComInteropProxy*.
- * Key: void*
- * Value: gchandle
- */
- static GHashTable* rcw_hash = NULL;
- static MonoMethod*
- mono_get_addref (void)
- {
- MONO_STATIC_POINTER_INIT (MonoMethod, AddRef)
- ERROR_DECL (error);
- AddRef = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "AddRef", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, AddRef)
- return AddRef;
- }
- int
- mono_cominterop_emit_marshal_com_interface (EmitMarshalContext *m, int argnum,
- MonoType *t,
- MonoMarshalSpec *spec,
- int conv_arg, MonoType **conv_arg_type,
- MarshalAction action)
- {
- MonoMethodBuilder *mb = m->mb;
- MonoClass *klass = t->data.klass;
- ERROR_DECL (error);
- MONO_STATIC_POINTER_INIT (MonoMethod, get_object_for_iunknown)
- get_object_for_iunknown = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetObjectForIUnknown", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_object_for_iunknown)
- MONO_STATIC_POINTER_INIT (MonoMethod, get_iunknown_for_object_internal)
- get_iunknown_for_object_internal = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetIUnknownForObjectInternal", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_iunknown_for_object_internal)
- MONO_STATIC_POINTER_INIT (MonoMethod, get_idispatch_for_object_internal)
- get_idispatch_for_object_internal = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetIDispatchForObjectInternal", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_idispatch_for_object_internal)
- MONO_STATIC_POINTER_INIT (MonoMethod, get_com_interface_for_object_internal)
- get_com_interface_for_object_internal = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetComInterfaceForObjectInternal", 2, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_com_interface_for_object_internal)
- MONO_STATIC_POINTER_INIT (MonoMethod, marshal_release)
- marshal_release = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "Release", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, marshal_release)
- #ifdef DISABLE_JIT
- switch (action) {
- case MARSHAL_ACTION_CONV_IN:
- *conv_arg_type = mono_get_int_type ();
- break;
- case MARSHAL_ACTION_MANAGED_CONV_IN:
- *conv_arg_type = mono_get_int_type ();
- break;
- default:
- break;
- }
- #else
- switch (action) {
- case MARSHAL_ACTION_CONV_IN: {
- guint32 pos_null = 0;
- MonoType *int_type = mono_get_int_type ();
- *conv_arg_type = int_type;
- conv_arg = mono_mb_add_local (mb, int_type);
- mono_mb_emit_ptr (mb, NULL);
- mono_mb_emit_stloc (mb, conv_arg);
- /* we dont need any conversions for out parameters */
- if (t->byref && t->attrs & PARAM_ATTRIBUTE_OUT)
- break;
- mono_mb_emit_ldarg (mb, argnum);
- if (t->byref)
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- /* if null just break, conv arg was already inited to 0 */
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldarg (mb, argnum);
- if (t->byref)
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- if (klass && klass != mono_defaults.object_class) {
- mono_mb_emit_ptr (mb, t);
- mono_mb_emit_icall (mb, cominterop_type_from_handle);
- mono_mb_emit_managed_call (mb, get_com_interface_for_object_internal, NULL);
- }
- else if (spec->native == MONO_NATIVE_IUNKNOWN)
- mono_mb_emit_managed_call (mb, get_iunknown_for_object_internal, NULL);
- else if (spec->native == MONO_NATIVE_IDISPATCH)
- mono_mb_emit_managed_call (mb, get_idispatch_for_object_internal, NULL);
- else if (!klass && spec->native == MONO_NATIVE_INTERFACE)
- mono_mb_emit_managed_call (mb, get_iunknown_for_object_internal, NULL);
- else
- g_assert_not_reached ();
- mono_mb_emit_stloc (mb, conv_arg);
- mono_mb_patch_short_branch (mb, pos_null);
- break;
- }
- case MARSHAL_ACTION_CONV_OUT: {
- if (t->byref && (t->attrs & PARAM_ATTRIBUTE_OUT)) {
- int ccw_obj;
- guint32 pos_null = 0, pos_ccw = 0, pos_end = 0;
- ccw_obj = mono_mb_add_local (mb, mono_get_object_type ());
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_byte (mb, CEE_LDNULL);
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- mono_mb_emit_ldloc (mb, conv_arg);
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_icon (mb, TRUE);
- mono_mb_emit_icall (mb, cominterop_get_ccw_object);
- mono_mb_emit_stloc (mb, ccw_obj);
- mono_mb_emit_ldloc (mb, ccw_obj);
- pos_ccw = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_managed_call (mb, get_object_for_iunknown, NULL);
- if (klass && klass != mono_defaults.object_class)
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- pos_end = mono_mb_emit_short_branch (mb, CEE_BR_S);
- /* is already managed object */
- mono_mb_patch_short_branch (mb, pos_ccw);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, ccw_obj);
- if (klass && klass != mono_defaults.object_class)
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- mono_mb_patch_short_branch (mb, pos_end);
- /* need to call Release to follow COM rules of ownership */
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_managed_call (mb, marshal_release, NULL);
- mono_mb_emit_byte (mb, CEE_POP);
- /* case if null */
- mono_mb_patch_short_branch (mb, pos_null);
- }
- break;
- }
- case MARSHAL_ACTION_PUSH:
- if (t->byref)
- mono_mb_emit_ldloc_addr (mb, conv_arg);
- else
- mono_mb_emit_ldloc (mb, conv_arg);
- break;
- case MARSHAL_ACTION_CONV_RESULT: {
- int ccw_obj, ret_ptr;
- guint32 pos_null = 0, pos_ccw = 0, pos_end = 0;
- ccw_obj = mono_mb_add_local (mb, mono_get_object_type ());
- ret_ptr = mono_mb_add_local (mb, mono_get_int_type ());
- /* store return value */
- mono_mb_emit_stloc (mb, ret_ptr);
- mono_mb_emit_ldloc (mb, ret_ptr);
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldloc (mb, ret_ptr);
- mono_mb_emit_icon (mb, TRUE);
- mono_mb_emit_icall (mb, cominterop_get_ccw_object);
- mono_mb_emit_stloc (mb, ccw_obj);
- mono_mb_emit_ldloc (mb, ccw_obj);
- pos_ccw = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- mono_mb_emit_ldloc (mb, ret_ptr);
- mono_mb_emit_managed_call (mb, get_object_for_iunknown, NULL);
- if (klass && klass != mono_defaults.object_class)
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- mono_mb_emit_stloc (mb, 3);
- pos_end = mono_mb_emit_short_branch (mb, CEE_BR_S);
- /* is already managed object */
- mono_mb_patch_short_branch (mb, pos_ccw);
- mono_mb_emit_ldloc (mb, ccw_obj);
- if (klass && klass != mono_defaults.object_class)
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- mono_mb_emit_stloc (mb, 3);
- mono_mb_patch_short_branch (mb, pos_end);
- /* need to call Release to follow COM rules of ownership */
- mono_mb_emit_ldloc (mb, ret_ptr);
- mono_mb_emit_managed_call (mb, marshal_release, NULL);
- mono_mb_emit_byte (mb, CEE_POP);
- /* case if null */
- mono_mb_patch_short_branch (mb, pos_null);
- break;
- }
- case MARSHAL_ACTION_MANAGED_CONV_IN: {
- int ccw_obj;
- guint32 pos_null = 0, pos_ccw = 0, pos_end = 0;
- ccw_obj = mono_mb_add_local (mb, mono_get_object_type ());
- klass = mono_class_from_mono_type_internal (t);
- conv_arg = mono_mb_add_local (mb, m_class_get_byval_arg (klass));
- *conv_arg_type = mono_get_int_type ();
- mono_mb_emit_byte (mb, CEE_LDNULL);
- mono_mb_emit_stloc (mb, conv_arg);
- if (t->attrs & PARAM_ATTRIBUTE_OUT)
- break;
- mono_mb_emit_ldarg (mb, argnum);
- if (t->byref)
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldarg (mb, argnum);
- if (t->byref)
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- mono_mb_emit_icon (mb, TRUE);
- mono_mb_emit_icall (mb, cominterop_get_ccw_object);
- mono_mb_emit_stloc (mb, ccw_obj);
- mono_mb_emit_ldloc (mb, ccw_obj);
- pos_ccw = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- mono_mb_emit_ldarg (mb, argnum);
- if (t->byref)
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- mono_mb_emit_managed_call (mb, get_object_for_iunknown, NULL);
- if (klass && klass != mono_defaults.object_class)
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- mono_mb_emit_stloc (mb, conv_arg);
- pos_end = mono_mb_emit_short_branch (mb, CEE_BR_S);
- /* is already managed object */
- mono_mb_patch_short_branch (mb, pos_ccw);
- mono_mb_emit_ldloc (mb, ccw_obj);
- if (klass && klass != mono_defaults.object_class)
- mono_mb_emit_op (mb, CEE_CASTCLASS, klass);
- mono_mb_emit_stloc (mb, conv_arg);
- mono_mb_patch_short_branch (mb, pos_end);
- /* case if null */
- mono_mb_patch_short_branch (mb, pos_null);
- break;
- }
- case MARSHAL_ACTION_MANAGED_CONV_OUT: {
- if (t->byref && t->attrs & PARAM_ATTRIBUTE_OUT) {
- guint32 pos_null = 0;
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_byte (mb, CEE_LDC_I4_0);
- mono_mb_emit_byte (mb, CEE_STIND_I);
- mono_mb_emit_ldloc (mb, conv_arg);
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- /* to store later */
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, conv_arg);
- if (klass && klass != mono_defaults.object_class) {
- mono_mb_emit_ptr (mb, t);
- mono_mb_emit_icall (mb, cominterop_type_from_handle);
- mono_mb_emit_managed_call (mb, get_com_interface_for_object_internal, NULL);
- }
- else if (spec->native == MONO_NATIVE_IUNKNOWN)
- mono_mb_emit_managed_call (mb, get_iunknown_for_object_internal, NULL);
- else if (spec->native == MONO_NATIVE_IDISPATCH)
- mono_mb_emit_managed_call (mb, get_idispatch_for_object_internal, NULL);
- else if (!klass && spec->native == MONO_NATIVE_INTERFACE)
- mono_mb_emit_managed_call (mb, get_iunknown_for_object_internal, NULL);
- else
- g_assert_not_reached ();
- mono_mb_emit_byte (mb, CEE_STIND_I);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_byte (mb, CEE_LDIND_I);
- mono_mb_emit_managed_call (mb, mono_get_addref (), NULL);
- mono_mb_emit_byte (mb, CEE_POP);
- mono_mb_patch_short_branch (mb, pos_null);
- }
- break;
- }
- case MARSHAL_ACTION_MANAGED_CONV_RESULT: {
- guint32 pos_null = 0;
- int ccw_obj;
- ccw_obj = mono_mb_add_local (mb, mono_get_object_type ());
- /* store return value */
- mono_mb_emit_stloc (mb, ccw_obj);
- mono_mb_emit_ldloc (mb, ccw_obj);
- /* if null just break, conv arg was already inited to 0 */
- pos_null = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- /* to store later */
- mono_mb_emit_ldloc (mb, ccw_obj);
- if (klass && klass != mono_defaults.object_class) {
- mono_mb_emit_ptr (mb, t);
- mono_mb_emit_icall (mb, cominterop_type_from_handle);
- mono_mb_emit_managed_call (mb, get_com_interface_for_object_internal, NULL);
- }
- else if (spec->native == MONO_NATIVE_IUNKNOWN)
- mono_mb_emit_managed_call (mb, get_iunknown_for_object_internal, NULL);
- else if (spec->native == MONO_NATIVE_IDISPATCH)
- mono_mb_emit_managed_call (mb, get_idispatch_for_object_internal, NULL);
- else if (!klass && spec->native == MONO_NATIVE_INTERFACE)
- mono_mb_emit_managed_call (mb, get_iunknown_for_object_internal, NULL);
- else
- g_assert_not_reached ();
- mono_mb_emit_stloc (mb, 3);
- mono_mb_emit_ldloc (mb, 3);
-
- mono_mb_emit_managed_call (mb, mono_get_addref (), NULL);
- mono_mb_emit_byte (mb, CEE_POP);
- mono_mb_patch_short_branch (mb, pos_null);
- break;
- }
- default:
- g_assert_not_reached ();
- }
- #endif /* DISABLE_JIT */
- return conv_arg;
- }
- #define MONO_S_OK 0x00000000L
- #define MONO_E_NOINTERFACE 0x80004002L
- #define MONO_E_NOTIMPL 0x80004001L
- #define MONO_E_INVALIDARG 0x80070057L
- #define MONO_E_DISP_E_UNKNOWNNAME 0x80020006L
- #define MONO_E_DISPID_UNKNOWN (gint32)-1
- int
- ves_icall_System_Runtime_InteropServices_Marshal_AddRefInternal (MonoIUnknown *pUnk)
- {
- return mono_IUnknown_AddRef (pUnk);
- }
- int
- ves_icall_System_Runtime_InteropServices_Marshal_QueryInterfaceInternal (MonoIUnknown *pUnk, gconstpointer riid, gpointer* ppv)
- {
- return mono_IUnknown_QueryInterface (pUnk, riid, ppv);
- }
- int
- ves_icall_System_Runtime_InteropServices_Marshal_ReleaseInternal (MonoIUnknown *pUnk)
- {
- g_assert (pUnk);
- return mono_IUnknown_Release (pUnk);
- }
- static gboolean
- cominterop_can_support_dispatch (MonoClass* klass)
- {
- if (!mono_class_is_public (klass))
- return FALSE;
- if (!cominterop_com_visible (klass))
- return FALSE;
- return TRUE;
- }
- void*
- ves_icall_System_Runtime_InteropServices_Marshal_GetIUnknownForObjectInternal (MonoObjectHandle object, MonoError *error)
- {
- return mono_cominterop_get_com_interface_internal (TRUE, object, NULL, error);
- }
- MonoObjectHandle
- ves_icall_System_Runtime_InteropServices_Marshal_GetObjectForCCW (void* pUnk, MonoError *error)
- {
- #ifndef DISABLE_COM
- /* see if it is a CCW */
- return pUnk ? cominterop_get_ccw_handle ((MonoCCWInterface*)pUnk, TRUE) : NULL_HANDLE;
- #else
- g_assert_not_reached ();
- #endif
- }
- void*
- ves_icall_System_Runtime_InteropServices_Marshal_GetIDispatchForObjectInternal (MonoObjectHandle object, MonoError *error)
- {
- #ifndef DISABLE_COM
- if (MONO_HANDLE_IS_NULL (object))
- return NULL;
- MonoRealProxyHandle real_proxy;
- if (cominterop_object_is_rcw_handle (object, &real_proxy)) {
- MonoComInteropProxyHandle com_interop_proxy = MONO_HANDLE_CAST (MonoComInteropProxy, real_proxy);
- MonoComObjectHandle com_object = MONO_HANDLE_NEW_GET (MonoComObject, com_interop_proxy, com_object);
- return cominterop_get_interface_checked (com_object, mono_class_get_idispatch_class (), error);
- }
- else if (!cominterop_can_support_dispatch (mono_handle_class (object)) ) {
- cominterop_set_hr_error (error, MONO_E_NOINTERFACE);
- return NULL;
- }
- return cominterop_get_ccw_checked (object, mono_class_get_idispatch_class (), error);
- #else
- g_assert_not_reached ();
- #endif
- }
- void*
- ves_icall_System_Runtime_InteropServices_Marshal_GetCCW (MonoObjectHandle object, MonoReflectionTypeHandle ref_type, MonoError *error)
- {
- #ifndef DISABLE_COM
- g_assert (!MONO_HANDLE_IS_NULL (ref_type));
- MonoType * const type = MONO_HANDLE_GETVAL (ref_type, type);
- g_assert (type);
- MonoClass * klass = mono_type_get_class_internal (type);
- g_assert (klass);
- if (!mono_class_init_checked (klass, error))
- return NULL;
- MonoCustomAttrInfo *cinfo = mono_custom_attrs_from_class_checked (klass, error);
- mono_error_assert_ok (error);
- if (cinfo) {
- MonoReflectionComDefaultInterfaceAttribute *attr = (MonoReflectionComDefaultInterfaceAttribute *)
- mono_custom_attrs_get_attr_checked (cinfo, mono_class_get_com_default_interface_attribute_class (), error);
- mono_error_assert_ok (error); /*FIXME proper error handling*/
- if (attr) {
- MonoType *def_itf = attr->type->type;
- if (def_itf->type == MONO_TYPE_CLASS)
- klass = mono_type_get_class_internal (def_itf);
- }
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- }
- return cominterop_get_ccw_checked (object, klass, error);
- #else
- g_assert_not_reached ();
- #endif
- }
- MonoBoolean
- ves_icall_System_Runtime_InteropServices_Marshal_IsComObject (MonoObjectHandle object, MonoError *error)
- {
- #ifndef DISABLE_COM
- MonoRealProxyHandle real_proxy;
- return (MonoBoolean)cominterop_object_is_rcw_handle (object, &real_proxy);
- #else
- g_assert_not_reached ();
- #endif
- }
- gint32
- ves_icall_System_Runtime_InteropServices_Marshal_ReleaseComObjectInternal (MonoObjectHandle object, MonoError *error)
- {
- #ifndef DISABLE_COM
- g_assert (!MONO_HANDLE_IS_NULL (object));
- MonoRealProxyHandle real_proxy;
- gboolean const is_rcw = cominterop_object_is_rcw_handle (object, &real_proxy);
- g_assert (is_rcw);
- MonoComInteropProxyHandle proxy = MONO_HANDLE_CAST (MonoComInteropProxy, real_proxy);
- g_assert (!MONO_HANDLE_IS_NULL (proxy));
- if (MONO_HANDLE_GETVAL (proxy, ref_count) == 0)
- return -1;
- gint32 ref_count = mono_atomic_dec_i32 (&MONO_HANDLE_GETVAL (proxy, ref_count));
- g_assert (ref_count >= 0);
- if (ref_count == 0)
- mono_System_ComObject_ReleaseInterfaces (MONO_HANDLE_NEW_GET (MonoComObject, proxy, com_object));
- return ref_count;
- #else
- g_assert_not_reached ();
- #endif
- }
- guint32
- ves_icall_System_Runtime_InteropServices_Marshal_GetComSlotForMethodInfoInternal (MonoReflectionMethodHandle m, MonoError *error)
- {
- #ifndef DISABLE_COM
- int const slot = cominterop_get_com_slot_for_method (MONO_HANDLE_GETVAL (m, method), error);
- mono_error_assert_ok (error);
- return slot;
- #else
- g_assert_not_reached ();
- #endif
- }
- /* Only used for COM RCWs */
- MonoObjectHandle
- ves_icall_System_ComObject_CreateRCW (MonoReflectionTypeHandle ref_type, MonoError *error)
- {
- MonoDomain * const domain = MONO_HANDLE_DOMAIN (ref_type);
- MonoType * const type = MONO_HANDLE_GETVAL (ref_type, type);
- MonoClass * const klass = mono_class_from_mono_type_internal (type);
- /* Call mono_object_new_alloc_by_vtable instead of mono_object_new_by_vtable
- * because we want to actually create object. mono_object_new_by_vtable checks
- * to see if type is import and creates transparent proxy. This method
- * is called by the corresponding real proxy to create the real RCW.
- * Constructor does not need to be called. Will be called later.
- */
- MonoVTable *vtable = mono_class_vtable_checked (domain, klass, error);
- return_val_if_nok (error, NULL_HANDLE);
- return mono_object_new_alloc_by_vtable (vtable, error);
- }
- static gboolean
- cominterop_rcw_interface_finalizer (gpointer key, gpointer value, gpointer user_data)
- {
- mono_IUnknown_Release ((MonoIUnknown*)value);
- return TRUE;
- }
- void
- mono_System_ComObject_ReleaseInterfaces (MonoComObjectHandle obj)
- {
- g_assert (!MONO_HANDLE_IS_NULL (obj));
- if (!MONO_HANDLE_GETVAL (obj, itf_hash))
- return;
- mono_cominterop_lock ();
- MonoGCHandle gchandle = (MonoGCHandle)g_hash_table_lookup (rcw_hash, MONO_HANDLE_GETVAL (obj, iunknown));
- if (gchandle) {
- mono_gchandle_free_internal (gchandle);
- g_hash_table_remove (rcw_hash, MONO_HANDLE_GETVAL (obj, iunknown));
- }
- g_hash_table_foreach_remove (MONO_HANDLE_GETVAL (obj, itf_hash), cominterop_rcw_interface_finalizer, NULL);
- g_hash_table_destroy (MONO_HANDLE_GETVAL (obj, itf_hash));
- mono_IUnknown_Release (MONO_HANDLE_GETVAL (obj, iunknown));
- MONO_HANDLE_SETVAL (obj, iunknown, MonoIUnknown*, NULL);
- MONO_HANDLE_SETVAL (obj, itf_hash, GHashTable*, NULL);
- mono_cominterop_unlock ();
- }
- void
- ves_icall_System_ComObject_ReleaseInterfaces (MonoComObjectHandle obj, MonoError *error)
- {
- mono_System_ComObject_ReleaseInterfaces (obj);
- }
- static gboolean
- cominterop_rcw_finalizer (gpointer key, gpointer value, gpointer user_data)
- {
- MonoGCHandle gchandle = NULL;
- gchandle = (MonoGCHandle)value;
- if (gchandle) {
- MonoComInteropProxy* proxy = (MonoComInteropProxy*)mono_gchandle_get_target_internal (gchandle);
- if (proxy) {
- if (proxy->com_object->itf_hash) {
- g_hash_table_foreach_remove (proxy->com_object->itf_hash, cominterop_rcw_interface_finalizer, NULL);
- g_hash_table_destroy (proxy->com_object->itf_hash);
- }
- mono_IUnknown_Release (proxy->com_object->iunknown);
- proxy->com_object->iunknown = NULL;
- proxy->com_object->itf_hash = NULL;
- }
-
- mono_gchandle_free_internal (gchandle);
- }
- return TRUE;
- }
- void
- mono_cominterop_release_all_rcws (void)
- {
- #ifndef DISABLE_COM
- if (!rcw_hash)
- return;
- mono_cominterop_lock ();
- g_hash_table_foreach_remove (rcw_hash, cominterop_rcw_finalizer, NULL);
- g_hash_table_destroy (rcw_hash);
- rcw_hash = NULL;
- mono_cominterop_unlock ();
- #endif
- }
- gpointer
- ves_icall_System_ComObject_GetInterfaceInternal (MonoComObjectHandle obj, MonoReflectionTypeHandle ref_type, MonoBoolean throw_exception, MonoError *error)
- {
- #ifndef DISABLE_COM
- MonoType * const type = MONO_HANDLE_GETVAL (ref_type, type);
- MonoClass * const klass = mono_class_from_mono_type_internal (type);
- if (!mono_class_init_checked (klass, error))
- return NULL;
- ERROR_DECL (error_ignored);
- gpointer const itf = cominterop_get_interface_checked (obj, klass, throw_exception ? error : error_ignored);
- mono_error_cleanup (error_ignored);
- return itf;
- #else
- g_assert_not_reached ();
- #endif
- }
- void
- ves_icall_Mono_Interop_ComInteropProxy_AddProxy (gpointer pUnk, MonoComInteropProxy *volatile* proxy_handle)
- {
- #ifndef DISABLE_COM
- MonoGCHandle gchandle = mono_gchandle_new_weakref_internal ((MonoObject*)*proxy_handle, FALSE);
- mono_cominterop_lock ();
- if (!rcw_hash)
- rcw_hash = g_hash_table_new (mono_aligned_addr_hash, NULL);
- g_hash_table_insert (rcw_hash, pUnk, gchandle);
- mono_cominterop_unlock ();
- #else
- g_assert_not_reached ();
- #endif
- }
- void
- ves_icall_Mono_Interop_ComInteropProxy_FindProxy (gpointer pUnk, MonoComInteropProxy *volatile* proxy_handle)
- {
- *proxy_handle = NULL;
- #ifndef DISABLE_COM
- MonoGCHandle gchandle = NULL;
- mono_cominterop_lock ();
- if (rcw_hash)
- gchandle = (MonoGCHandle)g_hash_table_lookup (rcw_hash, pUnk);
- mono_cominterop_unlock ();
- if (!gchandle)
- return;
- MonoComInteropProxy *proxy = (MonoComInteropProxy*)mono_gchandle_get_target_internal (gchandle);
- // proxy_handle is assumed to be on the stack, so no barrier is needed.
- *proxy_handle = proxy;
- /* proxy is null means we need to free up old RCW */
- if (!proxy) {
- mono_gchandle_free_internal (gchandle);
- g_hash_table_remove (rcw_hash, pUnk);
- }
- #else
- g_assert_not_reached ();
- #endif
- }
- /**
- * cominterop_get_ccw_object:
- * @ccw_entry: a pointer to the CCWEntry
- * @verify: verify ccw_entry is in fact a ccw
- *
- * Returns: the corresponding object for the CCW
- */
- static MonoGCHandle
- cominterop_get_ccw_gchandle (MonoCCWInterface* ccw_entry, gboolean verify)
- {
- /* no CCW's exist yet */
- if (!ccw_interface_hash)
- return 0;
- MonoCCW * const ccw = verify ? (MonoCCW *)g_hash_table_lookup (ccw_interface_hash, ccw_entry) : ccw_entry->ccw;
- g_assert (verify || ccw);
- return ccw ? ccw->gc_handle : 0;
- }
- static MonoObjectHandle
- cominterop_get_ccw_handle (MonoCCWInterface* ccw_entry, gboolean verify)
- {
- MonoGCHandle const gchandle = cominterop_get_ccw_gchandle (ccw_entry, verify);
- return gchandle ? mono_gchandle_get_target_handle (gchandle) : NULL_HANDLE;
- }
- static MonoObject*
- cominterop_get_ccw_object (MonoCCWInterface* ccw_entry, gboolean verify)
- {
- MonoGCHandle const gchandle = cominterop_get_ccw_gchandle (ccw_entry, verify);
- return gchandle ? mono_gchandle_get_target_internal (gchandle) : NULL;
- }
- static MonoDomain*
- cominterop_get_domain_for_appdomain (MonoAppDomain *ad_raw)
- {
- HANDLE_FUNCTION_ENTER ();
- MONO_HANDLE_DCL (MonoAppDomain, ad);
- MonoDomain * result = MONO_HANDLE_GETVAL (ad, data);
- HANDLE_FUNCTION_RETURN_VAL (result);
- }
- static MonoObject*
- cominterop_set_ccw_object_domain (MonoObject *object, MonoDomain **prev_domain)
- {
- MonoDomain *current = mono_domain_get (), *obj_domain;
- if (mono_object_class (object) == mono_defaults.appdomain_class)
- obj_domain = cominterop_get_domain_for_appdomain ((MonoAppDomain *)object);
- else
- obj_domain = mono_object_domain (object);
- if (obj_domain != current) {
- *prev_domain = current;
- mono_domain_set_internal_with_options (obj_domain, FALSE);
- }
- else
- *prev_domain = NULL;
- return object;
- }
- static void
- cominterop_restore_domain (MonoDomain *domain)
- {
- if (!domain)
- return;
- mono_domain_set_internal_with_options (domain, FALSE);
- }
- static void
- cominterop_setup_marshal_context (EmitMarshalContext *m, MonoMethod *method)
- {
- MonoMethodSignature *sig, *csig;
- MonoImage *method_klass_image = m_class_get_image (method->klass);
- sig = mono_method_signature_internal (method);
- /* we copy the signature, so that we can modify it */
- /* FIXME: which to use? */
- csig = mono_metadata_signature_dup_full (method_klass_image, sig);
- /* csig = mono_metadata_signature_dup (sig); */
-
- /* STDCALL on windows, CDECL everywhere else to work with XPCOM and MainWin COM */
- #ifdef HOST_WIN32
- csig->call_convention = MONO_CALL_STDCALL;
- #else
- csig->call_convention = MONO_CALL_C;
- #endif
- csig->hasthis = 0;
- csig->pinvoke = 1;
- m->image = method_klass_image;
- m->piinfo = NULL;
- m->retobj_var = 0;
- m->sig = sig;
- m->csig = csig;
- }
- static MonoMarshalSpec*
- cominterop_get_ccw_default_mspec (const MonoType *param_type)
- {
- MonoMarshalVariant elem_type;
- MonoMarshalNative native;
- MonoMarshalSpec *result;
- switch (param_type->type) {
- case MONO_TYPE_OBJECT:
- native = MONO_NATIVE_STRUCT;
- break;
- case MONO_TYPE_STRING:
- native = MONO_NATIVE_BSTR;
- break;
- case MONO_TYPE_CLASS:
- native = MONO_NATIVE_INTERFACE;
- break;
- case MONO_TYPE_BOOLEAN:
- native = MONO_NATIVE_VARIANTBOOL;
- break;
- case MONO_TYPE_SZARRAY:
- /* object[] -> SAFEARRAY(VARIANT) */
- native = MONO_NATIVE_SAFEARRAY;
- if (param_type->data.array->eklass == mono_defaults.object_class)
- elem_type = MONO_VARIANT_VARIANT;
- else
- return NULL;
- break;
- default:
- return NULL;
- }
- result = g_new0 (MonoMarshalSpec, 1);
- result->native = native;
- if (native == MONO_NATIVE_SAFEARRAY)
- result->data.safearray_data.elem_type = elem_type;
- return result;
- }
- static MonoClass*
- cominterop_get_default_iface (MonoClass *klass)
- {
- if (mono_class_is_interface (klass))
- return klass;
- ERROR_DECL (error);
- MonoCustomAttrInfo *cinfo = mono_custom_attrs_from_class_checked (klass, error);
- mono_error_assert_ok (error);
- if (!cinfo)
- return mono_class_get_idispatch_class ();
- MonoClassInterfaceAttribute *class_attr = (MonoClassInterfaceAttribute *)mono_custom_attrs_get_attr_checked (cinfo, mono_class_get_class_interface_attribute_class (), error);
- MonoClass *ret;
- if (class_attr)
- {
- if (class_attr->intType == 0) {
- ret = mono_defaults.object_class;
- for (guint16 i = 0; i < m_class_get_interface_count (klass); i++) {
- MonoClass *iface = m_class_get_interfaces (klass) [i];
- if (cominterop_com_visible (iface)) {
- ret = iface;
- break;
- }
- }
- }
- else if (class_attr->intType == 1)
- ret = mono_class_get_idispatch_class ();
- else
- ret = klass;
- } else
- ret = mono_class_get_idispatch_class ();
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- return ret;
- }
- static gboolean
- cominterop_class_method_is_visible (MonoMethod *method)
- {
- guint16 flags = method->flags;
- if ((flags & METHOD_ATTRIBUTE_MEMBER_ACCESS_MASK) != METHOD_ATTRIBUTE_PUBLIC)
- return FALSE;
- if (flags & METHOD_ATTRIBUTE_STATIC)
- return FALSE;
- if (flags & METHOD_ATTRIBUTE_RT_SPECIAL_NAME)
- return FALSE;
- if (!mono_cominterop_method_com_visible (method))
- return FALSE;
- /* if the method is an override, ignore it and use the original definition */
- if ((flags & METHOD_ATTRIBUTE_VIRTUAL) && !(flags & METHOD_ATTRIBUTE_NEW_SLOT))
- return FALSE;
- return TRUE;
- }
- static gpointer
- cominterop_get_ccw_method (MonoClass *iface, MonoMethod *method, MonoError *error)
- {
- int param_index = 0;
- MonoMethodBuilder *mb;
- MonoMarshalSpec ** mspecs;
- MonoMethod *wrapper_method, *adjust_method;
- MonoMethodSignature* sig_adjusted;
- MonoMethodSignature* sig = mono_method_signature_internal (method);
- gboolean const preserve_sig = (method->iflags & METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG) != 0;
- EmitMarshalContext m;
- mb = mono_mb_new (iface, method->name, MONO_WRAPPER_NATIVE_TO_MANAGED);
- adjust_method = cominterop_get_managed_wrapper_adjusted (method);
- sig_adjusted = mono_method_signature_internal (adjust_method);
- mspecs = g_new (MonoMarshalSpec*, sig_adjusted->param_count + 1);
- mono_method_get_marshal_info (method, mspecs);
- /* move managed args up one */
- for (param_index = sig->param_count; param_index >= 1; param_index--) {
- int mspec_index = param_index+1;
- mspecs [mspec_index] = mspecs [param_index];
- if (mspecs[mspec_index] == NULL) {
- mspecs[mspec_index] = cominterop_get_ccw_default_mspec (sig_adjusted->params[param_index]);
- } else {
- /* increase SizeParamIndex since we've added a param */
- if (sig_adjusted->params[param_index]->type == MONO_TYPE_ARRAY ||
- sig_adjusted->params[param_index]->type == MONO_TYPE_SZARRAY)
- if (mspecs[mspec_index]->data.array_data.param_num != -1)
- mspecs[mspec_index]->data.array_data.param_num++;
- }
- }
- /* first arg is IntPtr for interface */
- mspecs [1] = NULL;
- /* move return spec to last param */
- if (!preserve_sig && !MONO_TYPE_IS_VOID (sig->ret)) {
- if (mspecs [0] == NULL)
- mspecs[0] = cominterop_get_ccw_default_mspec (sig_adjusted->params[sig_adjusted->param_count-1]);
- mspecs [sig_adjusted->param_count] = mspecs [0];
- mspecs [0] = NULL;
- }
- #ifndef DISABLE_JIT
- /* skip visiblity since we call internal methods */
- mb->skip_visibility = TRUE;
- #endif
- cominterop_setup_marshal_context (&m, adjust_method);
- m.mb = mb;
- mono_marshal_emit_managed_wrapper (mb, sig_adjusted, mspecs, &m, adjust_method, 0);
- mono_cominterop_lock ();
- wrapper_method = mono_mb_create_method (mb, m.csig, m.csig->param_count + 16);
- mono_cominterop_unlock ();
- gpointer ret = mono_compile_method_checked (wrapper_method, error);
- mono_mb_free (mb);
- for (param_index = sig_adjusted->param_count; param_index >= 0; param_index--)
- if (mspecs [param_index])
- mono_metadata_free_marshal_spec (mspecs [param_index]);
- g_free (mspecs);
- return ret;
- }
- /**
- * cominterop_get_ccw_checked:
- * @object: a pointer to the object
- * @itf: interface type needed
- * @error: set on error
- *
- * Returns: a value indicating if the object is a
- * Runtime Callable Wrapper (RCW) for a COM object.
- * On failure returns NULL and sets @error.
- */
- static gpointer
- cominterop_get_ccw_checked (MonoObjectHandle object, MonoClass* itf, MonoError *error)
- {
- int i, j;
- MonoCCW *ccw = NULL;
- MonoCCWInterface* ccw_entry = NULL;
- gpointer *vtable = NULL;
- MonoClass* iface = NULL;
- int start_slot = 3;
- int method_count = 0;
- GList *ccw_list, *ccw_list_item;
- MonoCustomAttrInfo *cinfo = NULL;
- if (MONO_HANDLE_IS_NULL (object))
- return NULL;
- MonoClass* klass = mono_handle_class (object);
- mono_cominterop_lock ();
- if (!ccw_hash)
- ccw_hash = g_hash_table_new (mono_aligned_addr_hash, NULL);
- if (!ccw_interface_hash)
- ccw_interface_hash = g_hash_table_new (mono_aligned_addr_hash, NULL);
- ccw_list = (GList *)g_hash_table_lookup (ccw_hash, GINT_TO_POINTER (mono_handle_hash (object)));
- mono_cominterop_unlock ();
- ccw_list_item = ccw_list;
- while (ccw_list_item) {
- MonoCCW* ccw_iter = (MonoCCW *)ccw_list_item->data;
- if (mono_gchandle_target_equal (ccw_iter->gc_handle, object)) {
- ccw = ccw_iter;
- break;
- }
- ccw_list_item = g_list_next(ccw_list_item);
- }
- if (!ccw) {
- ccw = g_new0 (MonoCCW, 1);
- #ifdef HOST_WIN32
- ccw->free_marshaler = 0;
- #endif
- ccw->vtable_hash = g_hash_table_new (mono_aligned_addr_hash, NULL);
- ccw->ref_count = 0;
- /* just alloc a weak handle until we are addref'd*/
- ccw->gc_handle = mono_gchandle_new_weakref_from_handle (object);
- if (!ccw_list) {
- ccw_list = g_list_alloc ();
- ccw_list->data = ccw;
- }
- else
- ccw_list = g_list_append (ccw_list, ccw);
- mono_cominterop_lock ();
- g_hash_table_insert (ccw_hash, GINT_TO_POINTER (mono_handle_hash (object)), ccw_list);
- mono_cominterop_unlock ();
- /* register for finalization to clean up ccw */
- mono_object_register_finalizer_handle (object);
- }
- cinfo = mono_custom_attrs_from_class_checked (itf, error);
- mono_error_assert_ok (error);
- if (cinfo) {
- MONO_STATIC_POINTER_INIT (MonoClass, coclass_attribute)
- coclass_attribute = mono_class_load_from_name (mono_defaults.corlib, "System.Runtime.InteropServices", "CoClassAttribute");
- MONO_STATIC_POINTER_INIT_END (MonoClass, coclass_attribute)
- if (mono_custom_attrs_has_attr (cinfo, coclass_attribute)) {
- g_assert(m_class_get_interface_count (itf) && m_class_get_interfaces (itf)[0]);
- itf = m_class_get_interfaces (itf)[0];
- }
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- }
- iface = cominterop_get_default_iface(itf);
- if (iface == mono_class_get_iunknown_class ()) {
- start_slot = 3;
- }
- else if (iface == mono_class_get_idispatch_class ()) {
- start_slot = 7;
- }
- else if (mono_class_is_interface (iface)) {
- method_count += mono_class_get_method_count (iface);
- start_slot = cominterop_get_com_slot_begin (iface);
- }
- else {
- /* auto-dual object */
- start_slot = 7;
- MonoClass *klass_iter;
- for (klass_iter = iface; klass_iter; klass_iter = m_class_get_parent (klass_iter)) {
- int mcount = mono_class_get_method_count (klass_iter);
- if (mcount && !m_class_get_methods (klass_iter))
- mono_class_setup_methods (klass_iter);
- for (i = 0; i < mcount; ++i) {
- MonoMethod *method = m_class_get_methods (klass_iter) [i];
- if (cominterop_class_method_is_visible (method))
- ++method_count;
- }
- /* FIXME: accessors for public fields */
- }
- }
- ccw_entry = (MonoCCWInterface *)g_hash_table_lookup (ccw->vtable_hash, itf);
- if (!ccw_entry) {
- int vtable_index = method_count-1+start_slot;
- vtable = (void **)mono_image_alloc0 (m_class_get_image (klass), sizeof (gpointer)*(method_count+start_slot));
- vtable [0] = (gpointer)cominterop_ccw_queryinterface;
- vtable [1] = (gpointer)cominterop_ccw_addref;
- vtable [2] = (gpointer)cominterop_ccw_release;
- if (start_slot == 7) {
- vtable [3] = (gpointer)cominterop_ccw_get_type_info_count;
- vtable [4] = (gpointer)cominterop_ccw_get_type_info;
- vtable [5] = (gpointer)cominterop_ccw_get_ids_of_names;
- vtable [6] = (gpointer)cominterop_ccw_invoke;
- }
- if (mono_class_is_interface (iface)) {
- if (method_count && !m_class_get_methods (iface))
- mono_class_setup_methods (iface);
- for (i = method_count - 1; i >= 0; i--) {
- vtable [vtable_index--] = cominterop_get_ccw_method (iface, m_class_get_methods (iface) [i], error);
- return_val_if_nok (error, NULL);
- }
- }
- else {
- /* Auto-dual object. The methods on an auto-dual interface are
- * exposed starting from the innermost parent (i.e. Object) and
- * proceeding outwards. The methods within each interfaces are
- * exposed in the following order:
- *
- * 1. Virtual methods
- * 2. Interface methods
- * 3. Nonvirtual methods
- * 4. Fields (get, then put)
- *
- * Interface methods are exposed in the order that the interface
- * was declared. Child interface methods are exposed before parents.
- *
- * Because we need to expose superclass methods starting from the
- * innermost parent, we expose methods in reverse order, so that
- * we can just iterate using m_class_get_parent (). */
- mono_class_setup_vtable (iface);
- MonoClass *klass_iter;
- for (klass_iter = iface; klass_iter; klass_iter = m_class_get_parent (klass_iter)) {
- mono_class_setup_vtable (klass_iter);
- /* 3. Nonvirtual methods */
- for (i = mono_class_get_method_count (klass_iter) - 1; i >= 0; i--) {
- MonoMethod *method = m_class_get_methods (klass_iter) [i];
- if (cominterop_class_method_is_visible (method) && !(method->flags & METHOD_ATTRIBUTE_VIRTUAL)) {
- vtable [vtable_index--] = cominterop_get_ccw_method (iface, method, error);
- return_val_if_nok (error, NULL);
- }
- }
- /* 2. Interface methods */
- GPtrArray *ifaces = mono_class_get_implemented_interfaces (klass_iter, error);
- mono_error_assert_ok (error);
- if (ifaces) {
- for (i = ifaces->len - 1; i >= 0; i--) {
- MonoClass *ic = (MonoClass *)g_ptr_array_index (ifaces, i);
- int offset = mono_class_interface_offset (iface, ic);
- g_assert (offset >= 0);
- for (j = mono_class_get_method_count (ic) - 1; j >= 0; j--) {
- MonoMethod *method = m_class_get_methods (ic) [j];
- vtable [vtable_index--] = cominterop_get_ccw_method (iface, m_class_get_vtable (iface) [offset + method->slot], error);
- if (!is_ok (error)) {
- g_ptr_array_free (ifaces, TRUE);
- return NULL;
- }
- }
- }
- g_ptr_array_free (ifaces, TRUE);
- }
- /* 1. Virtual methods */
- for (i = mono_class_get_method_count (klass_iter) - 1; i >= 0; i--) {
- MonoMethod *method = m_class_get_methods (klass_iter) [i];
- if (cominterop_class_method_is_visible (method) && (method->flags & METHOD_ATTRIBUTE_VIRTUAL)
- && !cominterop_get_method_interface (method)) {
- vtable [vtable_index--] = cominterop_get_ccw_method (iface, m_class_get_vtable (iface) [method->slot], error);
- return_val_if_nok (error, NULL);
- }
- }
- }
- }
- ccw_entry = g_new0 (MonoCCWInterface, 1);
- ccw_entry->ccw = ccw;
- ccw_entry->vtable = vtable;
- g_hash_table_insert (ccw->vtable_hash, itf, ccw_entry);
- g_hash_table_insert (ccw_interface_hash, ccw_entry, ccw);
- }
- return ccw_entry;
- }
- /**
- * cominterop_get_ccw:
- * @object: a pointer to the object
- * @itf: interface type needed
- *
- * Returns: a value indicating if the object is a
- * Runtime Callable Wrapper (RCW) for a COM object
- */
- static gpointer
- cominterop_get_ccw (MonoObject* object_raw, MonoClass* itf)
- {
- HANDLE_FUNCTION_ENTER ();
- ERROR_DECL (error);
- MONO_HANDLE_DCL (MonoObject, object);
- gpointer const ccw_entry = cominterop_get_ccw_checked (object, itf, error);
- mono_error_set_pending_exception (error);
- HANDLE_FUNCTION_RETURN_VAL (ccw_entry);
- }
- static gboolean
- mono_marshal_free_ccw_entry (gpointer key, gpointer value, gpointer user_data)
- {
- g_hash_table_remove (ccw_interface_hash, value);
- g_assert (value);
- g_free (value);
- return TRUE;
- }
- /**
- * mono_marshal_free_ccw:
- * \param object the mono object
- * \returns whether the object had a CCW
- */
- static gboolean
- mono_marshal_free_ccw_handle (MonoObjectHandle object)
- {
- /* no ccw's were created */
- if (!ccw_hash || g_hash_table_size (ccw_hash) == 0)
- return FALSE;
- mono_cominterop_lock ();
- GList *ccw_list = (GList *)g_hash_table_lookup (ccw_hash, GINT_TO_POINTER (mono_handle_hash (object)));
- mono_cominterop_unlock ();
- if (!ccw_list)
- return FALSE;
- /* need to cache orig list address to remove from hash_table if empty */
- GList * const ccw_list_orig = ccw_list;
- for (GList* ccw_list_item = ccw_list; ccw_list_item; ) {
- MonoCCW* ccw_iter = (MonoCCW *)ccw_list_item->data;
- gboolean is_null = FALSE;
- gboolean is_equal = FALSE;
- mono_gchandle_target_is_null_or_equal (ccw_iter->gc_handle, object, &is_null, &is_equal);
- /* Looks like the GC NULLs the weakref handle target before running the
- * finalizer. So if we get a NULL target, destroy the CCW as well.
- * Unless looking up the object from the CCW shows it not the right object.
- */
- gboolean destroy_ccw = is_null || is_equal;
- if (is_null) {
- MonoCCWInterface* ccw_entry = (MonoCCWInterface *)g_hash_table_lookup (ccw_iter->vtable_hash, mono_class_get_iunknown_class ());
- MonoGCHandle gchandle = NULL;
- if (!(ccw_entry && (gchandle = cominterop_get_ccw_gchandle (ccw_entry, FALSE)) && mono_gchandle_target_equal (gchandle, object)))
- destroy_ccw = FALSE;
- }
- if (destroy_ccw) {
- /* remove all interfaces */
- g_hash_table_foreach_remove (ccw_iter->vtable_hash, mono_marshal_free_ccw_entry, NULL);
- g_hash_table_destroy (ccw_iter->vtable_hash);
- /* get next before we delete */
- ccw_list_item = g_list_next (ccw_list_item);
- /* remove ccw from list */
- ccw_list = g_list_remove (ccw_list, ccw_iter);
- #ifdef HOST_WIN32
- mono_IUnknown_Release (ccw_iter->free_marshaler);
- #endif
- g_free (ccw_iter);
- }
- else
- ccw_list_item = g_list_next (ccw_list_item);
- }
- /* if list is empty remove original address from hash */
- if (g_list_length (ccw_list) == 0)
- g_hash_table_remove (ccw_hash, GINT_TO_POINTER (mono_handle_hash (object)));
- else if (ccw_list != ccw_list_orig)
- g_hash_table_insert (ccw_hash, GINT_TO_POINTER (mono_handle_hash (object)), ccw_list);
- return TRUE;
- }
- gboolean
- mono_marshal_free_ccw (MonoObject* object_raw)
- {
- /* no ccw's were created */
- if (!ccw_hash || g_hash_table_size (ccw_hash) == 0)
- return FALSE;
- HANDLE_FUNCTION_ENTER ();
- MONO_HANDLE_DCL (MonoObject, object);
- gboolean const result = mono_marshal_free_ccw_handle (object);
- HANDLE_FUNCTION_RETURN_VAL (result);
- }
- /**
- * cominterop_get_managed_wrapper_adjusted:
- * @method: managed COM Interop method
- *
- * Returns: the generated method to call with signature matching
- * the unmanaged COM Method signature
- */
- static MonoMethod *
- cominterop_get_managed_wrapper_adjusted (MonoMethod *method)
- {
- MonoMethod *res = NULL;
- MonoMethodBuilder *mb;
- MonoMarshalSpec **mspecs;
- MonoMethodSignature *sig, *sig_native;
- MonoExceptionClause *main_clause = NULL;
- int hr = 0, retval = 0;
- int pos_leave, domain_var;
- int i;
- gboolean const preserve_sig = (method->iflags & METHOD_IMPL_ATTRIBUTE_PRESERVE_SIG) != 0;
- MonoType *int_type = mono_get_int_type ();
- MONO_STATIC_POINTER_INIT (MonoMethod, get_hr_for_exception)
- ERROR_DECL (error);
- get_hr_for_exception = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetHRForException", -1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_hr_for_exception)
- sig = mono_method_signature_internal (method);
- /* create unmanaged wrapper */
- mb = mono_mb_new (method->klass, method->name, MONO_WRAPPER_COMINTEROP);
- sig_native = cominterop_method_signature (method);
- mspecs = g_new0 (MonoMarshalSpec*, sig_native->param_count+1);
- mono_method_get_marshal_info (method, mspecs);
- /* move managed args up one */
- for (i = sig->param_count; i >= 1; i--)
- mspecs [i+1] = mspecs [i];
- /* first arg is IntPtr for interface */
- mspecs [1] = NULL;
- /* move return spec to last param */
- if (!preserve_sig && !MONO_TYPE_IS_VOID (sig->ret))
- mspecs [sig_native->param_count] = mspecs [0];
- mspecs [0] = NULL;
- #ifndef DISABLE_JIT
- if (!preserve_sig) {
- if (!MONO_TYPE_IS_VOID (sig->ret))
- retval = mono_mb_add_local (mb, sig->ret);
- hr = mono_mb_add_local (mb, mono_get_int32_type ());
- }
- else if (!MONO_TYPE_IS_VOID (sig->ret))
- hr = mono_mb_add_local (mb, sig->ret);
- /* try */
- main_clause = g_new0 (MonoExceptionClause, 1);
- main_clause->try_offset = mono_mb_get_label (mb);
- domain_var = mono_mb_add_local (mb, int_type);
- /* the CCW -> object conversion */
- mono_mb_emit_ldarg (mb, 0);
- mono_mb_emit_icon (mb, FALSE);
- mono_mb_emit_icall (mb, cominterop_get_ccw_object);
- /* Object is left on stack */
- mono_mb_emit_ldloc_addr (mb, domain_var);
- mono_mb_emit_icall (mb, cominterop_set_ccw_object_domain);
- for (i = 0; i < sig->param_count; i++)
- mono_mb_emit_ldarg (mb, i+1);
- mono_mb_emit_managed_call (mb, method, NULL);
- if (!MONO_TYPE_IS_VOID (sig->ret)) {
- if (!preserve_sig) {
- mono_mb_emit_stloc (mb, retval);
- mono_mb_emit_ldarg (mb, sig_native->param_count - 1);
- const int pos_null = mono_mb_emit_branch (mb, CEE_BRFALSE);
- mono_mb_emit_ldarg (mb, sig_native->param_count - 1);
- mono_mb_emit_ldloc (mb, retval);
- MonoClass *rclass = mono_class_from_mono_type_internal (sig->ret);
- if (m_class_is_valuetype (rclass)) {
- mono_mb_emit_op (mb, CEE_STOBJ, rclass);
- } else {
- mono_mb_emit_byte (mb, mono_type_to_stind (sig->ret));
- }
- mono_mb_patch_branch (mb, pos_null);
- } else
- mono_mb_emit_stloc (mb, hr);
- }
- pos_leave = mono_mb_emit_branch (mb, CEE_LEAVE);
- /* Main exception catch */
- main_clause->flags = MONO_EXCEPTION_CLAUSE_NONE;
- main_clause->try_len = mono_mb_get_pos (mb) - main_clause->try_offset;
- main_clause->data.catch_class = mono_defaults.object_class;
-
- /* handler code */
- main_clause->handler_offset = mono_mb_get_label (mb);
-
- if (!preserve_sig || (sig->ret && !sig->ret->byref && (sig->ret->type == MONO_TYPE_U4 || sig->ret->type == MONO_TYPE_I4))) {
- mono_mb_emit_managed_call (mb, get_hr_for_exception, NULL);
- mono_mb_emit_stloc (mb, hr);
- }
- else {
- mono_mb_emit_byte (mb, CEE_POP);
- }
- mono_mb_emit_branch (mb, CEE_LEAVE);
- main_clause->handler_len = mono_mb_get_pos (mb) - main_clause->handler_offset;
- /* end catch */
- mono_mb_set_clauses (mb, 1, main_clause);
- mono_mb_patch_branch (mb, pos_leave);
- if (!preserve_sig || !MONO_TYPE_IS_VOID (sig->ret))
- mono_mb_emit_ldloc (mb, hr);
- mono_mb_emit_ldloc (mb, domain_var);
- mono_mb_emit_icall (mb, cominterop_restore_domain);
- mono_mb_emit_byte (mb, CEE_RET);
- #endif /* DISABLE_JIT */
- mono_cominterop_lock ();
- res = mono_mb_create_method (mb, sig_native, sig_native->param_count + 16);
- mono_cominterop_unlock ();
- mono_mb_free (mb);
- for (i = sig_native->param_count; i >= 0; i--)
- mono_metadata_free_marshal_spec (mspecs [i]);
- g_free (mspecs);
- return res;
- }
- static gboolean
- cominterop_class_guid_equal (const guint8* guid, MonoClass* klass)
- {
- guint8 klass_guid [16];
- if (cominterop_class_guid (klass, klass_guid))
- return !memcmp (guid, klass_guid, sizeof (klass_guid));
- return FALSE;
- }
- static int STDCALL
- cominterop_ccw_addref_impl (MonoCCWInterface* ccwe);
- static int STDCALL
- cominterop_ccw_addref (MonoCCWInterface* ccwe)
- {
- int result;
- gpointer dummy;
- gpointer orig_domain = mono_threads_attach_coop (mono_domain_get (), &dummy);
- MONO_ENTER_GC_UNSAFE;
- result = cominterop_ccw_addref_impl (ccwe);
- MONO_EXIT_GC_UNSAFE;
- mono_threads_detach_coop (orig_domain, &dummy);
- return result;
- }
- static int STDCALL
- cominterop_ccw_addref_impl (MonoCCWInterface* ccwe)
- {
- MONO_REQ_GC_UNSAFE_MODE;
- MonoCCW* ccw = ccwe->ccw;
- g_assert (ccw);
- g_assert (ccw->gc_handle);
- gint32 const ref_count = mono_atomic_inc_i32 ((gint32*)&ccw->ref_count);
- if (ref_count == 1) {
- MonoGCHandle oldhandle = ccw->gc_handle;
- g_assert (oldhandle);
- /* since we now have a ref count, alloc a strong handle*/
- ccw->gc_handle = mono_gchandle_from_handle (mono_gchandle_get_target_handle (oldhandle), FALSE);
- mono_gchandle_free_internal (oldhandle);
- }
- return ref_count;
- }
- static int STDCALL
- cominterop_ccw_release_impl (MonoCCWInterface* ccwe);
- static int STDCALL
- cominterop_ccw_release (MonoCCWInterface* ccwe)
- {
- int result;
- gpointer dummy;
- gpointer orig_domain = mono_threads_attach_coop (mono_domain_get (), &dummy);
- MONO_ENTER_GC_UNSAFE;
- result = cominterop_ccw_release_impl (ccwe);
- MONO_EXIT_GC_UNSAFE;
- mono_threads_detach_coop (orig_domain, &dummy);
- return result;
- }
- static int STDCALL
- cominterop_ccw_release_impl (MonoCCWInterface* ccwe)
- {
- MONO_REQ_GC_UNSAFE_MODE;
- MonoCCW* ccw = ccwe->ccw;
- g_assert (ccw);
- g_assert (ccw->ref_count > 0);
- gint32 const ref_count = mono_atomic_dec_i32 ((gint32*)&ccw->ref_count);
- if (ref_count == 0) {
- /* allow gc of object */
- MonoGCHandle oldhandle = ccw->gc_handle;
- g_assert (oldhandle);
- ccw->gc_handle = mono_gchandle_new_weakref_from_handle (mono_gchandle_get_target_handle (oldhandle));
- mono_gchandle_free_internal (oldhandle);
- }
- return ref_count;
- }
- #ifdef HOST_WIN32
- static const IID MONO_IID_IMarshal = {0x3, 0x0, 0x0, {0xC0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x46}};
- /* All ccw objects are free threaded */
- static int
- cominterop_ccw_getfreethreadedmarshaler (MonoCCW* ccw, MonoObjectHandle object, gpointer* ppv, MonoError *error)
- {
- if (!ccw->free_marshaler) {
- gpointer const tunk = cominterop_get_ccw_checked (object, mono_class_get_iunknown_class (), error);
- return_val_if_nok (error, MONO_E_NOINTERFACE);
- int const ret = CoCreateFreeThreadedMarshaler ((LPUNKNOWN)tunk, (LPUNKNOWN*)&ccw->free_marshaler);
- }
- return ccw->free_marshaler ? mono_IUnknown_QueryInterface (ccw->free_marshaler, &MONO_IID_IMarshal, ppv)
- : MONO_E_NOINTERFACE;
- }
- #endif
- static int STDCALL
- cominterop_ccw_queryinterface_impl (MonoCCWInterface* ccwe, const guint8* riid, gpointer* ppv);
- static int STDCALL
- cominterop_ccw_queryinterface (MonoCCWInterface* ccwe, const guint8* riid, gpointer* ppv)
- {
- int result;
- gpointer dummy;
- gpointer orig_domain = mono_threads_attach_coop (mono_domain_get (), &dummy);
- MONO_ENTER_GC_UNSAFE;
- result = cominterop_ccw_queryinterface_impl (ccwe, riid, ppv);
- MONO_EXIT_GC_UNSAFE;
- mono_threads_detach_coop (orig_domain, &dummy);
- return result;
- }
- static int STDCALL
- cominterop_ccw_queryinterface_impl (MonoCCWInterface* ccwe, const guint8* riid, gpointer* ppv)
- {
- MONO_REQ_GC_UNSAFE_MODE;
- ERROR_DECL (error);
- GPtrArray *ifaces;
- MonoClass *itf = NULL;
- int i;
- MonoCCW* ccw = ccwe->ccw;
- MonoClass* klass_iter = NULL;
- MonoObjectHandle object = mono_gchandle_get_target_handle (ccw->gc_handle);
-
- g_assert (!MONO_HANDLE_IS_NULL (object));
- MonoClass* const klass = mono_handle_class (object);
- if (ppv)
- *ppv = NULL;
- if (!mono_domain_get ())
- mono_thread_attach_external_native_thread (mono_get_root_domain (), FALSE);
- /* handle IUnknown special */
- if (cominterop_class_guid_equal (riid, mono_class_get_iunknown_class ())) {
- *ppv = cominterop_get_ccw_checked (object, mono_class_get_iunknown_class (), error);
- mono_error_assert_ok (error);
- /* remember to addref on QI */
- cominterop_ccw_addref_impl ((MonoCCWInterface *)*ppv);
- return MONO_S_OK;
- }
- /* handle IDispatch special */
- if (cominterop_class_guid_equal (riid, mono_class_get_idispatch_class ())) {
- if (!cominterop_can_support_dispatch (klass))
- return MONO_E_NOINTERFACE;
-
- *ppv = cominterop_get_ccw_checked (object, mono_class_get_idispatch_class (), error);
- mono_error_assert_ok (error);
- /* remember to addref on QI */
- cominterop_ccw_addref_impl ((MonoCCWInterface *)*ppv);
- return MONO_S_OK;
- }
- #ifdef HOST_WIN32
- /* handle IMarshal special */
- if (0 == memcmp (riid, &MONO_IID_IMarshal, sizeof (IID))) {
- int const res = cominterop_ccw_getfreethreadedmarshaler (ccw, object, ppv, error);
- mono_error_assert_ok (error);
- return res;
- }
- #endif
- klass_iter = klass;
- while (klass_iter && klass_iter != mono_defaults.object_class) {
- ifaces = mono_class_get_implemented_interfaces (klass_iter, error);
- mono_error_assert_ok (error);
- if (ifaces) {
- for (i = 0; i < ifaces->len; ++i) {
- MonoClass *ic = NULL;
- ic = (MonoClass *)g_ptr_array_index (ifaces, i);
- if (cominterop_class_guid_equal (riid, ic)) {
- itf = ic;
- break;
- }
- }
- g_ptr_array_free (ifaces, TRUE);
- }
- if (itf)
- break;
- klass_iter = m_class_get_parent (klass_iter);
- }
- if (itf) {
- *ppv = cominterop_get_ccw_checked (object, itf, error);
- if (!is_ok (error)) {
- mono_error_cleanup (error); /* FIXME don't swallow the error */
- return MONO_E_NOINTERFACE;
- }
- /* remember to addref on QI */
- cominterop_ccw_addref_impl ((MonoCCWInterface *)*ppv);
- return MONO_S_OK;
- }
- return MONO_E_NOINTERFACE;
- }
- static int STDCALL
- cominterop_ccw_get_type_info_count (MonoCCWInterface* ccwe, guint32 *pctinfo)
- {
- if(!pctinfo)
- return MONO_E_INVALIDARG;
- *pctinfo = 1;
- return MONO_S_OK;
- }
- static int STDCALL
- cominterop_ccw_get_type_info (MonoCCWInterface* ccwe, guint32 iTInfo, guint32 lcid, gpointer *ppTInfo)
- {
- return MONO_E_NOTIMPL;
- }
- static int STDCALL
- cominterop_ccw_get_ids_of_names_impl (MonoCCWInterface* ccwe, gpointer riid,
- gunichar2** rgszNames, guint32 cNames,
- guint32 lcid, gint32 *rgDispId);
- static int STDCALL
- cominterop_ccw_get_ids_of_names (MonoCCWInterface* ccwe, gpointer riid,
- gunichar2** rgszNames, guint32 cNames,
- guint32 lcid, gint32 *rgDispId)
- {
- int result;
- gpointer dummy;
- gpointer orig_domain = mono_threads_attach_coop (mono_domain_get(), &dummy);
- MONO_ENTER_GC_UNSAFE;
- result = cominterop_ccw_get_ids_of_names_impl (ccwe, riid, rgszNames, cNames, lcid, rgDispId);
- MONO_EXIT_GC_UNSAFE;
- mono_threads_detach_coop (orig_domain, &dummy);
- return result;
- }
- static int STDCALL
- cominterop_ccw_get_ids_of_names_impl (MonoCCWInterface* ccwe, gpointer riid,
- gunichar2** rgszNames, guint32 cNames,
- guint32 lcid, gint32 *rgDispId)
- {
- MONO_REQ_GC_UNSAFE_MODE;
- ERROR_DECL (error);
- MonoCustomAttrInfo *cinfo = NULL;
- int i,ret = MONO_S_OK;
- MonoMethod* method;
- gchar* methodname;
- MonoClass *klass = NULL;
- MonoCCW* ccw = ccwe->ccw;
- MonoObject* object = mono_gchandle_get_target_internal (ccw->gc_handle);
- /* Handle DispIdAttribute */
- MONO_STATIC_POINTER_INIT (MonoClass, ComDispIdAttribute)
- ComDispIdAttribute = mono_class_load_from_name (mono_defaults.corlib, "System.Runtime.InteropServices", "DispIdAttribute");
- MONO_STATIC_POINTER_INIT_END (MonoClass, ComDispIdAttribute)
- g_assert (object);
- klass = mono_object_class (object);
- if (!mono_domain_get ())
- mono_thread_attach_external_native_thread (mono_get_root_domain (), FALSE);
- for (i=0; i < cNames; i++) {
- methodname = mono_unicode_to_external (rgszNames[i]);
- method = mono_class_get_method_from_name_checked(klass, methodname, -1, 0, error);
- if (method && is_ok (error)) {
- cinfo = mono_custom_attrs_from_method_checked (method, error);
- mono_error_assert_ok (error); /* FIXME what's reasonable to do here */
- if (cinfo) {
- MonoObject *result = mono_custom_attrs_get_attr_checked (cinfo, ComDispIdAttribute, error);
- mono_error_assert_ok (error); /*FIXME proper error handling*/;
- if (result)
- rgDispId[i] = *(gint32*)mono_object_unbox_internal (result);
- else
- rgDispId[i] = (gint32)method->token;
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- }
- else
- rgDispId[i] = (gint32)method->token;
- } else {
- mono_error_cleanup (error);
- error_init (error); /* reuse for next iteration */
- rgDispId[i] = MONO_E_DISPID_UNKNOWN;
- ret = MONO_E_DISP_E_UNKNOWNNAME;
- }
- }
- return ret;
- }
- static int STDCALL
- cominterop_ccw_invoke (MonoCCWInterface* ccwe, guint32 dispIdMember,
- gpointer riid, guint32 lcid,
- guint16 wFlags, gpointer pDispParams,
- gpointer pVarResult, gpointer pExcepInfo,
- guint32 *puArgErr)
- {
- return MONO_E_NOTIMPL;
- }
- #ifndef HOST_WIN32
- typedef mono_bstr (STDCALL *SysAllocStringLenFunc)(const gunichar* str, guint32 len);
- typedef guint32 (STDCALL *SysStringLenFunc)(mono_bstr_const bstr);
- typedef void (STDCALL *SysFreeStringFunc)(mono_bstr_const str);
- static SysAllocStringLenFunc sys_alloc_string_len_ms = NULL;
- static SysStringLenFunc sys_string_len_ms = NULL;
- static SysFreeStringFunc sys_free_string_ms = NULL;
- typedef struct tagSAFEARRAYBOUND {
- ULONG cElements;
- LONG lLbound;
- }SAFEARRAYBOUND,*LPSAFEARRAYBOUND;
- #define VT_VARIANT 12
- typedef guint32 (STDCALL *SafeArrayGetDimFunc)(gpointer psa);
- typedef int (STDCALL *SafeArrayGetLBoundFunc)(gpointer psa, guint32 nDim, glong* plLbound);
- typedef int (STDCALL *SafeArrayGetUBoundFunc)(gpointer psa, guint32 nDim, glong* plUbound);
- typedef int (STDCALL *SafeArrayPtrOfIndexFunc)(gpointer psa, glong* rgIndices, gpointer* ppvData);
- typedef int (STDCALL *SafeArrayDestroyFunc)(gpointer psa);
- typedef int (STDCALL *SafeArrayPutElementFunc)(gpointer psa, glong* rgIndices, gpointer* ppvData);
- typedef gpointer (STDCALL *SafeArrayCreateFunc)(int vt, guint32 cDims, SAFEARRAYBOUND* rgsabound);
- static SafeArrayGetDimFunc safe_array_get_dim_ms = NULL;
- static SafeArrayGetLBoundFunc safe_array_get_lbound_ms = NULL;
- static SafeArrayGetUBoundFunc safe_array_get_ubound_ms = NULL;
- static SafeArrayPtrOfIndexFunc safe_array_ptr_of_index_ms = NULL;
- static SafeArrayDestroyFunc safe_array_destroy_ms = NULL;
- static SafeArrayPutElementFunc safe_array_put_element_ms = NULL;
- static SafeArrayCreateFunc safe_array_create_ms = NULL;
- static gboolean
- init_com_provider_ms (void)
- {
- static gboolean initialized = FALSE;
- char *error_msg;
- MonoDl *module = NULL;
- const char* scope = "liboleaut32.so";
- if (initialized) {
- // Barrier here prevents reads of sys_alloc_string_len_ms etc.
- // from being reordered before initialized.
- mono_memory_barrier ();
- return TRUE;
- }
- module = mono_dl_open(scope, MONO_DL_LAZY, &error_msg);
- if (error_msg) {
- g_warning ("Error loading COM support library '%s': %s", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SysAllocStringLen", (gpointer*)&sys_alloc_string_len_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SysAllocStringLen", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SysStringLen", (gpointer*)&sys_string_len_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SysStringLen", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SysFreeString", (gpointer*)&sys_free_string_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SysFreeString", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayGetDim", (gpointer*)&safe_array_get_dim_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayGetDim", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayGetLBound", (gpointer*)&safe_array_get_lbound_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayGetLBound", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayGetUBound", (gpointer*)&safe_array_get_ubound_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayGetUBound", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayPtrOfIndex", (gpointer*)&safe_array_ptr_of_index_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayPtrOfIndex", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayDestroy", (gpointer*)&safe_array_destroy_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayDestroy", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayPutElement", (gpointer*)&safe_array_put_element_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayPutElement", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- error_msg = mono_dl_symbol (module, "SafeArrayCreate", (gpointer*)&safe_array_create_ms);
- if (error_msg) {
- g_warning ("Error loading entry point '%s' in COM support library '%s': %s", "SafeArrayCreate", scope, error_msg);
- g_assert_not_reached ();
- return FALSE;
- }
- mono_memory_barrier ();
- initialized = TRUE;
- return TRUE;
- }
- #endif // WIN32
- #endif // DISABLE_COM
- // This function is used regardless of the BSTR type, so cast the return value
- // Inputted string length, in bytes, should include the null terminator
- // Returns the start of the string itself
- static gpointer
- mono_bstr_alloc (size_t str_byte_len)
- {
- // Allocate string length plus pointer-size integer to store the length, aligned to 16 bytes
- size_t alloc_size = str_byte_len + SIZEOF_VOID_P;
- alloc_size += (16 - 1);
- alloc_size &= ~(16 - 1);
- gpointer ret = g_malloc0 (alloc_size);
- return ret ? (char *)ret + SIZEOF_VOID_P : NULL;
- }
- static void
- mono_bstr_set_length (gunichar2 *bstr, int slen)
- {
- *((guint32 *)bstr - 1) = slen * sizeof (gunichar2);
- }
- static mono_bstr
- default_ptr_to_bstr (const gunichar2* ptr, int slen)
- {
- // In Mono, historically BSTR was allocated with a guaranteed size prefix of 4 bytes regardless of platform.
- // Presumably this is due to the BStr documentation page, which indicates that behavior and then directs you to call
- // SysAllocString on Windows to handle the allocation for you. Unfortunately, this is not actually how it works:
- // The allocation pre-string is pointer-sized, and then only 4 bytes are used for the length regardless. Additionally,
- // the total length is also aligned to a 16-byte boundary. This preserves the old behavior on legacy and fixes it for
- // netcore moving forward.
- #ifdef ENABLE_NETCORE
- mono_bstr const s = (mono_bstr)mono_bstr_alloc ((slen + 1) * sizeof (gunichar2));
- if (s == NULL)
- return NULL;
- #else
- /* allocate len + 1 utf16 characters plus 4 byte integer for length*/
- guint32 * const ret = (guint32 *)g_malloc ((slen + 1) * sizeof (gunichar2) + sizeof (guint32));
- if (ret == NULL)
- return NULL;
- mono_bstr const s = (mono_bstr)(ret + 1);
- #endif
- mono_bstr_set_length (s, slen);
- if (ptr)
- memcpy (s, ptr, slen * sizeof (gunichar2));
- s [slen] = 0;
- return s;
- }
- /* PTR can be NULL */
- mono_bstr
- mono_ptr_to_bstr (const gunichar2* ptr, int slen)
- {
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_BSTR
- return SysAllocStringLen (ptr, slen);
- #else
- return default_ptr_to_bstr (ptr, slen);
- #endif
- #else
- #ifndef DISABLE_COM
- if (com_provider == MONO_COM_DEFAULT) {
- #endif
- return default_ptr_to_bstr (ptr, slen);
- #ifndef DISABLE_COM
- }
- else if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- guint32 const len = slen;
- gunichar* const str = ptr ? g_utf16_to_ucs4 (ptr, len, NULL, NULL, NULL) : NULL;
- mono_bstr const ret = sys_alloc_string_len_ms (str, len);
- g_free (str);
- return ret;
- }
- else {
- g_assert_not_reached();
- }
- #endif
- #endif
- }
- char *
- mono_ptr_to_ansibstr (const char *ptr, size_t slen)
- {
- // FIXME: should this behave differently without DISABLE_COM?
- char *s = (char *)mono_bstr_alloc ((slen + 1) * sizeof(char));
- if (s == NULL)
- return NULL;
- *((guint32 *)s - 1) = slen * sizeof (char);
- if (ptr)
- memcpy (s, ptr, slen * sizeof (char));
- s [slen] = 0;
- return s;
- }
- MonoStringHandle
- mono_string_from_bstr_checked (mono_bstr_const bstr, MonoError *error)
- {
- if (!bstr)
- return NULL_HANDLE_STRING;
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_BSTR
- return mono_string_new_utf16_handle (mono_domain_get (), bstr, SysStringLen ((BSTR)bstr), error);
- #else
- return mono_string_new_utf16_handle (mono_domain_get (), bstr, *((guint32 *)bstr - 1) / sizeof (gunichar2), error);
- #endif /* HAVE_API_SUPPORT_WIN32_BSTR */
- #else
- #ifndef DISABLE_COM
- if (com_provider == MONO_COM_DEFAULT)
- #endif
- return mono_string_new_utf16_handle (mono_domain_get (), bstr, *((guint32 *)bstr - 1) / sizeof (gunichar2), error);
- #ifndef DISABLE_COM
- else if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- glong written = 0;
- // FIXME mono_string_new_utf32_handle to combine g_ucs4_to_utf16 and mono_string_new_utf16_handle.
- gunichar2* utf16 = g_ucs4_to_utf16 ((const gunichar *)bstr, sys_string_len_ms (bstr), NULL, &written, NULL);
- MonoStringHandle res = mono_string_new_utf16_handle (mono_domain_get (), utf16, written, error);
- g_free (utf16);
- return res;
- } else {
- g_assert_not_reached ();
- }
- #endif // DISABLE_COM
- #endif // HOST_WIN32
- }
- MonoString *
- mono_string_from_bstr (/*mono_bstr_const*/gpointer bstr)
- {
- // FIXME gcmode
- HANDLE_FUNCTION_ENTER ();
- ERROR_DECL (error);
- MonoStringHandle result = mono_string_from_bstr_checked ((mono_bstr_const)bstr, error);
- mono_error_cleanup (error);
- HANDLE_FUNCTION_RETURN_OBJ (result);
- }
- MonoStringHandle
- mono_string_from_bstr_icall_impl (mono_bstr_const bstr, MonoError *error)
- {
- return mono_string_from_bstr_checked (bstr, error);
- }
- MONO_API void
- mono_free_bstr (/*mono_bstr_const*/gpointer bstr)
- {
- if (!bstr)
- return;
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_BSTR
- SysFreeString ((BSTR)bstr);
- #else
- g_free (((char *)bstr) - 4);
- #endif /* HAVE_API_SUPPORT_WIN32_BSTR */
- #else
- #ifndef DISABLE_COM
- if (com_provider == MONO_COM_DEFAULT) {
- #endif
- #ifdef ENABLE_NETCORE
- g_free (((char *)bstr) - SIZEOF_VOID_P);
- #else // In Mono, historically BSTR was allocated with a guaranteed size prefix of 4 bytes regardless of platform
- g_free (((char *)bstr) - 4);
- #endif
- #ifndef DISABLE_COM
- } else if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- sys_free_string_ms ((mono_bstr_const)bstr);
- } else {
- g_assert_not_reached ();
- }
- #endif // DISABLE_COM
- #endif // HOST_WIN32
- }
- // FIXME There are multiple caches of "GetObjectForNativeVariant".
- G_GNUC_UNUSED
- static MonoMethod*
- mono_get_Marshal_GetObjectForNativeVariant (void)
- {
- MONO_STATIC_POINTER_INIT (MonoMethod, get_object_for_native_variant)
- ERROR_DECL (error);
- get_object_for_native_variant = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetObjectForNativeVariant", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_object_for_native_variant)
- g_assert (get_object_for_native_variant);
- return get_object_for_native_variant;
- }
- // FIXME There are multiple caches of "GetNativeVariantForObject".
- G_GNUC_UNUSED
- static MonoMethod*
- mono_get_Marshal_GetNativeVariantForObject (void)
- {
- MONO_STATIC_POINTER_INIT (MonoMethod, get_native_variant_for_object)
- ERROR_DECL (error);
- get_native_variant_for_object = mono_class_get_method_from_name_checked (mono_defaults.marshal_class, "GetNativeVariantForObject", 2, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_native_variant_for_object)
- g_assert (get_native_variant_for_object);
- return get_native_variant_for_object;
- }
- G_GNUC_UNUSED
- static MonoMethod*
- mono_get_Array_SetValueImpl (void)
- {
- MONO_STATIC_POINTER_INIT (MonoMethod, set_value_impl)
- ERROR_DECL (error);
- set_value_impl = mono_class_get_method_from_name_checked (mono_defaults.array_class, "SetValueImpl", 2, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, set_value_impl)
- g_assert (set_value_impl);
- return set_value_impl;
- }
- #ifndef DISABLE_COM
- // FIXME There are multiple caches of "Clear".
- G_GNUC_UNUSED
- static MonoMethod*
- mono_get_Variant_Clear (void)
- {
- MONO_STATIC_POINTER_INIT (MonoMethod, variant_clear)
- ERROR_DECL (error);
- variant_clear = mono_class_get_method_from_name_checked (mono_class_get_variant_class (), "Clear", 0, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, variant_clear)
- g_assert (variant_clear);
- return variant_clear;
- }
- /* SAFEARRAY marshalling */
- int
- mono_cominterop_emit_marshal_safearray (EmitMarshalContext *m, int argnum, MonoType *t,
- MonoMarshalSpec *spec,
- int conv_arg, MonoType **conv_arg_type,
- MarshalAction action)
- {
- MonoMethodBuilder *mb = m->mb;
- #ifndef DISABLE_JIT
- switch (action) {
- case MARSHAL_ACTION_CONV_IN: {
- if ((t->attrs & (PARAM_ATTRIBUTE_IN | PARAM_ATTRIBUTE_OUT)) == PARAM_ATTRIBUTE_OUT)
- break;
- /* Generates IL code for the following algorithm:
- SafeArray safearray; // safearray_var
- IntPtr indices; // indices_var
- int empty; // empty_var
- if (mono_marshal_safearray_create (array, out safearray, out indices, out empty)) {
- if (!empty) {
- int index=0; // index_var
- do { // label3
- variant elem = Marshal.GetNativeVariantForObject (array.GetValueImpl(index));
- mono_marshal_safearray_set_value (safearray, indices, elem);
- ++index;
- }
- while (mono_marshal_safearray_next (safearray, indices));
- } // label2
- mono_marshal_safearray_free_indices (indices);
- } // label1
- */
- int safearray_var, indices_var, empty_var, elem_var, index_var;
- guint32 label1 = 0, label2 = 0, label3 = 0;
- MonoType *int_type = mono_get_int_type ();
- conv_arg = safearray_var = mono_mb_add_local (mb, mono_get_object_type ());
- indices_var = mono_mb_add_local (mb, int_type);
- empty_var = mono_mb_add_local (mb, int_type);
- if (t->byref) {
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- } else
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc_addr (mb, safearray_var);
- mono_mb_emit_ldloc_addr (mb, indices_var);
- mono_mb_emit_ldloc_addr (mb, empty_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_create);
- label1 = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldloc (mb, empty_var);
- label2 = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- index_var = mono_mb_add_local (mb, mono_get_int32_type ());
- mono_mb_emit_byte (mb, CEE_LDC_I4_0);
- mono_mb_emit_stloc (mb, index_var);
- label3 = mono_mb_get_label (mb);
- MONO_STATIC_POINTER_INIT (MonoMethod, get_value_impl)
- ERROR_DECL (error);
- get_value_impl = mono_class_get_method_from_name_checked (mono_defaults.array_class, "GetValueImpl", 1, 0, error);
- mono_error_assert_ok (error);
- MONO_STATIC_POINTER_INIT_END (MonoMethod, get_value_impl)
- g_assert (get_value_impl);
- if (t->byref) {
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_byte (mb, CEE_LDIND_REF);
- } else
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, index_var);
- mono_mb_emit_managed_call (mb, get_value_impl, NULL);
- elem_var = mono_mb_add_local (mb, m_class_get_byval_arg (mono_class_get_variant_class ()));
- mono_mb_emit_ldloc_addr (mb, elem_var);
- mono_mb_emit_managed_call (mb, mono_get_Marshal_GetNativeVariantForObject (), NULL);
- mono_mb_emit_ldloc (mb, safearray_var);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_ldloc_addr (mb, elem_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_set_value);
- mono_mb_emit_ldloc_addr (mb, elem_var);
- mono_mb_emit_managed_call (mb, mono_get_Variant_Clear (), NULL);
- mono_mb_emit_add_to_local (mb, index_var, 1);
- mono_mb_emit_ldloc (mb, safearray_var);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_next);
- mono_mb_emit_branch_label (mb, CEE_BRTRUE, label3);
- mono_mb_patch_short_branch (mb, label2);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_free_indices);
- mono_mb_patch_short_branch (mb, label1);
- break;
- }
- case MARSHAL_ACTION_PUSH:
- if (t->byref)
- mono_mb_emit_ldloc_addr (mb, conv_arg);
- else
- mono_mb_emit_ldloc (mb, conv_arg);
- break;
- case MARSHAL_ACTION_CONV_OUT: {
- if (t->attrs & PARAM_ATTRIBUTE_OUT) {
- /* Generates IL code for the following algorithm:
- Array result; // result_var
- IntPtr indices; // indices_var
- int empty; // empty_var
- bool byValue = !t->byref && (t->attrs & PARAM_ATTRIBUTE_IN);
- if (mono_marshal_safearray_begin(safearray, out result, out indices, out empty, parameter, byValue)) {
- if (!empty) {
- int index=0; // index_var
- do { // label3
- if (!byValue || (index < parameter.Length)) {
- object elem = Variant.GetObjectForNativeVariant(mono_marshal_safearray_get_value(safearray, indices));
- result.SetValueImpl(elem, index);
- }
- ++index;
- }
- while (mono_marshal_safearray_next(safearray, indices));
- } // label2
- mono_marshal_safearray_end(safearray, indices);
- } // label1
- if (!byValue)
- return result;
- */
- int result_var, indices_var, empty_var, elem_var, index_var;
- guint32 label1 = 0, label2 = 0, label3 = 0, label4 = 0;
- gboolean byValue = !t->byref && (t->attrs & PARAM_ATTRIBUTE_IN);
- MonoType *object_type = mono_get_object_type ();
- MonoType *int_type = mono_get_int_type ();
- result_var = mono_mb_add_local (mb, object_type);
- indices_var = mono_mb_add_local (mb, int_type);
- empty_var = mono_mb_add_local (mb, int_type);
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_ldloc_addr (mb, result_var);
- mono_mb_emit_ldloc_addr (mb, indices_var);
- mono_mb_emit_ldloc_addr (mb, empty_var);
- mono_mb_emit_ldarg (mb, argnum);
- if (byValue)
- mono_mb_emit_byte (mb, CEE_LDC_I4_0);
- else
- mono_mb_emit_byte (mb, CEE_LDC_I4_1);
- mono_mb_emit_icall (mb, mono_marshal_safearray_begin);
- label1 = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldloc (mb, empty_var);
- label2 = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- index_var = mono_mb_add_local (mb, int_type);
- mono_mb_emit_byte (mb, CEE_LDC_I4_0);
- mono_mb_emit_stloc (mb, index_var);
- label3 = mono_mb_get_label (mb);
- if (byValue) {
- mono_mb_emit_ldloc (mb, index_var);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_byte (mb, CEE_LDLEN);
- label4 = mono_mb_emit_branch (mb, CEE_BGE);
- }
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_get_value);
- elem_var = mono_mb_add_local (mb, object_type);
- mono_mb_emit_managed_call (mb, mono_get_Marshal_GetObjectForNativeVariant (), NULL);
- mono_mb_emit_stloc (mb, elem_var);
- mono_mb_emit_ldloc (mb, result_var);
- mono_mb_emit_ldloc (mb, elem_var);
- mono_mb_emit_ldloc (mb, index_var);
- mono_mb_emit_managed_call (mb, mono_get_Array_SetValueImpl (), NULL);
- if (byValue)
- mono_mb_patch_short_branch (mb, label4);
- mono_mb_emit_add_to_local (mb, index_var, 1);
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_next);
- mono_mb_emit_branch_label (mb, CEE_BRTRUE, label3);
- mono_mb_patch_short_branch (mb, label2);
- mono_mb_emit_ldloc (mb, conv_arg);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_end);
- mono_mb_patch_short_branch (mb, label1);
- if (!byValue) {
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, result_var);
- mono_mb_emit_byte (mb, CEE_STIND_REF);
- }
- }
- break;
- }
- case MARSHAL_ACTION_MANAGED_CONV_IN: {
- if ((t->attrs & (PARAM_ATTRIBUTE_IN | PARAM_ATTRIBUTE_OUT)) == PARAM_ATTRIBUTE_OUT)
- break;
- /* Generates IL code for the following algorithm:
- Array result; // result_var
- IntPtr indices; // indices_var
- int empty; // empty_var
- if (mono_marshal_safearray_begin(safearray, out result, out indices, out empty, NULL, TRUE)) {
- if (!empty) {
- int index=0; // index_var
- do { // label3
- object elem = Variant.GetObjectForNativeVariant(mono_marshal_safearray_get_value(safearray, indices));
- result.SetValueImpl(elem, index);
- ++index;
- }
- while (mono_marshal_safearray_next(safearray, indices));
- } // label2
- mono_marshal_safearray_free_indices(indices);
- } // label1
- */
- int result_var, indices_var, empty_var, elem_var, index_var;
- guint32 label1 = 0, label2 = 0, label3 = 0;
- MonoType *object_type = mono_get_object_type ();
- MonoType *int_type = mono_get_int_type ();
- result_var = mono_mb_add_local (mb, object_type);
- indices_var = mono_mb_add_local (mb, int_type);
- empty_var = mono_mb_add_local (mb, int_type);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc_addr (mb, result_var);
- mono_mb_emit_ldloc_addr (mb, indices_var);
- mono_mb_emit_ldloc_addr (mb, empty_var);
- mono_mb_emit_byte (mb, CEE_LDC_I4_0);
- mono_mb_emit_byte (mb, CEE_CONV_I);
- mono_mb_emit_byte (mb, CEE_LDC_I4_1);
- mono_mb_emit_icall (mb, mono_marshal_safearray_begin);
- label1 = mono_mb_emit_short_branch (mb, CEE_BRFALSE_S);
- mono_mb_emit_ldloc (mb, empty_var);
- label2 = mono_mb_emit_short_branch (mb, CEE_BRTRUE_S);
- index_var = mono_mb_add_local (mb, int_type);
- mono_mb_emit_byte (mb, CEE_LDC_I4_0);
- mono_mb_emit_stloc (mb, index_var);
- label3 = mono_mb_get_label (mb);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_get_value);
- elem_var = mono_mb_add_local (mb, object_type);
- mono_mb_emit_managed_call (mb, mono_get_Marshal_GetObjectForNativeVariant (), NULL);
- mono_mb_emit_stloc (mb, elem_var);
- mono_mb_emit_ldloc (mb, result_var);
- mono_mb_emit_ldloc (mb, elem_var);
- mono_mb_emit_ldloc (mb, index_var);
- mono_mb_emit_managed_call (mb, mono_get_Array_SetValueImpl (), NULL);
- mono_mb_emit_add_to_local (mb, index_var, 1);
- mono_mb_emit_ldarg (mb, argnum);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_next);
- mono_mb_emit_branch_label (mb, CEE_BRTRUE, label3);
- mono_mb_patch_short_branch (mb, label2);
- mono_mb_emit_ldloc (mb, indices_var);
- mono_mb_emit_icall (mb, mono_marshal_safearray_free_indices);
- mono_mb_patch_short_branch (mb, label1);
- mono_mb_emit_ldloc (mb, result_var);
- mono_mb_emit_stloc (mb, conv_arg);
- break;
- }
- default:
- g_assert_not_reached ();
- }
- #endif /* DISABLE_JIT */
- return conv_arg;
- }
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static guint32
- mono_marshal_win_safearray_get_dim (gpointer safearray)
- {
- return SafeArrayGetDim ((SAFEARRAY*)safearray);
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static guint32
- mono_marshal_win_safearray_get_dim (gpointer safearray)
- {
- g_unsupported_api ("SafeArrayGetDim");
- SetLastError (ERROR_NOT_SUPPORTED);
- return MONO_E_NOTIMPL;
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- static guint32
- mono_marshal_safearray_get_dim (gpointer safearray)
- {
- return mono_marshal_win_safearray_get_dim (safearray);
- }
- #else /* HOST_WIN32 */
- static guint32
- mono_marshal_safearray_get_dim (gpointer safearray)
- {
- guint32 result=0;
- if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- result = safe_array_get_dim_ms (safearray);
- } else {
- g_assert_not_reached ();
- }
- return result;
- }
- #endif /* HOST_WIN32 */
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safe_array_get_lbound (gpointer psa, guint nDim, glong* plLbound)
- {
- return SafeArrayGetLBound ((SAFEARRAY*)psa, nDim, plLbound);
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safe_array_get_lbound (gpointer psa, guint nDim, glong* plLbound)
- {
- g_unsupported_api ("SafeArrayGetLBound");
- SetLastError (ERROR_NOT_SUPPORTED);
- return MONO_E_NOTIMPL;
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- static int
- mono_marshal_safe_array_get_lbound (gpointer psa, guint nDim, glong* plLbound)
- {
- return mono_marshal_win_safe_array_get_lbound (psa, nDim, plLbound);
- }
- #else /* HOST_WIN32 */
- static int
- mono_marshal_safe_array_get_lbound (gpointer psa, guint nDim, glong* plLbound)
- {
- int result=MONO_S_OK;
- if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- result = safe_array_get_lbound_ms (psa, nDim, plLbound);
- } else {
- g_assert_not_reached ();
- }
- return result;
- }
- #endif /* HOST_WIN32 */
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safe_array_get_ubound (gpointer psa, guint nDim, glong* plUbound)
- {
- return SafeArrayGetUBound ((SAFEARRAY*)psa, nDim, plUbound);
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safe_array_get_ubound (gpointer psa, guint nDim, glong* plUbound)
- {
- g_unsupported_api ("SafeArrayGetUBound");
- SetLastError (ERROR_NOT_SUPPORTED);
- return MONO_E_NOTIMPL;
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- static int
- mono_marshal_safe_array_get_ubound (gpointer psa, guint nDim, glong* plUbound)
- {
- return mono_marshal_win_safe_array_get_ubound (psa, nDim, plUbound);
- }
- #else /* HOST_WIN32 */
- static int
- mono_marshal_safe_array_get_ubound (gpointer psa, guint nDim, glong* plUbound)
- {
- int result=MONO_S_OK;
- if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- result = safe_array_get_ubound_ms (psa, nDim, plUbound);
- } else {
- g_assert_not_reached ();
- }
- return result;
- }
- #endif /* HOST_WIN32 */
- /* This is an icall */
- static gboolean
- mono_marshal_safearray_begin (gpointer safearray, MonoArray **result, gpointer *indices, gpointer empty, gpointer parameter, gboolean allocateNewArray)
- {
- ERROR_DECL (error);
- int dim;
- uintptr_t *sizes;
- intptr_t *bounds;
- MonoClass *aklass;
- int i;
- gboolean bounded = FALSE;
- #ifndef HOST_WIN32
- // If not on windows, check that the MS provider is used as it is
- // required for SAFEARRAY support.
- // If SAFEARRAYs are not supported, returning FALSE from this
- // function will prevent the other mono_marshal_safearray_xxx functions
- // from being called.
- if ((com_provider != MONO_COM_MS) || !init_com_provider_ms ()) {
- return FALSE;
- }
- #endif
- (*(int*)empty) = TRUE;
- if (safearray != NULL) {
- dim = mono_marshal_safearray_get_dim (safearray);
- if (dim > 0) {
- *indices = g_malloc (dim * sizeof(int));
- sizes = g_newa (uintptr_t, dim);
- bounds = g_newa (intptr_t, dim);
- for (i=0; i<dim; ++i) {
- glong lbound, ubound;
- int cursize;
- int hr;
- hr = mono_marshal_safe_array_get_lbound (safearray, i+1, &lbound);
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- if (mono_error_set_pending_exception (error))
- return FALSE;
- }
- if (lbound != 0)
- bounded = TRUE;
- hr = mono_marshal_safe_array_get_ubound (safearray, i+1, &ubound);
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- if (mono_error_set_pending_exception (error))
- return FALSE;
- }
- cursize = ubound-lbound+1;
- sizes [i] = cursize;
- bounds [i] = lbound;
- ((int*)*indices) [i] = lbound;
- if (cursize != 0)
- (*(int*)empty) = FALSE;
- }
- if (allocateNewArray) {
- aklass = mono_class_create_bounded_array (mono_defaults.object_class, dim, bounded);
- *result = mono_array_new_full_checked (mono_domain_get (), aklass, sizes, bounds, error);
- if (mono_error_set_pending_exception (error))
- return FALSE;
- } else {
- *result = (MonoArray *)parameter;
- }
- }
- }
- return TRUE;
- }
- /* This is an icall */
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safearray_get_value (gpointer safearray, gpointer indices, gpointer *result)
- {
- return SafeArrayPtrOfIndex ((SAFEARRAY*)safearray, (LONG*)indices, result);
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safearray_get_value (gpointer safearray, gpointer indices, gpointer *result)
- {
- ERROR_DECL (error);
- g_unsupported_api ("SafeArrayPtrOfIndex");
- mono_error_set_not_supported (error, G_UNSUPPORTED_API, "SafeArrayPtrOfIndex");
- mono_error_set_pending_exception (error);
- SetLastError (ERROR_NOT_SUPPORTED);
- return MONO_E_NOTIMPL;
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- static gpointer
- mono_marshal_safearray_get_value (gpointer safearray, gpointer indices)
- {
- ERROR_DECL (error);
- gpointer result;
- int hr = mono_marshal_win_safearray_get_value (safearray, indices, &result);
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- mono_error_set_pending_exception (error);
- result = NULL;
- }
- return result;
- }
- #else /* HOST_WIN32 */
- static gpointer
- mono_marshal_safearray_get_value (gpointer safearray, gpointer indices)
- {
- ERROR_DECL (error);
- gpointer result;
- if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- int hr = safe_array_ptr_of_index_ms (safearray, (glong *)indices, &result);
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- mono_error_set_pending_exception (error);
- return NULL;
- }
- } else {
- g_assert_not_reached ();
- }
- return result;
- }
- #endif /* HOST_WIN32 */
- /* This is an icall */
- static
- gboolean mono_marshal_safearray_next (gpointer safearray, gpointer indices)
- {
- ERROR_DECL (error);
- int i;
- int dim = mono_marshal_safearray_get_dim (safearray);
- gboolean ret= TRUE;
- int *pIndices = (int*) indices;
- int hr;
- for (i=dim-1; i>=0; --i)
- {
- glong lbound, ubound;
- hr = mono_marshal_safe_array_get_ubound (safearray, i+1, &ubound);
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- mono_error_set_pending_exception (error);
- return FALSE;
- }
- if (++pIndices[i] <= ubound) {
- break;
- }
- hr = mono_marshal_safe_array_get_lbound (safearray, i+1, &lbound);
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- mono_error_set_pending_exception (error);
- return FALSE;
- }
- pIndices[i] = lbound;
- if (i == 0)
- ret = FALSE;
- }
- return ret;
- }
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static void
- mono_marshal_win_safearray_end (gpointer safearray, gpointer indices)
- {
- g_free(indices);
- SafeArrayDestroy ((SAFEARRAY*)safearray);
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static void
- mono_marshal_win_safearray_end (gpointer safearray, gpointer indices)
- {
- g_free(indices);
- g_unsupported_api ("SafeArrayDestroy");
- SetLastError (ERROR_NOT_SUPPORTED);
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- static void
- mono_marshal_safearray_end (gpointer safearray, gpointer indices)
- {
- mono_marshal_win_safearray_end (safearray, indices);
- }
- #else /* HOST_WIN32 */
- static void
- mono_marshal_safearray_end (gpointer safearray, gpointer indices)
- {
- g_free(indices);
- if (com_provider == MONO_COM_MS && init_com_provider_ms ()) {
- safe_array_destroy_ms (safearray);
- } else {
- g_assert_not_reached ();
- }
- }
- #endif /* HOST_WIN32 */
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static gboolean
- mono_marshal_win_safearray_create_internal (UINT cDims, SAFEARRAYBOUND *rgsabound, gpointer *newsafearray)
- {
- *newsafearray = SafeArrayCreate (VT_VARIANT, cDims, rgsabound);
- return TRUE;
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static gboolean
- mono_marshal_win_safearray_create_internal (UINT cDims, SAFEARRAYBOUND *rgsabound, gpointer *newsafearray)
- {
- g_unsupported_api ("SafeArrayCreate");
- SetLastError (ERROR_NOT_SUPPORTED);
- *newsafearray = NULL;
- return FALSE;
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- static gboolean
- mono_marshal_safearray_create_internal (UINT cDims, SAFEARRAYBOUND *rgsabound, gpointer *newsafearray)
- {
- return mono_marshal_win_safearray_create_internal (cDims, rgsabound, newsafearray);
- }
- #else /* HOST_WIN32 */
- static gboolean
- mono_marshal_safearray_create_internal (UINT cDims, SAFEARRAYBOUND *rgsabound, gpointer *newsafearray)
- {
- *newsafearray = safe_array_create_ms (VT_VARIANT, cDims, rgsabound);
- return TRUE;
- }
- #endif /* HOST_WIN32 */
- static gboolean
- mono_marshal_safearray_create (MonoArray *input, gpointer *newsafearray, gpointer *indices, gpointer empty)
- {
- #ifndef HOST_WIN32
- // If not on windows, check that the MS provider is used as it is
- // required for SAFEARRAY support.
- // If SAFEARRAYs are not supported, returning FALSE from this
- // function will prevent the other mono_marshal_safearray_xxx functions
- // from being called.
- if (com_provider != MONO_COM_MS || !init_com_provider_ms ()) {
- return FALSE;
- }
- #endif
- int const max_array_length = mono_array_length_internal (input);
- int const dim = m_class_get_rank (mono_object_class (input));
- *indices = g_malloc (dim * sizeof (int));
- SAFEARRAYBOUND * const bounds = g_newa (SAFEARRAYBOUND, dim);
- (*(int*)empty) = (max_array_length == 0);
- if (dim > 1) {
- for (int i = 0; i < dim; ++i) {
- ((int*)*indices) [i] = bounds [i].lLbound = input->bounds [i].lower_bound;
- bounds [i].cElements = input->bounds [i].length;
- }
- } else {
- ((int*)*indices) [0] = 0;
- bounds [0].cElements = max_array_length;
- bounds [0].lLbound = 0;
- }
- return mono_marshal_safearray_create_internal (dim, bounds, newsafearray);
- }
- /* This is an icall */
- #ifdef HOST_WIN32
- #if HAVE_API_SUPPORT_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safearray_set_value (gpointer safearray, gpointer indices, gpointer value)
- {
- return SafeArrayPutElement ((SAFEARRAY*)safearray, (LONG*)indices, value);
- }
- #elif !HAVE_EXTERN_DEFINED_WIN32_SAFE_ARRAY
- static int
- mono_marshal_win_safearray_set_value (gpointer safearray, gpointer indices, gpointer value)
- {
- ERROR_DECL (error);
- g_unsupported_api ("SafeArrayPutElement");
- mono_error_set_not_supported (error, G_UNSUPPORTED_API, "SafeArrayPutElement");
- mono_error_set_pending_exception (error);
- SetLastError (ERROR_NOT_SUPPORTED);
- return MONO_E_NOTIMPL;
- }
- #endif /* HAVE_API_SUPPORT_WIN32_SAFE_ARRAY */
- #endif /* HOST_WIN32 */
- static void
- mono_marshal_safearray_set_value (gpointer safearray, gpointer indices, gpointer value)
- {
- ERROR_DECL (error);
- #ifdef HOST_WIN32
- int const hr = mono_marshal_win_safearray_set_value (safearray, indices, value);
- #else
- int hr = 0;
- if (com_provider == MONO_COM_MS && init_com_provider_ms ())
- hr = safe_array_put_element_ms (safearray, (glong *)indices, (void **)value);
- else
- g_assert_not_reached ();
- #endif
- if (hr < 0) {
- cominterop_set_hr_error (error, hr);
- mono_error_set_pending_exception (error);
- }
- }
- static
- void mono_marshal_safearray_free_indices (gpointer indices)
- {
- g_free (indices);
- }
- #else /* DISABLE_COM */
- void
- mono_cominterop_cleanup (void)
- {
- }
- void
- mono_cominterop_release_all_rcws (void)
- {
- }
- gboolean
- mono_marshal_free_ccw (MonoObject* object)
- {
- return FALSE;
- }
- #ifdef HOST_WIN32
- int
- ves_icall_System_Runtime_InteropServices_Marshal_AddRefInternal (MonoIUnknown *pUnk)
- {
- return mono_IUnknown_AddRef (pUnk);
- }
- int
- ves_icall_System_Runtime_InteropServices_Marshal_ReleaseInternal (MonoIUnknown *pUnk)
- {
- g_assert (pUnk);
- return mono_IUnknown_Release (pUnk);
- }
- int
- ves_icall_System_Runtime_InteropServices_Marshal_QueryInterfaceInternal (MonoIUnknown *pUnk, gconstpointer riid, gpointer* ppv)
- {
- return mono_IUnknown_QueryInterface (pUnk, riid, ppv);
- }
- #else /* HOST_WIN32 */
- int
- ves_icall_System_Runtime_InteropServices_Marshal_AddRefInternal (MonoIUnknown *pUnk)
- {
- g_assert_not_reached ();
- return 0;
- }
- int
- ves_icall_System_Runtime_InteropServices_Marshal_ReleaseInternal (MonoIUnknown *pUnk)
- {
- g_assert_not_reached ();
- return 0;
- }
- int
- ves_icall_System_Runtime_InteropServices_Marshal_QueryInterfaceInternal (MonoIUnknown *pUnk, gconstpointer riid, gpointer* ppv)
- {
- g_assert_not_reached ();
- return 0;
- }
- #endif /* HOST_WIN32 */
- #endif /* DISABLE_COM */
- #ifndef ENABLE_NETCORE
- MonoStringHandle
- ves_icall_System_Runtime_InteropServices_Marshal_PtrToStringBSTR (mono_bstr_const ptr, MonoError *error)
- {
- if (ptr == NULL) {
- mono_error_set_argument_null (error, "ptr", NULL);
- return NULL_HANDLE_STRING;
- }
- return mono_string_from_bstr_checked (ptr, error);
- }
- #endif
- mono_bstr
- ves_icall_System_Runtime_InteropServices_Marshal_BufferToBSTR (const gunichar2* ptr, int len)
- {
- return mono_ptr_to_bstr (ptr, len);
- }
- void
- ves_icall_System_Runtime_InteropServices_Marshal_FreeBSTR (mono_bstr_const ptr)
- {
- mono_free_bstr ((gpointer)ptr);
- }
- void*
- mono_cominterop_get_com_interface (MonoObject *object_raw, MonoClass *ic, MonoError *error)
- {
- HANDLE_FUNCTION_ENTER ();
- MONO_HANDLE_DCL (MonoObject, object);
- void* const result = mono_cominterop_get_com_interface_internal (FALSE, object, ic, error);
- HANDLE_FUNCTION_RETURN_VAL (result);
- }
- static void*
- mono_cominterop_get_com_interface_internal (gboolean icall, MonoObjectHandle object, MonoClass *ic, MonoError *error)
- {
- // Common code for mono_cominterop_get_com_interface and
- // ves_icall_System_Runtime_InteropServices_Marshal_GetIUnknownForObjectInternal,
- // which are almost identical.
- #ifndef DISABLE_COM
- if (MONO_HANDLE_IS_NULL (object))
- return NULL;
- MonoRealProxyHandle real_proxy;
- if (cominterop_object_is_rcw_handle (object, &real_proxy)) {
- MonoClass *klass = NULL;
- klass = mono_handle_class (object);
- if (!mono_class_is_transparent_proxy (klass)) {
- g_assertf (!icall, "Class is not transparent");
- mono_error_set_invalid_operation (error, "Class is not transparent");
- return NULL;
- }
- if (MONO_HANDLE_IS_NULL (real_proxy)) {
- g_assertf (!icall, "RealProxy is null");
- mono_error_set_invalid_operation (error, "RealProxy is null");
- return NULL;
- }
- klass = mono_handle_class (real_proxy);
- if (klass != mono_class_get_interop_proxy_class ()) {
- g_assertf (!icall, "Object is not a proxy");
- mono_error_set_invalid_operation (error, "Object is not a proxy");
- return NULL;
- }
- MonoComInteropProxyHandle com_interop_proxy = MONO_HANDLE_CAST (MonoComInteropProxy, real_proxy);
- MonoComObjectHandle com_object = MONO_HANDLE_NEW_GET (MonoComObject, com_interop_proxy, com_object);
- if (MONO_HANDLE_IS_NULL (com_object)) {
- g_assertf (!icall, "Proxy points to null COM object");
- mono_error_set_invalid_operation (error, "Proxy points to null COM object");
- return NULL;
- }
- if (icall)
- return MONO_HANDLE_GETVAL (com_object, iunknown);
- return cominterop_get_interface_checked (com_object, ic, error);
- }
- else {
- if (icall)
- ic = mono_class_get_iunknown_class ();
- return cominterop_get_ccw_checked (object, ic, error);
- }
- #else
- g_assert_not_reached ();
- #endif
- }
- gboolean
- mono_cominterop_is_interface (MonoClass* klass)
- {
- #ifndef DISABLE_COM
- ERROR_DECL (error);
- MonoCustomAttrInfo* cinfo = NULL;
- gboolean ret = FALSE;
- int i;
- cinfo = mono_custom_attrs_from_class_checked (klass, error);
- mono_error_assert_ok (error);
- if (cinfo) {
- for (i = 0; i < cinfo->num_attrs; ++i) {
- MonoClass *ctor_class = cinfo->attrs [i].ctor->klass;
- if (mono_class_has_parent (ctor_class, mono_class_get_interface_type_attribute_class ())) {
- ret = TRUE;
- break;
- }
- }
- if (!cinfo->cached)
- mono_custom_attrs_free (cinfo);
- }
- return ret;
- #else
- g_assert_not_reached ();
- #endif
- }
|