Scene.cs 211 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198
  1. /*
  2. * Copyright (c) Contributors, http://opensimulator.org/
  3. * See CONTRIBUTORS.TXT for a full list of copyright holders.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are met:
  7. * * Redistributions of source code must retain the above copyright
  8. * notice, this list of conditions and the following disclaimer.
  9. * * Redistributions in binary form must reproduce the above copyrightD
  10. * notice, this list of conditions and the following disclaimer in the
  11. * documentation and/or other materials provided with the distribution.
  12. * * Neither the name of the OpenSimulator Project nor the
  13. * names of its contributors may be used to endorse or promote products
  14. * derived from this software without specific prior written permission.
  15. *
  16. * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
  17. * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  18. * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  19. * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
  20. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  21. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  22. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  23. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  24. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  25. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. */
  27. using System;
  28. using System.Collections.Generic;
  29. using System.Diagnostics;
  30. using System.Drawing;
  31. using System.Drawing.Imaging;
  32. using System.IO;
  33. using System.Text;
  34. using System.Threading;
  35. using System.Timers;
  36. using System.Xml;
  37. using Nini.Config;
  38. using OpenMetaverse;
  39. using OpenMetaverse.Packets;
  40. using OpenMetaverse.Imaging;
  41. using OpenSim.Framework;
  42. using OpenSim.Services.Interfaces;
  43. using OpenSim.Framework.Communications;
  44. using OpenSim.Framework.Console;
  45. using OpenSim.Region.Framework.Interfaces;
  46. using OpenSim.Region.Framework.Scenes.Scripting;
  47. using OpenSim.Region.Framework.Scenes.Serialization;
  48. using OpenSim.Region.Physics.Manager;
  49. using Timer=System.Timers.Timer;
  50. using TPFlags = OpenSim.Framework.Constants.TeleportFlags;
  51. using GridRegion = OpenSim.Services.Interfaces.GridRegion;
  52. namespace OpenSim.Region.Framework.Scenes
  53. {
  54. public delegate bool FilterAvatarList(ScenePresence avatar);
  55. public partial class Scene : SceneBase
  56. {
  57. private const long DEFAULT_MIN_TIME_FOR_PERSISTENCE = 60L;
  58. private const long DEFAULT_MAX_TIME_FOR_PERSISTENCE = 600L;
  59. public delegate void SynchronizeSceneHandler(Scene scene);
  60. #region Fields
  61. public bool EmergencyMonitoring = false;
  62. public SynchronizeSceneHandler SynchronizeScene;
  63. public SimStatsReporter StatsReporter;
  64. public List<Border> NorthBorders = new List<Border>();
  65. public List<Border> EastBorders = new List<Border>();
  66. public List<Border> SouthBorders = new List<Border>();
  67. public List<Border> WestBorders = new List<Border>();
  68. /// <summary>Are we applying physics to any of the prims in this scene?</summary>
  69. public bool m_physicalPrim;
  70. public float m_maxNonphys = 256;
  71. public float m_maxPhys = 10;
  72. public bool m_clampPrimSize;
  73. public bool m_trustBinaries;
  74. public bool m_allowScriptCrossings;
  75. public bool m_useFlySlow;
  76. public bool m_usePreJump;
  77. public bool m_seeIntoRegionFromNeighbor;
  78. protected float m_defaultDrawDistance = 255.0f;
  79. public float DefaultDrawDistance
  80. {
  81. get { return m_defaultDrawDistance; }
  82. }
  83. // TODO: need to figure out how allow client agents but deny
  84. // root agents when ACL denies access to root agent
  85. public bool m_strictAccessControl = true;
  86. public int MaxUndoCount = 5;
  87. // Using this for RegionReady module to prevent LoginsDisabled from changing under our feet;
  88. public bool LoginLock = false;
  89. public bool LoginsDisabled = true;
  90. public bool StartDisabled = false;
  91. public bool LoadingPrims;
  92. public IXfer XferManager;
  93. // the minimum time that must elapse before a changed object will be considered for persisted
  94. public long m_dontPersistBefore = DEFAULT_MIN_TIME_FOR_PERSISTENCE * 10000000L;
  95. // the maximum time that must elapse before a changed object will be considered for persisted
  96. public long m_persistAfter = DEFAULT_MAX_TIME_FOR_PERSISTENCE * 10000000L;
  97. protected int m_splitRegionID;
  98. protected Timer m_restartWaitTimer = new Timer();
  99. protected List<RegionInfo> m_regionRestartNotifyList = new List<RegionInfo>();
  100. protected List<RegionInfo> m_neighbours = new List<RegionInfo>();
  101. protected string m_simulatorVersion = "OpenSimulator Server";
  102. protected ModuleLoader m_moduleLoader;
  103. protected AgentCircuitManager m_authenticateHandler;
  104. protected SceneCommunicationService m_sceneGridService;
  105. protected ISimulationDataService m_SimulationDataService;
  106. protected IEstateDataService m_EstateDataService;
  107. protected IAssetService m_AssetService;
  108. protected IAuthorizationService m_AuthorizationService;
  109. protected IInventoryService m_InventoryService;
  110. protected IGridService m_GridService;
  111. protected ILibraryService m_LibraryService;
  112. protected ISimulationService m_simulationService;
  113. protected IAuthenticationService m_AuthenticationService;
  114. protected IPresenceService m_PresenceService;
  115. protected IUserAccountService m_UserAccountService;
  116. protected IAvatarService m_AvatarService;
  117. protected IGridUserService m_GridUserService;
  118. protected IXMLRPC m_xmlrpcModule;
  119. protected IWorldComm m_worldCommModule;
  120. protected IAvatarFactory m_AvatarFactory;
  121. protected IConfigSource m_config;
  122. protected IRegionSerialiserModule m_serialiser;
  123. protected IDialogModule m_dialogModule;
  124. protected IEntityTransferModule m_teleportModule;
  125. protected ICapabilitiesModule m_capsModule;
  126. // Central Update Loop
  127. protected int m_fps = 10;
  128. /// <summary>
  129. /// Current scene frame number
  130. /// </summary>
  131. public uint Frame
  132. {
  133. get;
  134. protected set;
  135. }
  136. protected float m_timespan = 0.089f;
  137. protected DateTime m_lastupdate = DateTime.UtcNow;
  138. // TODO: Possibly stop other classes being able to manipulate this directly.
  139. private SceneGraph m_sceneGraph;
  140. private volatile int m_bordersLocked;
  141. // private int m_RestartTimerCounter;
  142. private readonly Timer m_restartTimer = new Timer(15000); // Wait before firing
  143. // private int m_incrementsof15seconds;
  144. private volatile bool m_backingup;
  145. private Dictionary<UUID, ReturnInfo> m_returns = new Dictionary<UUID, ReturnInfo>();
  146. private Dictionary<UUID, SceneObjectGroup> m_groupsWithTargets = new Dictionary<UUID, SceneObjectGroup>();
  147. private Object m_heartbeatLock = new Object();
  148. private int m_update_physics = 1;
  149. private int m_update_entitymovement = 1;
  150. private int m_update_objects = 1; // Update objects which have scheduled themselves for updates
  151. private int m_update_presences = 1; // Update scene presence movements
  152. private int m_update_events = 1;
  153. private int m_update_backup = 200;
  154. private int m_update_terrain = 50;
  155. // private int m_update_land = 1;
  156. private int m_update_coarse_locations = 50;
  157. private int frameMS;
  158. private int physicsMS2;
  159. private int physicsMS;
  160. private int otherMS;
  161. private int tempOnRezMS;
  162. private int eventMS;
  163. private int backupMS;
  164. private int terrainMS;
  165. private int landMS;
  166. private int lastCompletedFrame;
  167. private bool m_physics_enabled = true;
  168. private bool m_scripts_enabled = true;
  169. private string m_defaultScriptEngine;
  170. private int m_LastLogin;
  171. private Thread HeartbeatThread;
  172. private volatile bool shuttingdown;
  173. private int m_lastUpdate;
  174. private bool m_firstHeartbeat = true;
  175. private object m_deleting_scene_object = new object();
  176. private object m_cleaningAttachments = new object();
  177. private bool m_cleaningTemps = false;
  178. private UpdatePrioritizationSchemes m_priorityScheme = UpdatePrioritizationSchemes.Time;
  179. private bool m_reprioritizationEnabled = true;
  180. private double m_reprioritizationInterval = 5000.0;
  181. private double m_rootReprioritizationDistance = 10.0;
  182. private double m_childReprioritizationDistance = 20.0;
  183. private Timer m_mapGenerationTimer = new Timer();
  184. private bool m_generateMaptiles;
  185. private bool m_useBackup = true;
  186. // private Dictionary<UUID, string[]> m_UserNamesCache = new Dictionary<UUID, string[]>();
  187. #endregion Fields
  188. #region Properties
  189. /* Used by the loadbalancer plugin on GForge */
  190. public int SplitRegionID
  191. {
  192. get { return m_splitRegionID; }
  193. set { m_splitRegionID = value; }
  194. }
  195. public bool BordersLocked
  196. {
  197. get { return m_bordersLocked == 1; }
  198. set
  199. {
  200. if (value == true)
  201. m_bordersLocked = 1;
  202. else
  203. m_bordersLocked = 0;
  204. }
  205. }
  206. public new float TimeDilation
  207. {
  208. get { return m_sceneGraph.PhysicsScene.TimeDilation; }
  209. }
  210. public SceneCommunicationService SceneGridService
  211. {
  212. get { return m_sceneGridService; }
  213. }
  214. public ISimulationDataService SimulationDataService
  215. {
  216. get
  217. {
  218. if (m_SimulationDataService == null)
  219. {
  220. m_SimulationDataService = RequestModuleInterface<ISimulationDataService>();
  221. if (m_SimulationDataService == null)
  222. {
  223. throw new Exception("No ISimulationDataService available.");
  224. }
  225. }
  226. return m_SimulationDataService;
  227. }
  228. }
  229. public IEstateDataService EstateDataService
  230. {
  231. get
  232. {
  233. if (m_EstateDataService == null)
  234. {
  235. m_EstateDataService = RequestModuleInterface<IEstateDataService>();
  236. if (m_EstateDataService == null)
  237. {
  238. throw new Exception("No IEstateDataService available.");
  239. }
  240. }
  241. return m_EstateDataService;
  242. }
  243. }
  244. public IAssetService AssetService
  245. {
  246. get
  247. {
  248. if (m_AssetService == null)
  249. {
  250. m_AssetService = RequestModuleInterface<IAssetService>();
  251. if (m_AssetService == null)
  252. {
  253. throw new Exception("No IAssetService available.");
  254. }
  255. }
  256. return m_AssetService;
  257. }
  258. }
  259. public IAuthorizationService AuthorizationService
  260. {
  261. get
  262. {
  263. if (m_AuthorizationService == null)
  264. {
  265. m_AuthorizationService = RequestModuleInterface<IAuthorizationService>();
  266. //if (m_AuthorizationService == null)
  267. //{
  268. // // don't throw an exception if no authorization service is set for the time being
  269. // m_log.InfoFormat("[SCENE]: No Authorization service is configured");
  270. //}
  271. }
  272. return m_AuthorizationService;
  273. }
  274. }
  275. public IInventoryService InventoryService
  276. {
  277. get
  278. {
  279. if (m_InventoryService == null)
  280. {
  281. m_InventoryService = RequestModuleInterface<IInventoryService>();
  282. if (m_InventoryService == null)
  283. {
  284. throw new Exception("No IInventoryService available. This could happen if the config_include folder doesn't exist or if the OpenSim.ini [Architecture] section isn't set. Please also check that you have the correct version of your inventory service dll. Sometimes old versions of this dll will still exist. Do a clean checkout and re-create the opensim.ini from the opensim.ini.example.");
  285. }
  286. }
  287. return m_InventoryService;
  288. }
  289. }
  290. public IGridService GridService
  291. {
  292. get
  293. {
  294. if (m_GridService == null)
  295. {
  296. m_GridService = RequestModuleInterface<IGridService>();
  297. if (m_GridService == null)
  298. {
  299. throw new Exception("No IGridService available. This could happen if the config_include folder doesn't exist or if the OpenSim.ini [Architecture] section isn't set. Please also check that you have the correct version of your inventory service dll. Sometimes old versions of this dll will still exist. Do a clean checkout and re-create the opensim.ini from the opensim.ini.example.");
  300. }
  301. }
  302. return m_GridService;
  303. }
  304. }
  305. public ILibraryService LibraryService
  306. {
  307. get
  308. {
  309. if (m_LibraryService == null)
  310. m_LibraryService = RequestModuleInterface<ILibraryService>();
  311. return m_LibraryService;
  312. }
  313. }
  314. public ISimulationService SimulationService
  315. {
  316. get
  317. {
  318. if (m_simulationService == null)
  319. m_simulationService = RequestModuleInterface<ISimulationService>();
  320. return m_simulationService;
  321. }
  322. }
  323. public IAuthenticationService AuthenticationService
  324. {
  325. get
  326. {
  327. if (m_AuthenticationService == null)
  328. m_AuthenticationService = RequestModuleInterface<IAuthenticationService>();
  329. return m_AuthenticationService;
  330. }
  331. }
  332. public IPresenceService PresenceService
  333. {
  334. get
  335. {
  336. if (m_PresenceService == null)
  337. m_PresenceService = RequestModuleInterface<IPresenceService>();
  338. return m_PresenceService;
  339. }
  340. }
  341. public IUserAccountService UserAccountService
  342. {
  343. get
  344. {
  345. if (m_UserAccountService == null)
  346. m_UserAccountService = RequestModuleInterface<IUserAccountService>();
  347. return m_UserAccountService;
  348. }
  349. }
  350. public IAvatarService AvatarService
  351. {
  352. get
  353. {
  354. if (m_AvatarService == null)
  355. m_AvatarService = RequestModuleInterface<IAvatarService>();
  356. return m_AvatarService;
  357. }
  358. }
  359. public IGridUserService GridUserService
  360. {
  361. get
  362. {
  363. if (m_GridUserService == null)
  364. m_GridUserService = RequestModuleInterface<IGridUserService>();
  365. return m_GridUserService;
  366. }
  367. }
  368. public IAttachmentsModule AttachmentsModule { get; set; }
  369. public IAvatarFactory AvatarFactory
  370. {
  371. get { return m_AvatarFactory; }
  372. }
  373. public ICapabilitiesModule CapsModule
  374. {
  375. get { return m_capsModule; }
  376. }
  377. public int MonitorFrameTime { get { return frameMS; } }
  378. public int MonitorPhysicsUpdateTime { get { return physicsMS; } }
  379. public int MonitorPhysicsSyncTime { get { return physicsMS2; } }
  380. public int MonitorOtherTime { get { return otherMS; } }
  381. public int MonitorTempOnRezTime { get { return tempOnRezMS; } }
  382. public int MonitorEventTime { get { return eventMS; } } // This may need to be divided into each event?
  383. public int MonitorBackupTime { get { return backupMS; } }
  384. public int MonitorTerrainTime { get { return terrainMS; } }
  385. public int MonitorLandTime { get { return landMS; } }
  386. public int MonitorLastFrameTick { get { return lastCompletedFrame; } }
  387. public UpdatePrioritizationSchemes UpdatePrioritizationScheme { get { return m_priorityScheme; } }
  388. public bool IsReprioritizationEnabled { get { return m_reprioritizationEnabled; } }
  389. public double ReprioritizationInterval { get { return m_reprioritizationInterval; } }
  390. public double RootReprioritizationDistance { get { return m_rootReprioritizationDistance; } }
  391. public double ChildReprioritizationDistance { get { return m_childReprioritizationDistance; } }
  392. public AgentCircuitManager AuthenticateHandler
  393. {
  394. get { return m_authenticateHandler; }
  395. }
  396. public SceneGraph SceneContents
  397. {
  398. get { return m_sceneGraph; }
  399. }
  400. public bool UseBackup
  401. {
  402. get { return m_useBackup; }
  403. }
  404. // an instance to the physics plugin's Scene object.
  405. public PhysicsScene PhysicsScene
  406. {
  407. get { return m_sceneGraph.PhysicsScene; }
  408. set
  409. {
  410. // If we're not doing the initial set
  411. // Then we've got to remove the previous
  412. // event handler
  413. if (PhysicsScene != null && PhysicsScene.SupportsNINJAJoints)
  414. {
  415. PhysicsScene.OnJointMoved -= jointMoved;
  416. PhysicsScene.OnJointDeactivated -= jointDeactivated;
  417. PhysicsScene.OnJointErrorMessage -= jointErrorMessage;
  418. }
  419. m_sceneGraph.PhysicsScene = value;
  420. if (PhysicsScene != null && m_sceneGraph.PhysicsScene.SupportsNINJAJoints)
  421. {
  422. // register event handlers to respond to joint movement/deactivation
  423. PhysicsScene.OnJointMoved += jointMoved;
  424. PhysicsScene.OnJointDeactivated += jointDeactivated;
  425. PhysicsScene.OnJointErrorMessage += jointErrorMessage;
  426. }
  427. }
  428. }
  429. // This gets locked so things stay thread safe.
  430. public object SyncRoot
  431. {
  432. get { return m_sceneGraph.m_syncRoot; }
  433. }
  434. /// <summary>
  435. /// This is for llGetRegionFPS
  436. /// </summary>
  437. public float SimulatorFPS
  438. {
  439. get { return StatsReporter.getLastReportedSimFPS(); }
  440. }
  441. public float[] SimulatorStats
  442. {
  443. get { return StatsReporter.getLastReportedSimStats(); }
  444. }
  445. public string DefaultScriptEngine
  446. {
  447. get { return m_defaultScriptEngine; }
  448. }
  449. public EntityManager Entities
  450. {
  451. get { return m_sceneGraph.Entities; }
  452. }
  453. #endregion Properties
  454. #region Constructors
  455. public Scene(RegionInfo regInfo, AgentCircuitManager authen,
  456. SceneCommunicationService sceneGridService,
  457. ISimulationDataService simDataService, IEstateDataService estateDataService,
  458. ModuleLoader moduleLoader, bool dumpAssetsToFile, bool physicalPrim,
  459. bool SeeIntoRegionFromNeighbor, IConfigSource config, string simulatorVersion)
  460. {
  461. m_config = config;
  462. Random random = new Random();
  463. BordersLocked = true;
  464. Border northBorder = new Border();
  465. northBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue, (int)Constants.RegionSize); //<---
  466. northBorder.CrossDirection = Cardinals.N;
  467. NorthBorders.Add(northBorder);
  468. Border southBorder = new Border();
  469. southBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue, 0); //--->
  470. southBorder.CrossDirection = Cardinals.S;
  471. SouthBorders.Add(southBorder);
  472. Border eastBorder = new Border();
  473. eastBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue, (int)Constants.RegionSize); //<---
  474. eastBorder.CrossDirection = Cardinals.E;
  475. EastBorders.Add(eastBorder);
  476. Border westBorder = new Border();
  477. westBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue, 0); //--->
  478. westBorder.CrossDirection = Cardinals.W;
  479. WestBorders.Add(westBorder);
  480. BordersLocked = false;
  481. m_lastAllocatedLocalId = (uint)(random.NextDouble() * (double)(uint.MaxValue/2))+(uint)(uint.MaxValue/4);
  482. m_moduleLoader = moduleLoader;
  483. m_authenticateHandler = authen;
  484. m_sceneGridService = sceneGridService;
  485. m_SimulationDataService = simDataService;
  486. m_EstateDataService = estateDataService;
  487. m_regInfo = regInfo;
  488. m_regionHandle = m_regInfo.RegionHandle;
  489. m_regionName = m_regInfo.RegionName;
  490. m_lastUpdate = Util.EnvironmentTickCount();
  491. m_physicalPrim = physicalPrim;
  492. m_seeIntoRegionFromNeighbor = SeeIntoRegionFromNeighbor;
  493. m_eventManager = new EventManager();
  494. m_permissions = new ScenePermissions(this);
  495. m_asyncSceneObjectDeleter = new AsyncSceneObjectGroupDeleter(this);
  496. m_asyncSceneObjectDeleter.Enabled = true;
  497. m_asyncInventorySender = new AsyncInventorySender(this);
  498. #region Region Settings
  499. // Load region settings
  500. m_regInfo.RegionSettings = simDataService.LoadRegionSettings(m_regInfo.RegionID);
  501. if (estateDataService != null)
  502. m_regInfo.EstateSettings = estateDataService.LoadEstateSettings(m_regInfo.RegionID, false);
  503. #endregion Region Settings
  504. MainConsole.Instance.Commands.AddCommand("region", false, "reload estate",
  505. "reload estate",
  506. "Reload the estate data", HandleReloadEstate);
  507. MainConsole.Instance.Commands.AddCommand("region", false, "delete object owner",
  508. "delete object owner <UUID>",
  509. "Delete object by owner", HandleDeleteObject);
  510. MainConsole.Instance.Commands.AddCommand("region", false, "delete object creator",
  511. "delete object creator <UUID>",
  512. "Delete object by creator", HandleDeleteObject);
  513. MainConsole.Instance.Commands.AddCommand("region", false, "delete object uuid",
  514. "delete object uuid <UUID>",
  515. "Delete object by uuid", HandleDeleteObject);
  516. MainConsole.Instance.Commands.AddCommand("region", false, "delete object name",
  517. "delete object name <name>",
  518. "Delete object by name", HandleDeleteObject);
  519. MainConsole.Instance.Commands.AddCommand("region", false, "delete object outside",
  520. "delete object outside",
  521. "Delete all objects outside boundaries", HandleDeleteObject);
  522. //Bind Storage Manager functions to some land manager functions for this scene
  523. EventManager.OnLandObjectAdded +=
  524. new EventManager.LandObjectAdded(simDataService.StoreLandObject);
  525. EventManager.OnLandObjectRemoved +=
  526. new EventManager.LandObjectRemoved(simDataService.RemoveLandObject);
  527. m_sceneGraph = new SceneGraph(this, m_regInfo);
  528. // If the scene graph has an Unrecoverable error, restart this sim.
  529. // Currently the only thing that causes it to happen is two kinds of specific
  530. // Physics based crashes.
  531. //
  532. // Out of memory
  533. // Operating system has killed the plugin
  534. m_sceneGraph.UnRecoverableError += RestartNow;
  535. RegisterDefaultSceneEvents();
  536. DumpAssetsToFile = dumpAssetsToFile;
  537. m_scripts_enabled = !RegionInfo.RegionSettings.DisableScripts;
  538. m_physics_enabled = !RegionInfo.RegionSettings.DisablePhysics;
  539. StatsReporter = new SimStatsReporter(this);
  540. StatsReporter.OnSendStatsResult += SendSimStatsPackets;
  541. StatsReporter.OnStatsIncorrect += m_sceneGraph.RecalculateStats;
  542. // Old
  543. /*
  544. m_simulatorVersion = simulatorVersion
  545. + " (OS " + Util.GetOperatingSystemInformation() + ")"
  546. + " ChilTasks:" + m_seeIntoRegionFromNeighbor.ToString()
  547. + " PhysPrim:" + m_physicalPrim.ToString();
  548. */
  549. m_simulatorVersion = simulatorVersion + " (" + Util.GetRuntimeInformation() + ")";
  550. #region Region Config
  551. try
  552. {
  553. // Region config overrides global config
  554. //
  555. IConfig startupConfig = m_config.Configs["Startup"];
  556. m_defaultDrawDistance = startupConfig.GetFloat("DefaultDrawDistance",m_defaultDrawDistance);
  557. m_useBackup = startupConfig.GetBoolean("UseSceneBackup", m_useBackup);
  558. if (!m_useBackup)
  559. m_log.InfoFormat("[SCENE]: Backup has been disabled for {0}", RegionInfo.RegionName);
  560. //Animation states
  561. m_useFlySlow = startupConfig.GetBoolean("enableflyslow", false);
  562. // TODO: Change default to true once the feature is supported
  563. m_usePreJump = startupConfig.GetBoolean("enableprejump", false);
  564. m_maxNonphys = startupConfig.GetFloat("NonPhysicalPrimMax", m_maxNonphys);
  565. if (RegionInfo.NonphysPrimMax > 0)
  566. {
  567. m_maxNonphys = RegionInfo.NonphysPrimMax;
  568. }
  569. m_maxPhys = startupConfig.GetFloat("PhysicalPrimMax", m_maxPhys);
  570. if (RegionInfo.PhysPrimMax > 0)
  571. {
  572. m_maxPhys = RegionInfo.PhysPrimMax;
  573. }
  574. // Here, if clamping is requested in either global or
  575. // local config, it will be used
  576. //
  577. m_clampPrimSize = startupConfig.GetBoolean("ClampPrimSize", m_clampPrimSize);
  578. if (RegionInfo.ClampPrimSize)
  579. {
  580. m_clampPrimSize = true;
  581. }
  582. m_trustBinaries = startupConfig.GetBoolean("TrustBinaries", m_trustBinaries);
  583. m_allowScriptCrossings = startupConfig.GetBoolean("AllowScriptCrossing", m_allowScriptCrossings);
  584. m_dontPersistBefore =
  585. startupConfig.GetLong("MinimumTimeBeforePersistenceConsidered", DEFAULT_MIN_TIME_FOR_PERSISTENCE);
  586. m_dontPersistBefore *= 10000000;
  587. m_persistAfter =
  588. startupConfig.GetLong("MaximumTimeBeforePersistenceConsidered", DEFAULT_MAX_TIME_FOR_PERSISTENCE);
  589. m_persistAfter *= 10000000;
  590. m_defaultScriptEngine = startupConfig.GetString("DefaultScriptEngine", "XEngine");
  591. IConfig packetConfig = m_config.Configs["PacketPool"];
  592. if (packetConfig != null)
  593. {
  594. PacketPool.Instance.RecyclePackets = packetConfig.GetBoolean("RecyclePackets", true);
  595. PacketPool.Instance.RecycleDataBlocks = packetConfig.GetBoolean("RecycleDataBlocks", true);
  596. }
  597. m_strictAccessControl = startupConfig.GetBoolean("StrictAccessControl", m_strictAccessControl);
  598. m_generateMaptiles = startupConfig.GetBoolean("GenerateMaptiles", true);
  599. if (m_generateMaptiles)
  600. {
  601. int maptileRefresh = startupConfig.GetInt("MaptileRefresh", 0);
  602. if (maptileRefresh != 0)
  603. {
  604. m_mapGenerationTimer.Interval = maptileRefresh * 1000;
  605. m_mapGenerationTimer.Elapsed += RegenerateMaptile;
  606. m_mapGenerationTimer.AutoReset = true;
  607. m_mapGenerationTimer.Start();
  608. }
  609. }
  610. else
  611. {
  612. string tile = startupConfig.GetString("MaptileStaticUUID", UUID.Zero.ToString());
  613. UUID tileID;
  614. if (UUID.TryParse(tile, out tileID))
  615. {
  616. RegionInfo.RegionSettings.TerrainImageID = tileID;
  617. }
  618. }
  619. }
  620. catch
  621. {
  622. m_log.Warn("[SCENE]: Failed to load StartupConfig");
  623. }
  624. #endregion Region Config
  625. #region Interest Management
  626. if (m_config != null)
  627. {
  628. IConfig interestConfig = m_config.Configs["InterestManagement"];
  629. if (interestConfig != null)
  630. {
  631. string update_prioritization_scheme = interestConfig.GetString("UpdatePrioritizationScheme", "Time").Trim().ToLower();
  632. try
  633. {
  634. m_priorityScheme = (UpdatePrioritizationSchemes)Enum.Parse(typeof(UpdatePrioritizationSchemes), update_prioritization_scheme, true);
  635. }
  636. catch (Exception)
  637. {
  638. m_log.Warn("[PRIORITIZER]: UpdatePrioritizationScheme was not recognized, setting to default prioritizer Time");
  639. m_priorityScheme = UpdatePrioritizationSchemes.Time;
  640. }
  641. m_reprioritizationEnabled = interestConfig.GetBoolean("ReprioritizationEnabled", true);
  642. m_reprioritizationInterval = interestConfig.GetDouble("ReprioritizationInterval", 5000.0);
  643. m_rootReprioritizationDistance = interestConfig.GetDouble("RootReprioritizationDistance", 10.0);
  644. m_childReprioritizationDistance = interestConfig.GetDouble("ChildReprioritizationDistance", 20.0);
  645. }
  646. }
  647. m_log.Info("[SCENE]: Using the " + m_priorityScheme + " prioritization scheme");
  648. #endregion Interest Management
  649. }
  650. /// <summary>
  651. /// Mock constructor for scene group persistency unit tests.
  652. /// SceneObjectGroup RegionId property is delegated to Scene.
  653. /// </summary>
  654. /// <param name="regInfo"></param>
  655. public Scene(RegionInfo regInfo)
  656. {
  657. BordersLocked = true;
  658. Border northBorder = new Border();
  659. northBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue, (int)Constants.RegionSize); //<---
  660. northBorder.CrossDirection = Cardinals.N;
  661. NorthBorders.Add(northBorder);
  662. Border southBorder = new Border();
  663. southBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue,0); //--->
  664. southBorder.CrossDirection = Cardinals.S;
  665. SouthBorders.Add(southBorder);
  666. Border eastBorder = new Border();
  667. eastBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue, (int)Constants.RegionSize); //<---
  668. eastBorder.CrossDirection = Cardinals.E;
  669. EastBorders.Add(eastBorder);
  670. Border westBorder = new Border();
  671. westBorder.BorderLine = new Vector3(float.MinValue, float.MaxValue,0); //--->
  672. westBorder.CrossDirection = Cardinals.W;
  673. WestBorders.Add(westBorder);
  674. BordersLocked = false;
  675. m_regInfo = regInfo;
  676. m_eventManager = new EventManager();
  677. m_permissions = new ScenePermissions(this);
  678. m_lastUpdate = Util.EnvironmentTickCount();
  679. }
  680. #endregion
  681. #region Startup / Close Methods
  682. public bool ShuttingDown
  683. {
  684. get { return shuttingdown; }
  685. }
  686. /// <value>
  687. /// The scene graph for this scene
  688. /// </value>
  689. /// TODO: Possibly stop other classes being able to manipulate this directly.
  690. public SceneGraph SceneGraph
  691. {
  692. get { return m_sceneGraph; }
  693. }
  694. protected virtual void RegisterDefaultSceneEvents()
  695. {
  696. IDialogModule dm = RequestModuleInterface<IDialogModule>();
  697. if (dm != null)
  698. m_eventManager.OnPermissionError += dm.SendAlertToUser;
  699. }
  700. public override string GetSimulatorVersion()
  701. {
  702. return m_simulatorVersion;
  703. }
  704. /// <summary>
  705. /// Another region is up.
  706. ///
  707. /// We only add it to the neighbor list if it's within 1 region from here.
  708. /// Agents may have draw distance values that cross two regions though, so
  709. /// we add it to the notify list regardless of distance. We'll check
  710. /// the agent's draw distance before notifying them though.
  711. /// </summary>
  712. /// <param name="otherRegion">RegionInfo handle for the new region.</param>
  713. /// <returns>True after all operations complete, throws exceptions otherwise.</returns>
  714. public override void OtherRegionUp(GridRegion otherRegion)
  715. {
  716. uint xcell = (uint)((int)otherRegion.RegionLocX / (int)Constants.RegionSize);
  717. uint ycell = (uint)((int)otherRegion.RegionLocY / (int)Constants.RegionSize);
  718. //m_log.InfoFormat("[SCENE]: (on region {0}): Region {1} up in coords {2}-{3}",
  719. // RegionInfo.RegionName, otherRegion.RegionName, xcell, ycell);
  720. if (RegionInfo.RegionHandle != otherRegion.RegionHandle)
  721. {
  722. // If these are cast to INT because long + negative values + abs returns invalid data
  723. int resultX = Math.Abs((int)xcell - (int)RegionInfo.RegionLocX);
  724. int resultY = Math.Abs((int)ycell - (int)RegionInfo.RegionLocY);
  725. if (resultX <= 1 && resultY <= 1)
  726. {
  727. // Let the grid service module know, so this can be cached
  728. m_eventManager.TriggerOnRegionUp(otherRegion);
  729. try
  730. {
  731. ForEachScenePresence(delegate(ScenePresence agent)
  732. {
  733. // If agent is a root agent.
  734. if (!agent.IsChildAgent)
  735. {
  736. //agent.ControllingClient.new
  737. //this.CommsManager.InterRegion.InformRegionOfChildAgent(otherRegion.RegionHandle, agent.ControllingClient.RequestClientInfo());
  738. List<ulong> old = new List<ulong>();
  739. old.Add(otherRegion.RegionHandle);
  740. agent.DropOldNeighbours(old);
  741. if (m_teleportModule != null)
  742. m_teleportModule.EnableChildAgent(agent, otherRegion);
  743. }
  744. }
  745. );
  746. }
  747. catch (NullReferenceException)
  748. {
  749. // This means that we're not booted up completely yet.
  750. // This shouldn't happen too often anymore.
  751. m_log.Error("[SCENE]: Couldn't inform client of regionup because we got a null reference exception");
  752. }
  753. }
  754. else
  755. {
  756. m_log.Info("[INTERGRID]: Got notice about far away Region: " + otherRegion.RegionName.ToString() +
  757. " at (" + otherRegion.RegionLocX.ToString() + ", " +
  758. otherRegion.RegionLocY.ToString() + ")");
  759. }
  760. }
  761. }
  762. public void AddNeighborRegion(RegionInfo region)
  763. {
  764. lock (m_neighbours)
  765. {
  766. if (!CheckNeighborRegion(region))
  767. {
  768. m_neighbours.Add(region);
  769. }
  770. }
  771. }
  772. public bool CheckNeighborRegion(RegionInfo region)
  773. {
  774. bool found = false;
  775. lock (m_neighbours)
  776. {
  777. foreach (RegionInfo reg in m_neighbours)
  778. {
  779. if (reg.RegionHandle == region.RegionHandle)
  780. {
  781. found = true;
  782. break;
  783. }
  784. }
  785. }
  786. return found;
  787. }
  788. // Alias IncomingHelloNeighbour OtherRegionUp, for now
  789. public GridRegion IncomingHelloNeighbour(RegionInfo neighbour)
  790. {
  791. OtherRegionUp(new GridRegion(neighbour));
  792. return new GridRegion(RegionInfo);
  793. }
  794. // This causes the region to restart immediatley.
  795. public void RestartNow()
  796. {
  797. IConfig startupConfig = m_config.Configs["Startup"];
  798. if (startupConfig != null)
  799. {
  800. if (startupConfig.GetBoolean("InworldRestartShutsDown", false))
  801. {
  802. MainConsole.Instance.RunCommand("shutdown");
  803. return;
  804. }
  805. }
  806. if (PhysicsScene != null)
  807. {
  808. PhysicsScene.Dispose();
  809. }
  810. m_log.Error("[REGION]: Closing");
  811. Close();
  812. m_log.Error("[REGION]: Firing Region Restart Message");
  813. base.Restart();
  814. }
  815. // This is a helper function that notifies root agents in this region that a new sim near them has come up
  816. // This is in the form of a timer because when an instance of OpenSim.exe is started,
  817. // Even though the sims initialize, they don't listen until 'all of the sims are initialized'
  818. // If we tell an agent about a sim that's not listening yet, the agent will not be able to connect to it.
  819. // subsequently the agent will never see the region come back online.
  820. public void RestartNotifyWaitElapsed(object sender, ElapsedEventArgs e)
  821. {
  822. m_restartWaitTimer.Stop();
  823. lock (m_regionRestartNotifyList)
  824. {
  825. foreach (RegionInfo region in m_regionRestartNotifyList)
  826. {
  827. GridRegion r = new GridRegion(region);
  828. try
  829. {
  830. ForEachScenePresence(delegate(ScenePresence agent)
  831. {
  832. // If agent is a root agent.
  833. if (!agent.IsChildAgent)
  834. {
  835. if (m_teleportModule != null)
  836. m_teleportModule.EnableChildAgent(agent, r);
  837. }
  838. }
  839. );
  840. }
  841. catch (NullReferenceException)
  842. {
  843. // This means that we're not booted up completely yet.
  844. // This shouldn't happen too often anymore.
  845. }
  846. }
  847. // Reset list to nothing.
  848. m_regionRestartNotifyList.Clear();
  849. }
  850. }
  851. public void SetSceneCoreDebug(bool ScriptEngine, bool CollisionEvents, bool PhysicsEngine)
  852. {
  853. if (m_scripts_enabled != !ScriptEngine)
  854. {
  855. if (ScriptEngine)
  856. {
  857. m_log.Info("Stopping all Scripts in Scene");
  858. EntityBase[] entities = Entities.GetEntities();
  859. foreach (EntityBase ent in entities)
  860. {
  861. if (ent is SceneObjectGroup)
  862. ((SceneObjectGroup)ent).RemoveScriptInstances(false);
  863. }
  864. }
  865. else
  866. {
  867. m_log.Info("Starting all Scripts in Scene");
  868. EntityBase[] entities = Entities.GetEntities();
  869. foreach (EntityBase ent in entities)
  870. {
  871. if (ent is SceneObjectGroup)
  872. {
  873. SceneObjectGroup sog = (SceneObjectGroup)ent;
  874. sog.CreateScriptInstances(0, false, DefaultScriptEngine, 0);
  875. sog.ResumeScripts();
  876. }
  877. }
  878. }
  879. m_scripts_enabled = !ScriptEngine;
  880. m_log.Info("[TOTEDD]: Here is the method to trigger disabling of the scripting engine");
  881. }
  882. if (m_physics_enabled != !PhysicsEngine)
  883. {
  884. m_physics_enabled = !PhysicsEngine;
  885. }
  886. }
  887. public int GetInaccurateNeighborCount()
  888. {
  889. return m_neighbours.Count;
  890. }
  891. // This is the method that shuts down the scene.
  892. public override void Close()
  893. {
  894. m_log.InfoFormat("[SCENE]: Closing down the single simulator: {0}", RegionInfo.RegionName);
  895. m_restartTimer.Stop();
  896. m_restartTimer.Close();
  897. // Kick all ROOT agents with the message, 'The simulator is going down'
  898. ForEachScenePresence(delegate(ScenePresence avatar)
  899. {
  900. if (avatar.KnownChildRegionHandles.Contains(RegionInfo.RegionHandle))
  901. avatar.KnownChildRegionHandles.Remove(RegionInfo.RegionHandle);
  902. if (!avatar.IsChildAgent)
  903. avatar.ControllingClient.Kick("The simulator is going down.");
  904. avatar.ControllingClient.SendShutdownConnectionNotice();
  905. });
  906. // Wait here, or the kick messages won't actually get to the agents before the scene terminates.
  907. Thread.Sleep(500);
  908. // Stop all client threads.
  909. ForEachScenePresence(delegate(ScenePresence avatar) { avatar.ControllingClient.Close(); });
  910. // Stop updating the scene objects and agents.
  911. //m_heartbeatTimer.Close();
  912. shuttingdown = true;
  913. m_log.Debug("[SCENE]: Persisting changed objects");
  914. EventManager.TriggerSceneShuttingDown(this);
  915. EntityBase[] entities = GetEntities();
  916. foreach (EntityBase entity in entities)
  917. {
  918. if (!entity.IsDeleted && entity is SceneObjectGroup && ((SceneObjectGroup)entity).HasGroupChanged)
  919. {
  920. ((SceneObjectGroup)entity).ProcessBackup(SimulationDataService, false);
  921. }
  922. }
  923. m_sceneGraph.Close();
  924. // De-register with region communications (events cleanup)
  925. UnRegisterRegionWithComms();
  926. // call the base class Close method.
  927. base.Close();
  928. }
  929. /// <summary>
  930. /// Start the timer which triggers regular scene updates
  931. /// </summary>
  932. public void StartTimer()
  933. {
  934. //m_log.Debug("[SCENE]: Starting timer");
  935. //m_heartbeatTimer.Enabled = true;
  936. //m_heartbeatTimer.Interval = (int)(m_timespan * 1000);
  937. //m_heartbeatTimer.Elapsed += new ElapsedEventHandler(Heartbeat);
  938. if (HeartbeatThread != null)
  939. {
  940. HeartbeatThread.Abort();
  941. HeartbeatThread = null;
  942. }
  943. m_lastUpdate = Util.EnvironmentTickCount();
  944. HeartbeatThread = Watchdog.StartThread(Heartbeat, "Heartbeat for region " + RegionInfo.RegionName, ThreadPriority.Normal, false);
  945. }
  946. /// <summary>
  947. /// Sets up references to modules required by the scene
  948. /// </summary>
  949. public void SetModuleInterfaces()
  950. {
  951. m_xmlrpcModule = RequestModuleInterface<IXMLRPC>();
  952. m_worldCommModule = RequestModuleInterface<IWorldComm>();
  953. XferManager = RequestModuleInterface<IXfer>();
  954. m_AvatarFactory = RequestModuleInterface<IAvatarFactory>();
  955. AttachmentsModule = RequestModuleInterface<IAttachmentsModule>();
  956. m_serialiser = RequestModuleInterface<IRegionSerialiserModule>();
  957. m_dialogModule = RequestModuleInterface<IDialogModule>();
  958. m_capsModule = RequestModuleInterface<ICapabilitiesModule>();
  959. m_teleportModule = RequestModuleInterface<IEntityTransferModule>();
  960. // Shoving this in here for now, because we have the needed
  961. // interfaces at this point
  962. //
  963. // TODO: Find a better place for this
  964. //
  965. while (m_regInfo.EstateSettings.EstateOwner == UUID.Zero && MainConsole.Instance != null)
  966. {
  967. MainConsole.Instance.OutputFormat("Estate {0} has no owner set.", m_regInfo.EstateSettings.EstateName);
  968. List<char> excluded = new List<char>(new char[1]{' '});
  969. string first = MainConsole.Instance.CmdPrompt("Estate owner first name", "Test", excluded);
  970. string last = MainConsole.Instance.CmdPrompt("Estate owner last name", "User", excluded);
  971. UserAccount account = UserAccountService.GetUserAccount(m_regInfo.ScopeID, first, last);
  972. if (account == null)
  973. {
  974. // Create a new account
  975. account = new UserAccount(m_regInfo.ScopeID, first, last, String.Empty);
  976. if (account.ServiceURLs == null || (account.ServiceURLs != null && account.ServiceURLs.Count == 0))
  977. {
  978. account.ServiceURLs = new Dictionary<string, object>();
  979. account.ServiceURLs["HomeURI"] = string.Empty;
  980. account.ServiceURLs["GatekeeperURI"] = string.Empty;
  981. account.ServiceURLs["InventoryServerURI"] = string.Empty;
  982. account.ServiceURLs["AssetServerURI"] = string.Empty;
  983. }
  984. if (UserAccountService.StoreUserAccount(account))
  985. {
  986. string password = MainConsole.Instance.PasswdPrompt("Password");
  987. string email = MainConsole.Instance.CmdPrompt("Email", "");
  988. account.Email = email;
  989. UserAccountService.StoreUserAccount(account);
  990. bool success = false;
  991. success = AuthenticationService.SetPassword(account.PrincipalID, password);
  992. if (!success)
  993. m_log.WarnFormat("[USER ACCOUNT SERVICE]: Unable to set password for account {0} {1}.",
  994. first, last);
  995. GridRegion home = null;
  996. if (GridService != null)
  997. {
  998. List<GridRegion> defaultRegions = GridService.GetDefaultRegions(UUID.Zero);
  999. if (defaultRegions != null && defaultRegions.Count >= 1)
  1000. home = defaultRegions[0];
  1001. if (GridUserService != null && home != null)
  1002. GridUserService.SetHome(account.PrincipalID.ToString(), home.RegionID, new Vector3(128, 128, 0), new Vector3(0, 1, 0));
  1003. else
  1004. m_log.WarnFormat("[USER ACCOUNT SERVICE]: Unable to set home for account {0} {1}.",
  1005. first, last);
  1006. }
  1007. else
  1008. m_log.WarnFormat("[USER ACCOUNT SERVICE]: Unable to retrieve home region for account {0} {1}.",
  1009. first, last);
  1010. if (InventoryService != null)
  1011. success = InventoryService.CreateUserInventory(account.PrincipalID);
  1012. if (!success)
  1013. m_log.WarnFormat("[USER ACCOUNT SERVICE]: Unable to create inventory for account {0} {1}.",
  1014. first, last);
  1015. m_log.InfoFormat("[USER ACCOUNT SERVICE]: Account {0} {1} created successfully", first, last);
  1016. m_regInfo.EstateSettings.EstateOwner = account.PrincipalID;
  1017. m_regInfo.EstateSettings.Save();
  1018. }
  1019. else
  1020. m_log.ErrorFormat("[SCENE]: Unable to store account. If this simulator is connected to a grid, you must create the estate owner account first.");
  1021. }
  1022. else
  1023. {
  1024. m_regInfo.EstateSettings.EstateOwner = account.PrincipalID;
  1025. m_regInfo.EstateSettings.Save();
  1026. }
  1027. }
  1028. }
  1029. #endregion
  1030. #region Update Methods
  1031. /// <summary>
  1032. /// Performs per-frame updates regularly
  1033. /// </summary>
  1034. private void Heartbeat()
  1035. {
  1036. if (!Monitor.TryEnter(m_heartbeatLock))
  1037. {
  1038. Watchdog.RemoveThread();
  1039. return;
  1040. }
  1041. try
  1042. {
  1043. while (!shuttingdown)
  1044. Update();
  1045. m_lastUpdate = Util.EnvironmentTickCount();
  1046. m_firstHeartbeat = false;
  1047. }
  1048. catch (ThreadAbortException)
  1049. {
  1050. }
  1051. finally
  1052. {
  1053. Monitor.Pulse(m_heartbeatLock);
  1054. Monitor.Exit(m_heartbeatLock);
  1055. }
  1056. Watchdog.RemoveThread();
  1057. }
  1058. public override void Update()
  1059. {
  1060. TimeSpan SinceLastFrame = DateTime.UtcNow - m_lastupdate;
  1061. float physicsFPS = 0f;
  1062. int maintc = Util.EnvironmentTickCount();
  1063. int tmpFrameMS = maintc;
  1064. tempOnRezMS = eventMS = backupMS = terrainMS = landMS = 0;
  1065. // Increment the frame counter
  1066. ++Frame;
  1067. try
  1068. {
  1069. // Check if any objects have reached their targets
  1070. CheckAtTargets();
  1071. // Update SceneObjectGroups that have scheduled themselves for updates
  1072. // Objects queue their updates onto all scene presences
  1073. if (Frame % m_update_objects == 0)
  1074. m_sceneGraph.UpdateObjectGroups();
  1075. // Run through all ScenePresences looking for updates
  1076. // Presence updates and queued object updates for each presence are sent to clients
  1077. if (Frame % m_update_presences == 0)
  1078. m_sceneGraph.UpdatePresences();
  1079. // Coarse locations relate to positions of green dots on the mini-map (on a SecondLife client)
  1080. if (Frame % m_update_coarse_locations == 0)
  1081. {
  1082. List<Vector3> coarseLocations;
  1083. List<UUID> avatarUUIDs;
  1084. SceneGraph.GetCoarseLocations(out coarseLocations, out avatarUUIDs, 60);
  1085. // Send coarse locations to clients
  1086. ForEachScenePresence(delegate(ScenePresence presence)
  1087. {
  1088. presence.SendCoarseLocations(coarseLocations, avatarUUIDs);
  1089. });
  1090. }
  1091. int tmpPhysicsMS2 = Util.EnvironmentTickCount();
  1092. if ((Frame % m_update_physics == 0) && m_physics_enabled)
  1093. m_sceneGraph.UpdatePreparePhysics();
  1094. physicsMS2 = Util.EnvironmentTickCountSubtract(tmpPhysicsMS2);
  1095. // Apply any pending avatar force input to the avatar's velocity
  1096. if (Frame % m_update_entitymovement == 0)
  1097. m_sceneGraph.UpdateScenePresenceMovement();
  1098. // Perform the main physics update. This will do the actual work of moving objects and avatars according to their
  1099. // velocity
  1100. int tmpPhysicsMS = Util.EnvironmentTickCount();
  1101. if (Frame % m_update_physics == 0)
  1102. {
  1103. if (m_physics_enabled)
  1104. physicsFPS = m_sceneGraph.UpdatePhysics(Math.Max(SinceLastFrame.TotalSeconds, m_timespan));
  1105. if (SynchronizeScene != null)
  1106. SynchronizeScene(this);
  1107. }
  1108. physicsMS = Util.EnvironmentTickCountSubtract(tmpPhysicsMS);
  1109. // Delete temp-on-rez stuff
  1110. if (Frame % 1000 == 0 && !m_cleaningTemps)
  1111. {
  1112. int tmpTempOnRezMS = Util.EnvironmentTickCount();
  1113. m_cleaningTemps = true;
  1114. Util.FireAndForget(delegate { CleanTempObjects(); m_cleaningTemps = false; });
  1115. tempOnRezMS = Util.EnvironmentTickCountSubtract(tmpTempOnRezMS);
  1116. }
  1117. if (RegionStatus != RegionStatus.SlaveScene)
  1118. {
  1119. if (Frame % m_update_events == 0)
  1120. {
  1121. int evMS = Util.EnvironmentTickCount();
  1122. UpdateEvents();
  1123. eventMS = Util.EnvironmentTickCountSubtract(evMS); ;
  1124. }
  1125. if (Frame % m_update_backup == 0)
  1126. {
  1127. int backMS = Util.EnvironmentTickCount();
  1128. UpdateStorageBackup();
  1129. backupMS = Util.EnvironmentTickCountSubtract(backMS);
  1130. }
  1131. if (Frame % m_update_terrain == 0)
  1132. {
  1133. int terMS = Util.EnvironmentTickCount();
  1134. UpdateTerrain();
  1135. terrainMS = Util.EnvironmentTickCountSubtract(terMS);
  1136. }
  1137. //if (Frame % m_update_land == 0)
  1138. //{
  1139. // int ldMS = Util.EnvironmentTickCount();
  1140. // UpdateLand();
  1141. // landMS = Util.EnvironmentTickCountSubtract(ldMS);
  1142. //}
  1143. frameMS = Util.EnvironmentTickCountSubtract(tmpFrameMS);
  1144. otherMS = tempOnRezMS + eventMS + backupMS + terrainMS + landMS;
  1145. lastCompletedFrame = Util.EnvironmentTickCount();
  1146. // if (Frame%m_update_avatars == 0)
  1147. // UpdateInWorldTime();
  1148. StatsReporter.AddPhysicsFPS(physicsFPS);
  1149. StatsReporter.AddTimeDilation(TimeDilation);
  1150. StatsReporter.AddFPS(1);
  1151. StatsReporter.SetRootAgents(m_sceneGraph.GetRootAgentCount());
  1152. StatsReporter.SetChildAgents(m_sceneGraph.GetChildAgentCount());
  1153. StatsReporter.SetObjects(m_sceneGraph.GetTotalObjectsCount());
  1154. StatsReporter.SetActiveObjects(m_sceneGraph.GetActiveObjectsCount());
  1155. StatsReporter.addFrameMS(frameMS);
  1156. StatsReporter.addPhysicsMS(physicsMS + physicsMS2);
  1157. StatsReporter.addOtherMS(otherMS);
  1158. StatsReporter.SetActiveScripts(m_sceneGraph.GetActiveScriptsCount());
  1159. StatsReporter.addScriptLines(m_sceneGraph.GetScriptLPS());
  1160. }
  1161. if (LoginsDisabled && Frame == 20)
  1162. {
  1163. // In 99.9% of cases it is a bad idea to manually force garbage collection. However,
  1164. // this is a rare case where we know we have just went through a long cycle of heap
  1165. // allocations, and there is no more work to be done until someone logs in
  1166. GC.Collect();
  1167. IConfig startupConfig = m_config.Configs["Startup"];
  1168. if (startupConfig == null || !startupConfig.GetBoolean("StartDisabled", false))
  1169. {
  1170. // This handles a case of a region having no scripts for the RegionReady module
  1171. if (m_sceneGraph.GetActiveScriptsCount() == 0)
  1172. {
  1173. // need to be able to tell these have changed in RegionReady
  1174. LoginLock = false;
  1175. EventManager.TriggerLoginsEnabled(RegionInfo.RegionName);
  1176. }
  1177. m_log.DebugFormat("[REGION]: Enabling logins for {0}", RegionInfo.RegionName);
  1178. // For RegionReady lockouts
  1179. if( LoginLock == false)
  1180. {
  1181. LoginsDisabled = false;
  1182. }
  1183. m_sceneGridService.InformNeighborsThatRegionisUp(RequestModuleInterface<INeighbourService>(), RegionInfo);
  1184. }
  1185. else
  1186. {
  1187. StartDisabled = true;
  1188. LoginsDisabled = true;
  1189. }
  1190. }
  1191. }
  1192. catch (NotImplementedException)
  1193. {
  1194. throw;
  1195. }
  1196. catch (AccessViolationException e)
  1197. {
  1198. m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName);
  1199. }
  1200. //catch (NullReferenceException e)
  1201. //{
  1202. // m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName);
  1203. //}
  1204. catch (InvalidOperationException e)
  1205. {
  1206. m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName);
  1207. }
  1208. catch (Exception e)
  1209. {
  1210. m_log.Error("[REGION]: Failed with exception " + e.ToString() + " On Region: " + RegionInfo.RegionName);
  1211. }
  1212. finally
  1213. {
  1214. m_lastupdate = DateTime.UtcNow;
  1215. }
  1216. maintc = Util.EnvironmentTickCountSubtract(maintc);
  1217. maintc = (int)(m_timespan * 1000) - maintc;
  1218. if (maintc > 0)
  1219. Thread.Sleep(maintc);
  1220. // Tell the watchdog that this thread is still alive
  1221. Watchdog.UpdateThread();
  1222. }
  1223. public void AddGroupTarget(SceneObjectGroup grp)
  1224. {
  1225. lock (m_groupsWithTargets)
  1226. m_groupsWithTargets[grp.UUID] = grp;
  1227. }
  1228. public void RemoveGroupTarget(SceneObjectGroup grp)
  1229. {
  1230. lock (m_groupsWithTargets)
  1231. m_groupsWithTargets.Remove(grp.UUID);
  1232. }
  1233. private void CheckAtTargets()
  1234. {
  1235. Dictionary<UUID, SceneObjectGroup>.ValueCollection objs;
  1236. lock (m_groupsWithTargets)
  1237. objs = m_groupsWithTargets.Values;
  1238. foreach (SceneObjectGroup entry in objs)
  1239. entry.checkAtTargets();
  1240. }
  1241. /// <summary>
  1242. /// Send out simstats data to all clients
  1243. /// </summary>
  1244. /// <param name="stats">Stats on the Simulator's performance</param>
  1245. private void SendSimStatsPackets(SimStats stats)
  1246. {
  1247. ForEachScenePresence(
  1248. delegate(ScenePresence agent)
  1249. {
  1250. if (!agent.IsChildAgent)
  1251. agent.ControllingClient.SendSimStats(stats);
  1252. }
  1253. );
  1254. }
  1255. /// <summary>
  1256. /// Update the terrain if it needs to be updated.
  1257. /// </summary>
  1258. private void UpdateTerrain()
  1259. {
  1260. EventManager.TriggerTerrainTick();
  1261. }
  1262. /// <summary>
  1263. /// Back up queued up changes
  1264. /// </summary>
  1265. private void UpdateStorageBackup()
  1266. {
  1267. if (!m_backingup)
  1268. {
  1269. m_backingup = true;
  1270. Util.FireAndForget(BackupWaitCallback);
  1271. }
  1272. }
  1273. /// <summary>
  1274. /// Sends out the OnFrame event to the modules
  1275. /// </summary>
  1276. private void UpdateEvents()
  1277. {
  1278. m_eventManager.TriggerOnFrame();
  1279. }
  1280. /// <summary>
  1281. /// Wrapper for Backup() that can be called with Util.FireAndForget()
  1282. /// </summary>
  1283. private void BackupWaitCallback(object o)
  1284. {
  1285. Backup(false);
  1286. }
  1287. /// <summary>
  1288. /// Backup the scene. This acts as the main method of the backup thread.
  1289. /// </summary>
  1290. /// <param name="forced">
  1291. /// If true, then any changes that have not yet been persisted are persisted. If false,
  1292. /// then the persistence decision is left to the backup code (in some situations, such as object persistence,
  1293. /// it's much more efficient to backup multiple changes at once rather than every single one).
  1294. /// <returns></returns>
  1295. public void Backup(bool forced)
  1296. {
  1297. lock (m_returns)
  1298. {
  1299. EventManager.TriggerOnBackup(SimulationDataService, forced);
  1300. m_backingup = false;
  1301. foreach (KeyValuePair<UUID, ReturnInfo> ret in m_returns)
  1302. {
  1303. UUID transaction = UUID.Random();
  1304. GridInstantMessage msg = new GridInstantMessage();
  1305. msg.fromAgentID = new Guid(UUID.Zero.ToString()); // From server
  1306. msg.toAgentID = new Guid(ret.Key.ToString());
  1307. msg.imSessionID = new Guid(transaction.ToString());
  1308. msg.timestamp = (uint)Util.UnixTimeSinceEpoch();
  1309. msg.fromAgentName = "Server";
  1310. msg.dialog = (byte)19; // Object msg
  1311. msg.fromGroup = false;
  1312. msg.offline = (byte)1;
  1313. msg.ParentEstateID = RegionInfo.EstateSettings.ParentEstateID;
  1314. msg.Position = Vector3.Zero;
  1315. msg.RegionID = RegionInfo.RegionID.Guid;
  1316. msg.binaryBucket = new byte[0];
  1317. if (ret.Value.count > 1)
  1318. msg.message = string.Format("Your {0} objects were returned from {1} in region {2} due to {3}", ret.Value.count, ret.Value.location.ToString(), RegionInfo.RegionName, ret.Value.reason);
  1319. else
  1320. msg.message = string.Format("Your object {0} was returned from {1} in region {2} due to {3}", ret.Value.objectName, ret.Value.location.ToString(), RegionInfo.RegionName, ret.Value.reason);
  1321. IMessageTransferModule tr = RequestModuleInterface<IMessageTransferModule>();
  1322. if (tr != null)
  1323. tr.SendInstantMessage(msg, delegate(bool success) {});
  1324. }
  1325. m_returns.Clear();
  1326. }
  1327. }
  1328. /// <summary>
  1329. /// Synchronous force backup. For deletes and links/unlinks
  1330. /// </summary>
  1331. /// <param name="group">Object to be backed up</param>
  1332. public void ForceSceneObjectBackup(SceneObjectGroup group)
  1333. {
  1334. if (group != null)
  1335. {
  1336. group.ProcessBackup(SimulationDataService, true);
  1337. }
  1338. }
  1339. /// <summary>
  1340. /// Tell an agent that their object has been returned.
  1341. /// </summary>
  1342. /// <remarks>
  1343. /// The actual return is handled by the caller.
  1344. /// </remarks>
  1345. /// <param name="agentID">Avatar Unique Id</param>
  1346. /// <param name="objectName">Name of object returned</param>
  1347. /// <param name="location">Location of object returned</param>
  1348. /// <param name="reason">Reasion for object return</param>
  1349. public void AddReturn(UUID agentID, string objectName, Vector3 location, string reason)
  1350. {
  1351. lock (m_returns)
  1352. {
  1353. if (m_returns.ContainsKey(agentID))
  1354. {
  1355. ReturnInfo info = m_returns[agentID];
  1356. info.count++;
  1357. m_returns[agentID] = info;
  1358. }
  1359. else
  1360. {
  1361. ReturnInfo info = new ReturnInfo();
  1362. info.count = 1;
  1363. info.objectName = objectName;
  1364. info.location = location;
  1365. info.reason = reason;
  1366. m_returns[agentID] = info;
  1367. }
  1368. }
  1369. }
  1370. #endregion
  1371. #region Load Terrain
  1372. /// <summary>
  1373. /// Store the terrain in the persistant data store
  1374. /// </summary>
  1375. public void SaveTerrain()
  1376. {
  1377. SimulationDataService.StoreTerrain(Heightmap.GetDoubles(), RegionInfo.RegionID);
  1378. }
  1379. public void StoreWindlightProfile(RegionLightShareData wl)
  1380. {
  1381. m_regInfo.WindlightSettings = wl;
  1382. SimulationDataService.StoreRegionWindlightSettings(wl);
  1383. m_eventManager.TriggerOnSaveNewWindlightProfile();
  1384. }
  1385. public void LoadWindlightProfile()
  1386. {
  1387. m_regInfo.WindlightSettings = SimulationDataService.LoadRegionWindlightSettings(RegionInfo.RegionID);
  1388. m_eventManager.TriggerOnSaveNewWindlightProfile();
  1389. }
  1390. /// <summary>
  1391. /// Loads the World heightmap
  1392. /// </summary>
  1393. public override void LoadWorldMap()
  1394. {
  1395. try
  1396. {
  1397. double[,] map = SimulationDataService.LoadTerrain(RegionInfo.RegionID);
  1398. if (map == null)
  1399. {
  1400. m_log.Info("[TERRAIN]: No default terrain. Generating a new terrain.");
  1401. Heightmap = new TerrainChannel();
  1402. SimulationDataService.StoreTerrain(Heightmap.GetDoubles(), RegionInfo.RegionID);
  1403. }
  1404. else
  1405. {
  1406. Heightmap = new TerrainChannel(map);
  1407. }
  1408. }
  1409. catch (IOException e)
  1410. {
  1411. m_log.Warn("[TERRAIN]: Scene.cs: LoadWorldMap() - Failed with exception " + e.ToString() + " Regenerating");
  1412. // Non standard region size. If there's an old terrain in the database, it might read past the buffer
  1413. #pragma warning disable 0162
  1414. if ((int)Constants.RegionSize != 256)
  1415. {
  1416. Heightmap = new TerrainChannel();
  1417. SimulationDataService.StoreTerrain(Heightmap.GetDoubles(), RegionInfo.RegionID);
  1418. }
  1419. }
  1420. catch (Exception e)
  1421. {
  1422. m_log.Warn("[TERRAIN]: Scene.cs: LoadWorldMap() - Failed with exception " + e.ToString());
  1423. }
  1424. }
  1425. /// <summary>
  1426. /// Register this region with a grid service
  1427. /// </summary>
  1428. /// <exception cref="System.Exception">Thrown if registration of the region itself fails.</exception>
  1429. public void RegisterRegionWithGrid()
  1430. {
  1431. RegisterCommsEvents();
  1432. m_sceneGridService.SetScene(this);
  1433. GridRegion region = new GridRegion(RegionInfo);
  1434. string error = GridService.RegisterRegion(RegionInfo.ScopeID, region);
  1435. if (error != String.Empty)
  1436. {
  1437. throw new Exception(error);
  1438. }
  1439. // Generate the maptile asynchronously, because sometimes it can be very slow and we
  1440. // don't want this to delay starting the region.
  1441. if (m_generateMaptiles)
  1442. {
  1443. Util.FireAndForget(delegate {
  1444. RegenerateMaptile(null, null);
  1445. });
  1446. }
  1447. }
  1448. #endregion
  1449. #region Load Land
  1450. /// <summary>
  1451. /// Loads all Parcel data from the datastore for region identified by regionID
  1452. /// </summary>
  1453. /// <param name="regionID">Unique Identifier of the Region to load parcel data for</param>
  1454. public void loadAllLandObjectsFromStorage(UUID regionID)
  1455. {
  1456. m_log.Info("[SCENE]: Loading land objects from storage");
  1457. List<LandData> landData = SimulationDataService.LoadLandObjects(regionID);
  1458. if (LandChannel != null)
  1459. {
  1460. if (landData.Count == 0)
  1461. {
  1462. EventManager.TriggerNoticeNoLandDataFromStorage();
  1463. }
  1464. else
  1465. {
  1466. EventManager.TriggerIncomingLandDataFromStorage(landData);
  1467. }
  1468. }
  1469. else
  1470. {
  1471. m_log.Error("[SCENE]: Land Channel is not defined. Cannot load from storage!");
  1472. }
  1473. }
  1474. #endregion
  1475. #region Primitives Methods
  1476. /// <summary>
  1477. /// Loads the World's objects
  1478. /// </summary>
  1479. /// <param name="regionID"></param>
  1480. public virtual void LoadPrimsFromStorage(UUID regionID)
  1481. {
  1482. LoadingPrims = true;
  1483. m_log.Info("[SCENE]: Loading objects from datastore");
  1484. List<SceneObjectGroup> PrimsFromDB = SimulationDataService.LoadObjects(regionID);
  1485. m_log.Info("[SCENE]: Loaded " + PrimsFromDB.Count + " objects from the datastore");
  1486. foreach (SceneObjectGroup group in PrimsFromDB)
  1487. {
  1488. EventManager.TriggerOnSceneObjectLoaded(group);
  1489. if (group.RootPart == null)
  1490. {
  1491. m_log.ErrorFormat(
  1492. "[SCENE]: Found a SceneObjectGroup with m_rootPart == null and {0} children",
  1493. group.Parts == null ? 0 : group.PrimCount);
  1494. }
  1495. AddRestoredSceneObject(group, true, true);
  1496. SceneObjectPart rootPart = group.GetChildPart(group.UUID);
  1497. rootPart.Flags &= ~PrimFlags.Scripted;
  1498. rootPart.TrimPermissions();
  1499. // Don't do this here - it will get done later on when sculpt data is loaded.
  1500. // group.CheckSculptAndLoad();
  1501. }
  1502. m_log.Info("[SCENE]: Loaded " + PrimsFromDB.Count.ToString() + " SceneObject(s)");
  1503. LoadingPrims = false;
  1504. EventManager.TriggerPrimsLoaded(this);
  1505. }
  1506. /// <summary>
  1507. /// Gets a new rez location based on the raycast and the size of the object that is being rezzed.
  1508. /// </summary>
  1509. /// <param name="RayStart"></param>
  1510. /// <param name="RayEnd"></param>
  1511. /// <param name="RayTargetID"></param>
  1512. /// <param name="rot"></param>
  1513. /// <param name="bypassRayCast"></param>
  1514. /// <param name="RayEndIsIntersection"></param>
  1515. /// <param name="frontFacesOnly"></param>
  1516. /// <param name="scale"></param>
  1517. /// <param name="FaceCenter"></param>
  1518. /// <returns></returns>
  1519. public Vector3 GetNewRezLocation(Vector3 RayStart, Vector3 RayEnd, UUID RayTargetID, Quaternion rot, byte bypassRayCast, byte RayEndIsIntersection, bool frontFacesOnly, Vector3 scale, bool FaceCenter)
  1520. {
  1521. Vector3 pos = Vector3.Zero;
  1522. if (RayEndIsIntersection == (byte)1)
  1523. {
  1524. pos = RayEnd;
  1525. return pos;
  1526. }
  1527. if (RayTargetID != UUID.Zero)
  1528. {
  1529. SceneObjectPart target = GetSceneObjectPart(RayTargetID);
  1530. Vector3 direction = Vector3.Normalize(RayEnd - RayStart);
  1531. Vector3 AXOrigin = new Vector3(RayStart.X, RayStart.Y, RayStart.Z);
  1532. Vector3 AXdirection = new Vector3(direction.X, direction.Y, direction.Z);
  1533. if (target != null)
  1534. {
  1535. pos = target.AbsolutePosition;
  1536. //m_log.Info("[OBJECT_REZ]: TargetPos: " + pos.ToString() + ", RayStart: " + RayStart.ToString() + ", RayEnd: " + RayEnd.ToString() + ", Volume: " + Util.GetDistanceTo(RayStart,RayEnd).ToString() + ", mag1: " + Util.GetMagnitude(RayStart).ToString() + ", mag2: " + Util.GetMagnitude(RayEnd).ToString());
  1537. // TODO: Raytrace better here
  1538. //EntityIntersection ei = m_sceneGraph.GetClosestIntersectingPrim(new Ray(AXOrigin, AXdirection));
  1539. Ray NewRay = new Ray(AXOrigin, AXdirection);
  1540. // Ray Trace against target here
  1541. EntityIntersection ei = target.TestIntersectionOBB(NewRay, Quaternion.Identity, frontFacesOnly, FaceCenter);
  1542. // Un-comment out the following line to Get Raytrace results printed to the console.
  1543. // m_log.Info("[RAYTRACERESULTS]: Hit:" + ei.HitTF.ToString() + " Point: " + ei.ipoint.ToString() + " Normal: " + ei.normal.ToString());
  1544. float ScaleOffset = 0.5f;
  1545. // If we hit something
  1546. if (ei.HitTF)
  1547. {
  1548. Vector3 scaleComponent = new Vector3(ei.AAfaceNormal.X, ei.AAfaceNormal.Y, ei.AAfaceNormal.Z);
  1549. if (scaleComponent.X != 0) ScaleOffset = scale.X;
  1550. if (scaleComponent.Y != 0) ScaleOffset = scale.Y;
  1551. if (scaleComponent.Z != 0) ScaleOffset = scale.Z;
  1552. ScaleOffset = Math.Abs(ScaleOffset);
  1553. Vector3 intersectionpoint = new Vector3(ei.ipoint.X, ei.ipoint.Y, ei.ipoint.Z);
  1554. Vector3 normal = new Vector3(ei.normal.X, ei.normal.Y, ei.normal.Z);
  1555. // Set the position to the intersection point
  1556. Vector3 offset = (normal * (ScaleOffset / 2f));
  1557. pos = (intersectionpoint + offset);
  1558. //Seems to make no sense to do this as this call is used for rezzing from inventory as well, and with inventory items their size is not always 0.5f
  1559. //And in cases when we weren't rezzing from inventory we were re-adding the 0.25 straight after calling this method
  1560. // Un-offset the prim (it gets offset later by the consumer method)
  1561. //pos.Z -= 0.25F;
  1562. }
  1563. return pos;
  1564. }
  1565. else
  1566. {
  1567. // We don't have a target here, so we're going to raytrace all the objects in the scene.
  1568. EntityIntersection ei = m_sceneGraph.GetClosestIntersectingPrim(new Ray(AXOrigin, AXdirection), true, false);
  1569. // Un-comment the following line to print the raytrace results to the console.
  1570. //m_log.Info("[RAYTRACERESULTS]: Hit:" + ei.HitTF.ToString() + " Point: " + ei.ipoint.ToString() + " Normal: " + ei.normal.ToString());
  1571. if (ei.HitTF)
  1572. {
  1573. pos = new Vector3(ei.ipoint.X, ei.ipoint.Y, ei.ipoint.Z);
  1574. } else
  1575. {
  1576. // fall back to our stupid functionality
  1577. pos = RayEnd;
  1578. }
  1579. return pos;
  1580. }
  1581. }
  1582. else
  1583. {
  1584. // fall back to our stupid functionality
  1585. pos = RayEnd;
  1586. //increase height so its above the ground.
  1587. //should be getting the normal of the ground at the rez point and using that?
  1588. pos.Z += scale.Z / 2f;
  1589. return pos;
  1590. }
  1591. }
  1592. /// <summary>
  1593. /// Create a New SceneObjectGroup/Part by raycasting
  1594. /// </summary>
  1595. /// <param name="ownerID"></param>
  1596. /// <param name="groupID"></param>
  1597. /// <param name="RayEnd"></param>
  1598. /// <param name="rot"></param>
  1599. /// <param name="shape"></param>
  1600. /// <param name="bypassRaycast"></param>
  1601. /// <param name="RayStart"></param>
  1602. /// <param name="RayTargetID"></param>
  1603. /// <param name="RayEndIsIntersection"></param>
  1604. public virtual void AddNewPrim(UUID ownerID, UUID groupID, Vector3 RayEnd, Quaternion rot, PrimitiveBaseShape shape,
  1605. byte bypassRaycast, Vector3 RayStart, UUID RayTargetID,
  1606. byte RayEndIsIntersection)
  1607. {
  1608. Vector3 pos = GetNewRezLocation(RayStart, RayEnd, RayTargetID, rot, bypassRaycast, RayEndIsIntersection, true, new Vector3(0.5f, 0.5f, 0.5f), false);
  1609. if (Permissions.CanRezObject(1, ownerID, pos))
  1610. {
  1611. // rez ON the ground, not IN the ground
  1612. // pos.Z += 0.25F; The rez point should now be correct so that its not in the ground
  1613. AddNewPrim(ownerID, groupID, pos, rot, shape);
  1614. }
  1615. }
  1616. public virtual SceneObjectGroup AddNewPrim(
  1617. UUID ownerID, UUID groupID, Vector3 pos, Quaternion rot, PrimitiveBaseShape shape)
  1618. {
  1619. //m_log.DebugFormat(
  1620. // "[SCENE]: Scene.AddNewPrim() pcode {0} called for {1} in {2}", shape.PCode, ownerID, RegionInfo.RegionName);
  1621. SceneObjectGroup sceneObject = null;
  1622. // If an entity creator has been registered for this prim type then use that
  1623. if (m_entityCreators.ContainsKey((PCode)shape.PCode))
  1624. {
  1625. sceneObject = m_entityCreators[(PCode)shape.PCode].CreateEntity(ownerID, groupID, pos, rot, shape);
  1626. }
  1627. else
  1628. {
  1629. // Otherwise, use this default creation code;
  1630. sceneObject = new SceneObjectGroup(ownerID, pos, rot, shape);
  1631. AddNewSceneObject(sceneObject, true);
  1632. sceneObject.SetGroup(groupID, null);
  1633. }
  1634. IUserManagement uman = RequestModuleInterface<IUserManagement>();
  1635. if (uman != null)
  1636. sceneObject.RootPart.CreatorIdentification = uman.GetUserUUI(ownerID);
  1637. sceneObject.ScheduleGroupForFullUpdate();
  1638. return sceneObject;
  1639. }
  1640. /// <summary>
  1641. /// Add an object into the scene that has come from storage
  1642. /// </summary>
  1643. ///
  1644. /// <param name="sceneObject"></param>
  1645. /// <param name="attachToBackup">
  1646. /// If true, changes to the object will be reflected in its persisted data
  1647. /// If false, the persisted data will not be changed even if the object in the scene is changed
  1648. /// </param>
  1649. /// <param name="alreadyPersisted">
  1650. /// If true, we won't persist this object until it changes
  1651. /// If false, we'll persist this object immediately
  1652. /// </param>
  1653. /// <param name="sendClientUpdates">
  1654. /// If true, we send updates to the client to tell it about this object
  1655. /// If false, we leave it up to the caller to do this
  1656. /// </param>
  1657. /// <returns>
  1658. /// true if the object was added, false if an object with the same uuid was already in the scene
  1659. /// </returns>
  1660. public bool AddRestoredSceneObject(
  1661. SceneObjectGroup sceneObject, bool attachToBackup, bool alreadyPersisted, bool sendClientUpdates)
  1662. {
  1663. return m_sceneGraph.AddRestoredSceneObject(sceneObject, attachToBackup, alreadyPersisted, sendClientUpdates);
  1664. }
  1665. /// <summary>
  1666. /// Add an object into the scene that has come from storage
  1667. /// </summary>
  1668. ///
  1669. /// <param name="sceneObject"></param>
  1670. /// <param name="attachToBackup">
  1671. /// If true, changes to the object will be reflected in its persisted data
  1672. /// If false, the persisted data will not be changed even if the object in the scene is changed
  1673. /// </param>
  1674. /// <param name="alreadyPersisted">
  1675. /// If true, we won't persist this object until it changes
  1676. /// If false, we'll persist this object immediately
  1677. /// </param>
  1678. /// <returns>
  1679. /// true if the object was added, false if an object with the same uuid was already in the scene
  1680. /// </returns>
  1681. public bool AddRestoredSceneObject(
  1682. SceneObjectGroup sceneObject, bool attachToBackup, bool alreadyPersisted)
  1683. {
  1684. return AddRestoredSceneObject(sceneObject, attachToBackup, alreadyPersisted, true);
  1685. }
  1686. /// <summary>
  1687. /// Add a newly created object to the scene. Updates are also sent to viewers.
  1688. /// </summary>
  1689. /// <param name="sceneObject"></param>
  1690. /// <param name="attachToBackup">
  1691. /// If true, the object is made persistent into the scene.
  1692. /// If false, the object will not persist over server restarts
  1693. /// </param>
  1694. public bool AddNewSceneObject(SceneObjectGroup sceneObject, bool attachToBackup)
  1695. {
  1696. return AddNewSceneObject(sceneObject, attachToBackup, true);
  1697. }
  1698. /// <summary>
  1699. /// Add a newly created object to the scene
  1700. /// </summary>
  1701. /// <param name="sceneObject"></param>
  1702. /// <param name="attachToBackup">
  1703. /// If true, the object is made persistent into the scene.
  1704. /// If false, the object will not persist over server restarts
  1705. /// </param>
  1706. /// <param name="sendClientUpdates">
  1707. /// If true, updates for the new scene object are sent to all viewers in range.
  1708. /// If false, it is left to the caller to schedule the update
  1709. /// </param>
  1710. public bool AddNewSceneObject(SceneObjectGroup sceneObject, bool attachToBackup, bool sendClientUpdates)
  1711. {
  1712. if (m_sceneGraph.AddNewSceneObject(sceneObject, attachToBackup, sendClientUpdates))
  1713. {
  1714. EventManager.TriggerObjectAddedToScene(sceneObject);
  1715. return true;
  1716. }
  1717. return false;
  1718. }
  1719. /// <summary>
  1720. /// Add a newly created object to the scene.
  1721. /// </summary>
  1722. /// <remarks>
  1723. /// This method does not send updates to the client - callers need to handle this themselves.
  1724. /// </remarks>
  1725. /// <param name="sceneObject"></param>
  1726. /// <param name="attachToBackup"></param>
  1727. /// <param name="pos">Position of the object. If null then the position stored in the object is used.</param>
  1728. /// <param name="rot">Rotation of the object. If null then the rotation stored in the object is used.</param>
  1729. /// <param name="vel">Velocity of the object. This parameter only has an effect if the object is physical</param>
  1730. /// <returns></returns>
  1731. public bool AddNewSceneObject(
  1732. SceneObjectGroup sceneObject, bool attachToBackup, Vector3? pos, Quaternion? rot, Vector3 vel)
  1733. {
  1734. if (m_sceneGraph.AddNewSceneObject(sceneObject, attachToBackup, pos, rot, vel))
  1735. {
  1736. EventManager.TriggerObjectAddedToScene(sceneObject);
  1737. return true;
  1738. }
  1739. return false;
  1740. }
  1741. /// <summary>
  1742. /// Delete every object from the scene. This does not include attachments worn by avatars.
  1743. /// </summary>
  1744. public void DeleteAllSceneObjects()
  1745. {
  1746. lock (Entities)
  1747. {
  1748. EntityBase[] entities = Entities.GetEntities();
  1749. foreach (EntityBase e in entities)
  1750. {
  1751. if (e is SceneObjectGroup)
  1752. {
  1753. SceneObjectGroup sog = (SceneObjectGroup)e;
  1754. if (!sog.IsAttachment)
  1755. DeleteSceneObject((SceneObjectGroup)e, false);
  1756. }
  1757. }
  1758. }
  1759. }
  1760. /// <summary>
  1761. /// Synchronously delete the given object from the scene.
  1762. /// </summary>
  1763. /// <param name="group">Object Id</param>
  1764. /// <param name="silent">Suppress broadcasting changes to other clients.</param>
  1765. public void DeleteSceneObject(SceneObjectGroup group, bool silent)
  1766. {
  1767. // m_log.DebugFormat("[SCENE]: Deleting scene object {0} {1}", group.Name, group.UUID);
  1768. //SceneObjectPart rootPart = group.GetChildPart(group.UUID);
  1769. // Serialise calls to RemoveScriptInstances to avoid
  1770. // deadlocking on m_parts inside SceneObjectGroup
  1771. lock (m_deleting_scene_object)
  1772. {
  1773. group.RemoveScriptInstances(true);
  1774. }
  1775. SceneObjectPart[] partList = group.Parts;
  1776. foreach (SceneObjectPart part in partList)
  1777. {
  1778. if (part.IsJoint() && ((part.Flags & PrimFlags.Physics) != 0))
  1779. {
  1780. PhysicsScene.RequestJointDeletion(part.Name); // FIXME: what if the name changed?
  1781. }
  1782. else if (part.PhysActor != null)
  1783. {
  1784. PhysicsScene.RemovePrim(part.PhysActor);
  1785. part.PhysActor = null;
  1786. }
  1787. }
  1788. // if (rootPart.PhysActor != null)
  1789. // {
  1790. // PhysicsScene.RemovePrim(rootPart.PhysActor);
  1791. // rootPart.PhysActor = null;
  1792. // }
  1793. if (UnlinkSceneObject(group, false))
  1794. {
  1795. EventManager.TriggerObjectBeingRemovedFromScene(group);
  1796. EventManager.TriggerParcelPrimCountTainted();
  1797. }
  1798. group.DeleteGroupFromScene(silent);
  1799. // m_log.DebugFormat("[SCENE]: Exit DeleteSceneObject() for {0} {1}", group.Name, group.UUID);
  1800. }
  1801. /// <summary>
  1802. /// Unlink the given object from the scene. Unlike delete, this just removes the record of the object - the
  1803. /// object itself is not destroyed.
  1804. /// </summary>
  1805. /// <param name="so">The scene object.</param>
  1806. /// <param name="softDelete">If true, only deletes from scene, but keeps the object in the database.</param>
  1807. /// <returns>true if the object was in the scene, false if it was not</returns>
  1808. public bool UnlinkSceneObject(SceneObjectGroup so, bool softDelete)
  1809. {
  1810. if (m_sceneGraph.DeleteSceneObject(so.UUID, softDelete))
  1811. {
  1812. if (!softDelete)
  1813. {
  1814. // Force a database update so that the scene object group ID is accurate. It's possible that the
  1815. // group has recently been delinked from another group but that this change has not been persisted
  1816. // to the DB.
  1817. // This is an expensive thing to do so only do it if absolutely necessary.
  1818. if (so.HasGroupChangedDueToDelink)
  1819. ForceSceneObjectBackup(so);
  1820. so.DetachFromBackup();
  1821. SimulationDataService.RemoveObject(so.UUID, m_regInfo.RegionID);
  1822. }
  1823. // We need to keep track of this state in case this group is still queued for further backup.
  1824. so.IsDeleted = true;
  1825. return true;
  1826. }
  1827. return false;
  1828. }
  1829. /// <summary>
  1830. /// Move the given scene object into a new region depending on which region its absolute position has moved
  1831. /// into.
  1832. ///
  1833. /// </summary>
  1834. /// <param name="attemptedPosition">the attempted out of region position of the scene object</param>
  1835. /// <param name="grp">the scene object that we're crossing</param>
  1836. public void CrossPrimGroupIntoNewRegion(Vector3 attemptedPosition, SceneObjectGroup grp, bool silent)
  1837. {
  1838. if (grp == null)
  1839. return;
  1840. if (grp.IsDeleted)
  1841. return;
  1842. if (grp.RootPart.DIE_AT_EDGE)
  1843. {
  1844. // We remove the object here
  1845. try
  1846. {
  1847. DeleteSceneObject(grp, false);
  1848. }
  1849. catch (Exception)
  1850. {
  1851. m_log.Warn("[SCENE]: exception when trying to remove the prim that crossed the border.");
  1852. }
  1853. return;
  1854. }
  1855. if (grp.RootPart.RETURN_AT_EDGE)
  1856. {
  1857. // We remove the object here
  1858. try
  1859. {
  1860. List<SceneObjectGroup> objects = new List<SceneObjectGroup>();
  1861. objects.Add(grp);
  1862. SceneObjectGroup[] objectsArray = objects.ToArray();
  1863. returnObjects(objectsArray, UUID.Zero);
  1864. }
  1865. catch (Exception)
  1866. {
  1867. m_log.Warn("[SCENE]: exception when trying to return the prim that crossed the border.");
  1868. }
  1869. return;
  1870. }
  1871. if (m_teleportModule != null)
  1872. m_teleportModule.Cross(grp, attemptedPosition, silent);
  1873. }
  1874. public Border GetCrossedBorder(Vector3 position, Cardinals gridline)
  1875. {
  1876. if (BordersLocked)
  1877. {
  1878. switch (gridline)
  1879. {
  1880. case Cardinals.N:
  1881. lock (NorthBorders)
  1882. {
  1883. foreach (Border b in NorthBorders)
  1884. {
  1885. if (b.TestCross(position))
  1886. return b;
  1887. }
  1888. }
  1889. break;
  1890. case Cardinals.S:
  1891. lock (SouthBorders)
  1892. {
  1893. foreach (Border b in SouthBorders)
  1894. {
  1895. if (b.TestCross(position))
  1896. return b;
  1897. }
  1898. }
  1899. break;
  1900. case Cardinals.E:
  1901. lock (EastBorders)
  1902. {
  1903. foreach (Border b in EastBorders)
  1904. {
  1905. if (b.TestCross(position))
  1906. return b;
  1907. }
  1908. }
  1909. break;
  1910. case Cardinals.W:
  1911. lock (WestBorders)
  1912. {
  1913. foreach (Border b in WestBorders)
  1914. {
  1915. if (b.TestCross(position))
  1916. return b;
  1917. }
  1918. }
  1919. break;
  1920. }
  1921. }
  1922. else
  1923. {
  1924. switch (gridline)
  1925. {
  1926. case Cardinals.N:
  1927. foreach (Border b in NorthBorders)
  1928. {
  1929. if (b.TestCross(position))
  1930. return b;
  1931. }
  1932. break;
  1933. case Cardinals.S:
  1934. foreach (Border b in SouthBorders)
  1935. {
  1936. if (b.TestCross(position))
  1937. return b;
  1938. }
  1939. break;
  1940. case Cardinals.E:
  1941. foreach (Border b in EastBorders)
  1942. {
  1943. if (b.TestCross(position))
  1944. return b;
  1945. }
  1946. break;
  1947. case Cardinals.W:
  1948. foreach (Border b in WestBorders)
  1949. {
  1950. if (b.TestCross(position))
  1951. return b;
  1952. }
  1953. break;
  1954. }
  1955. }
  1956. return null;
  1957. }
  1958. public bool TestBorderCross(Vector3 position, Cardinals border)
  1959. {
  1960. if (BordersLocked)
  1961. {
  1962. switch (border)
  1963. {
  1964. case Cardinals.N:
  1965. lock (NorthBorders)
  1966. {
  1967. foreach (Border b in NorthBorders)
  1968. {
  1969. if (b.TestCross(position))
  1970. return true;
  1971. }
  1972. }
  1973. break;
  1974. case Cardinals.E:
  1975. lock (EastBorders)
  1976. {
  1977. foreach (Border b in EastBorders)
  1978. {
  1979. if (b.TestCross(position))
  1980. return true;
  1981. }
  1982. }
  1983. break;
  1984. case Cardinals.S:
  1985. lock (SouthBorders)
  1986. {
  1987. foreach (Border b in SouthBorders)
  1988. {
  1989. if (b.TestCross(position))
  1990. return true;
  1991. }
  1992. }
  1993. break;
  1994. case Cardinals.W:
  1995. lock (WestBorders)
  1996. {
  1997. foreach (Border b in WestBorders)
  1998. {
  1999. if (b.TestCross(position))
  2000. return true;
  2001. }
  2002. }
  2003. break;
  2004. }
  2005. }
  2006. else
  2007. {
  2008. switch (border)
  2009. {
  2010. case Cardinals.N:
  2011. foreach (Border b in NorthBorders)
  2012. {
  2013. if (b.TestCross(position))
  2014. return true;
  2015. }
  2016. break;
  2017. case Cardinals.E:
  2018. foreach (Border b in EastBorders)
  2019. {
  2020. if (b.TestCross(position))
  2021. return true;
  2022. }
  2023. break;
  2024. case Cardinals.S:
  2025. foreach (Border b in SouthBorders)
  2026. {
  2027. if (b.TestCross(position))
  2028. return true;
  2029. }
  2030. break;
  2031. case Cardinals.W:
  2032. foreach (Border b in WestBorders)
  2033. {
  2034. if (b.TestCross(position))
  2035. return true;
  2036. }
  2037. break;
  2038. }
  2039. }
  2040. return false;
  2041. }
  2042. /// <summary>
  2043. /// Called when objects or attachments cross the border, or teleport, between regions.
  2044. /// </summary>
  2045. /// <param name="sog"></param>
  2046. /// <returns></returns>
  2047. public bool IncomingCreateObject(ISceneObject sog)
  2048. {
  2049. //m_log.DebugFormat(" >>> IncomingCreateObject(sog) <<< {0} deleted? {1} isAttach? {2}", ((SceneObjectGroup)sog).AbsolutePosition,
  2050. // ((SceneObjectGroup)sog).IsDeleted, ((SceneObjectGroup)sog).RootPart.IsAttachment);
  2051. SceneObjectGroup newObject;
  2052. try
  2053. {
  2054. newObject = (SceneObjectGroup)sog;
  2055. }
  2056. catch (Exception e)
  2057. {
  2058. m_log.WarnFormat("[SCENE]: Problem casting object: " + e.ToString());
  2059. return false;
  2060. }
  2061. if (!AddSceneObject(newObject))
  2062. {
  2063. m_log.DebugFormat("[SCENE]: Problem adding scene object {0} in {1} ", sog.UUID, RegionInfo.RegionName);
  2064. return false;
  2065. }
  2066. // For attachments, we need to wait until the agent is root
  2067. // before we restart the scripts, or else some functions won't work.
  2068. if (!newObject.IsAttachment)
  2069. {
  2070. newObject.RootPart.ParentGroup.CreateScriptInstances(0, false, DefaultScriptEngine, GetStateSource(newObject));
  2071. newObject.ResumeScripts();
  2072. }
  2073. else
  2074. {
  2075. ScenePresence sp;
  2076. if (TryGetScenePresence(newObject.OwnerID, out sp))
  2077. {
  2078. // If the scene presence is here and already a root
  2079. // agent, we came from a ;egacy region. Start the scripts
  2080. // here as they used to start.
  2081. // TODO: Remove in 0.7.3
  2082. if (!sp.IsChildAgent)
  2083. {
  2084. newObject.RootPart.ParentGroup.CreateScriptInstances(0, false, DefaultScriptEngine, GetStateSource(newObject));
  2085. newObject.ResumeScripts();
  2086. }
  2087. }
  2088. }
  2089. // Do this as late as possible so that listeners have full access to the incoming object
  2090. EventManager.TriggerOnIncomingSceneObject(newObject);
  2091. return true;
  2092. }
  2093. /// <summary>
  2094. /// Attachment rezzing
  2095. /// </summary>
  2096. /// <param name="userID">Agent Unique ID</param>
  2097. /// <param name="itemID">Object ID</param>
  2098. /// <returns>False</returns>
  2099. public virtual bool IncomingCreateObject(UUID userID, UUID itemID)
  2100. {
  2101. //m_log.DebugFormat(" >>> IncomingCreateObject(userID, itemID) <<< {0} {1}", userID, itemID);
  2102. ScenePresence sp = GetScenePresence(userID);
  2103. if (sp != null && AttachmentsModule != null)
  2104. {
  2105. uint attPt = (uint)sp.Appearance.GetAttachpoint(itemID);
  2106. AttachmentsModule.RezSingleAttachmentFromInventory(sp.ControllingClient, itemID, attPt);
  2107. }
  2108. return false;
  2109. }
  2110. /// <summary>
  2111. /// Adds a Scene Object group to the Scene.
  2112. /// Verifies that the creator of the object is not banned from the simulator.
  2113. /// Checks if the item is an Attachment
  2114. /// </summary>
  2115. /// <param name="sceneObject"></param>
  2116. /// <returns>True if the SceneObjectGroup was added, False if it was not</returns>
  2117. public bool AddSceneObject(SceneObjectGroup sceneObject)
  2118. {
  2119. // If the user is banned, we won't let any of their objects
  2120. // enter. Period.
  2121. //
  2122. if (m_regInfo.EstateSettings.IsBanned(sceneObject.OwnerID))
  2123. {
  2124. m_log.Info("[INTERREGION]: Denied prim crossing for " +
  2125. "banned avatar");
  2126. return false;
  2127. }
  2128. sceneObject.SetScene(this);
  2129. // Force allocation of new LocalId
  2130. //
  2131. SceneObjectPart[] parts = sceneObject.Parts;
  2132. for (int i = 0; i < parts.Length; i++)
  2133. parts[i].LocalId = 0;
  2134. if (sceneObject.IsAttachmentCheckFull()) // Attachment
  2135. {
  2136. sceneObject.RootPart.AddFlag(PrimFlags.TemporaryOnRez);
  2137. sceneObject.RootPart.AddFlag(PrimFlags.Phantom);
  2138. // Don't sent a full update here because this will cause full updates to be sent twice for
  2139. // attachments on region crossings, resulting in viewer glitches.
  2140. AddRestoredSceneObject(sceneObject, false, false, false);
  2141. // Handle attachment special case
  2142. SceneObjectPart RootPrim = sceneObject.RootPart;
  2143. // Fix up attachment Parent Local ID
  2144. ScenePresence sp = GetScenePresence(sceneObject.OwnerID);
  2145. if (sp != null)
  2146. {
  2147. SceneObjectGroup grp = sceneObject;
  2148. m_log.DebugFormat(
  2149. "[ATTACHMENT]: Received attachment {0}, inworld asset id {1}", grp.GetFromItemID(), grp.UUID);
  2150. m_log.DebugFormat(
  2151. "[ATTACHMENT]: Attach to avatar {0} at position {1}", sp.UUID, grp.AbsolutePosition);
  2152. RootPrim.RemFlag(PrimFlags.TemporaryOnRez);
  2153. if (AttachmentsModule != null)
  2154. AttachmentsModule.AttachObject(sp.ControllingClient, grp, 0, false);
  2155. }
  2156. else
  2157. {
  2158. RootPrim.RemFlag(PrimFlags.TemporaryOnRez);
  2159. RootPrim.AddFlag(PrimFlags.TemporaryOnRez);
  2160. }
  2161. }
  2162. else
  2163. {
  2164. AddRestoredSceneObject(sceneObject, true, false);
  2165. if (!Permissions.CanObjectEntry(sceneObject.UUID,
  2166. true, sceneObject.AbsolutePosition))
  2167. {
  2168. // Deny non attachments based on parcel settings
  2169. //
  2170. m_log.Info("[INTERREGION]: Denied prim crossing " +
  2171. "because of parcel settings");
  2172. DeleteSceneObject(sceneObject, false);
  2173. return false;
  2174. }
  2175. }
  2176. return true;
  2177. }
  2178. private int GetStateSource(SceneObjectGroup sog)
  2179. {
  2180. ScenePresence sp = GetScenePresence(sog.OwnerID);
  2181. if (sp != null)
  2182. return sp.GetStateSource();
  2183. return 2; // StateSource.PrimCrossing
  2184. }
  2185. #endregion
  2186. #region Add/Remove Avatar Methods
  2187. /// <summary>
  2188. /// Add a new client and create a child agent for it.
  2189. /// </summary>
  2190. /// <param name="client"></param>
  2191. /// <param name="type">The type of agent to add.</param>
  2192. public override void AddNewClient(IClientAPI client, PresenceType type)
  2193. {
  2194. AgentCircuitData aCircuit = m_authenticateHandler.GetAgentCircuitData(client.CircuitCode);
  2195. bool vialogin = false;
  2196. if (aCircuit == null) // no good, didn't pass NewUserConnection successfully
  2197. return;
  2198. vialogin = (aCircuit.teleportFlags & (uint)Constants.TeleportFlags.ViaHGLogin) != 0 ||
  2199. (aCircuit.teleportFlags & (uint)Constants.TeleportFlags.ViaLogin) != 0;
  2200. CheckHeartbeat();
  2201. if (GetScenePresence(client.AgentId) == null) // ensure there is no SP here
  2202. {
  2203. m_log.Debug("[SCENE]: Adding new agent " + client.Name + " to scene " + RegionInfo.RegionName);
  2204. m_clientManager.Add(client);
  2205. SubscribeToClientEvents(client);
  2206. ScenePresence sp = m_sceneGraph.CreateAndAddChildScenePresence(client, aCircuit.Appearance, type);
  2207. m_eventManager.TriggerOnNewPresence(sp);
  2208. sp.TeleportFlags = (TeleportFlags)aCircuit.teleportFlags;
  2209. // HERE!!! Do the initial attachments right here
  2210. // first agent upon login is a root agent by design.
  2211. // All other AddNewClient calls find aCircuit.child to be true
  2212. if (aCircuit.child == false)
  2213. {
  2214. sp.IsChildAgent = false;
  2215. Util.FireAndForget(delegate(object o) { sp.RezAttachments(); });
  2216. }
  2217. }
  2218. ScenePresence createdSp = GetScenePresence(client.AgentId);
  2219. if (createdSp != null)
  2220. {
  2221. m_LastLogin = Util.EnvironmentTickCount();
  2222. // Cache the user's name
  2223. CacheUserName(createdSp, aCircuit);
  2224. EventManager.TriggerOnNewClient(client);
  2225. if (vialogin)
  2226. EventManager.TriggerOnClientLogin(client);
  2227. }
  2228. }
  2229. /// <summary>
  2230. /// Cache the user name for later use.
  2231. /// </summary>
  2232. /// <param name="sp"></param>
  2233. /// <param name="aCircuit"></param>
  2234. private void CacheUserName(ScenePresence sp, AgentCircuitData aCircuit)
  2235. {
  2236. IUserManagement uMan = RequestModuleInterface<IUserManagement>();
  2237. if (uMan != null)
  2238. {
  2239. string first = aCircuit.firstname, last = aCircuit.lastname;
  2240. if (sp.PresenceType == PresenceType.Npc)
  2241. {
  2242. uMan.AddUser(aCircuit.AgentID, first, last);
  2243. }
  2244. else
  2245. {
  2246. string homeURL = string.Empty;
  2247. if (aCircuit.ServiceURLs.ContainsKey("HomeURI"))
  2248. homeURL = aCircuit.ServiceURLs["HomeURI"].ToString();
  2249. if (aCircuit.lastname.StartsWith("@"))
  2250. {
  2251. string[] parts = aCircuit.firstname.Split('.');
  2252. if (parts.Length >= 2)
  2253. {
  2254. first = parts[0];
  2255. last = parts[1];
  2256. }
  2257. }
  2258. uMan.AddUser(aCircuit.AgentID, first, last, homeURL);
  2259. }
  2260. }
  2261. }
  2262. private bool VerifyClient(AgentCircuitData aCircuit, System.Net.IPEndPoint ep, out bool vialogin)
  2263. {
  2264. vialogin = false;
  2265. // Do the verification here
  2266. if ((aCircuit.teleportFlags & (uint)Constants.TeleportFlags.ViaHGLogin) != 0)
  2267. {
  2268. m_log.DebugFormat("[SCENE]: Incoming client {0} {1} in region {2} via HG login", aCircuit.firstname, aCircuit.lastname, RegionInfo.RegionName);
  2269. vialogin = true;
  2270. IUserAgentVerificationModule userVerification = RequestModuleInterface<IUserAgentVerificationModule>();
  2271. if (userVerification != null && ep != null)
  2272. {
  2273. if (!userVerification.VerifyClient(aCircuit, ep.Address.ToString()))
  2274. {
  2275. // uh-oh, this is fishy
  2276. m_log.DebugFormat("[SCENE]: User Client Verification for {0} {1} in {2} returned false", aCircuit.firstname, aCircuit.lastname, RegionInfo.RegionName);
  2277. return false;
  2278. }
  2279. else
  2280. m_log.DebugFormat("[SCENE]: User Client Verification for {0} {1} in {2} returned true", aCircuit.firstname, aCircuit.lastname, RegionInfo.RegionName);
  2281. }
  2282. }
  2283. else if ((aCircuit.teleportFlags & (uint)Constants.TeleportFlags.ViaLogin) != 0)
  2284. {
  2285. m_log.DebugFormat("[SCENE]: Incoming client {0} {1} in region {2} via regular login. Client IP verification not performed.",
  2286. aCircuit.firstname, aCircuit.lastname, RegionInfo.RegionName);
  2287. vialogin = true;
  2288. }
  2289. return true;
  2290. }
  2291. // Called by Caps, on the first HTTP contact from the client
  2292. public override bool CheckClient(UUID agentID, System.Net.IPEndPoint ep)
  2293. {
  2294. AgentCircuitData aCircuit = m_authenticateHandler.GetAgentCircuitData(agentID);
  2295. if (aCircuit != null)
  2296. {
  2297. bool vialogin = false;
  2298. if (!VerifyClient(aCircuit, ep, out vialogin))
  2299. {
  2300. // if it doesn't pass, we remove the agentcircuitdata altogether
  2301. // and the scene presence and the client, if they exist
  2302. try
  2303. {
  2304. // We need to wait for the client to make UDP contact first.
  2305. // It's the UDP contact that creates the scene presence
  2306. ScenePresence sp = WaitGetScenePresence(agentID);
  2307. if (sp != null)
  2308. {
  2309. PresenceService.LogoutAgent(sp.ControllingClient.SessionId);
  2310. sp.ControllingClient.Close();
  2311. }
  2312. else
  2313. {
  2314. m_log.WarnFormat("[SCENE]: Could not find scene presence for {0}", agentID);
  2315. }
  2316. // BANG! SLASH!
  2317. m_authenticateHandler.RemoveCircuit(agentID);
  2318. return false;
  2319. }
  2320. catch (Exception e)
  2321. {
  2322. m_log.DebugFormat("[SCENE]: Exception while closing aborted client: {0}", e.StackTrace);
  2323. }
  2324. }
  2325. else
  2326. return true;
  2327. }
  2328. return false;
  2329. }
  2330. /// <summary>
  2331. /// Register for events from the client
  2332. /// </summary>
  2333. /// <param name="client">The IClientAPI of the connected client</param>
  2334. public virtual void SubscribeToClientEvents(IClientAPI client)
  2335. {
  2336. SubscribeToClientTerrainEvents(client);
  2337. SubscribeToClientPrimEvents(client);
  2338. SubscribeToClientPrimRezEvents(client);
  2339. SubscribeToClientInventoryEvents(client);
  2340. SubscribeToClientTeleportEvents(client);
  2341. SubscribeToClientScriptEvents(client);
  2342. SubscribeToClientParcelEvents(client);
  2343. SubscribeToClientGridEvents(client);
  2344. SubscribeToClientNetworkEvents(client);
  2345. }
  2346. public virtual void SubscribeToClientTerrainEvents(IClientAPI client)
  2347. {
  2348. client.OnRegionHandShakeReply += SendLayerData;
  2349. }
  2350. public virtual void SubscribeToClientPrimEvents(IClientAPI client)
  2351. {
  2352. client.OnUpdatePrimGroupPosition += m_sceneGraph.UpdatePrimGroupPosition;
  2353. client.OnUpdatePrimSinglePosition += m_sceneGraph.UpdatePrimSinglePosition;
  2354. client.OnUpdatePrimGroupRotation += m_sceneGraph.UpdatePrimGroupRotation;
  2355. client.OnUpdatePrimGroupMouseRotation += m_sceneGraph.UpdatePrimGroupRotation;
  2356. client.OnUpdatePrimSingleRotation += m_sceneGraph.UpdatePrimSingleRotation;
  2357. client.OnUpdatePrimSingleRotationPosition += m_sceneGraph.UpdatePrimSingleRotationPosition;
  2358. client.OnUpdatePrimScale += m_sceneGraph.UpdatePrimScale;
  2359. client.OnUpdatePrimGroupScale += m_sceneGraph.UpdatePrimGroupScale;
  2360. client.OnUpdateExtraParams += m_sceneGraph.UpdateExtraParam;
  2361. client.OnUpdatePrimShape += m_sceneGraph.UpdatePrimShape;
  2362. client.OnUpdatePrimTexture += m_sceneGraph.UpdatePrimTexture;
  2363. client.OnObjectRequest += RequestPrim;
  2364. client.OnObjectSelect += SelectPrim;
  2365. client.OnObjectDeselect += DeselectPrim;
  2366. client.OnGrabUpdate += m_sceneGraph.MoveObject;
  2367. client.OnSpinStart += m_sceneGraph.SpinStart;
  2368. client.OnSpinUpdate += m_sceneGraph.SpinObject;
  2369. client.OnDeRezObject += DeRezObjects;
  2370. client.OnObjectName += m_sceneGraph.PrimName;
  2371. client.OnObjectClickAction += m_sceneGraph.PrimClickAction;
  2372. client.OnObjectMaterial += m_sceneGraph.PrimMaterial;
  2373. client.OnLinkObjects += LinkObjects;
  2374. client.OnDelinkObjects += DelinkObjects;
  2375. client.OnObjectDuplicate += m_sceneGraph.DuplicateObject;
  2376. client.OnObjectDuplicateOnRay += doObjectDuplicateOnRay;
  2377. client.OnUpdatePrimFlags += m_sceneGraph.UpdatePrimFlags;
  2378. client.OnRequestObjectPropertiesFamily += m_sceneGraph.RequestObjectPropertiesFamily;
  2379. client.OnObjectPermissions += HandleObjectPermissionsUpdate;
  2380. client.OnGrabObject += ProcessObjectGrab;
  2381. client.OnGrabUpdate += ProcessObjectGrabUpdate;
  2382. client.OnDeGrabObject += ProcessObjectDeGrab;
  2383. client.OnUndo += m_sceneGraph.HandleUndo;
  2384. client.OnRedo += m_sceneGraph.HandleRedo;
  2385. client.OnObjectDescription += m_sceneGraph.PrimDescription;
  2386. client.OnObjectDrop += m_sceneGraph.DropObject;
  2387. client.OnObjectIncludeInSearch += m_sceneGraph.MakeObjectSearchable;
  2388. client.OnObjectOwner += ObjectOwner;
  2389. }
  2390. public virtual void SubscribeToClientPrimRezEvents(IClientAPI client)
  2391. {
  2392. client.OnAddPrim += AddNewPrim;
  2393. client.OnRezObject += RezObject;
  2394. }
  2395. public virtual void SubscribeToClientInventoryEvents(IClientAPI client)
  2396. {
  2397. client.OnLinkInventoryItem += HandleLinkInventoryItem;
  2398. client.OnCreateNewInventoryFolder += HandleCreateInventoryFolder;
  2399. client.OnUpdateInventoryFolder += HandleUpdateInventoryFolder;
  2400. client.OnMoveInventoryFolder += HandleMoveInventoryFolder; // 2; //!!
  2401. client.OnFetchInventoryDescendents += HandleFetchInventoryDescendents;
  2402. client.OnPurgeInventoryDescendents += HandlePurgeInventoryDescendents; // 2; //!!
  2403. client.OnFetchInventory += m_asyncInventorySender.HandleFetchInventory;
  2404. client.OnUpdateInventoryItem += UpdateInventoryItemAsset;
  2405. client.OnCopyInventoryItem += CopyInventoryItem;
  2406. client.OnMoveInventoryItem += MoveInventoryItem;
  2407. client.OnRemoveInventoryItem += RemoveInventoryItem;
  2408. client.OnRemoveInventoryFolder += RemoveInventoryFolder;
  2409. client.OnRezScript += RezScript;
  2410. client.OnRequestTaskInventory += RequestTaskInventory;
  2411. client.OnRemoveTaskItem += RemoveTaskInventory;
  2412. client.OnUpdateTaskInventory += UpdateTaskInventory;
  2413. client.OnMoveTaskItem += ClientMoveTaskInventoryItem;
  2414. }
  2415. public virtual void SubscribeToClientTeleportEvents(IClientAPI client)
  2416. {
  2417. client.OnTeleportLocationRequest += RequestTeleportLocation;
  2418. }
  2419. public virtual void SubscribeToClientScriptEvents(IClientAPI client)
  2420. {
  2421. client.OnScriptReset += ProcessScriptReset;
  2422. client.OnGetScriptRunning += GetScriptRunning;
  2423. client.OnSetScriptRunning += SetScriptRunning;
  2424. }
  2425. public virtual void SubscribeToClientParcelEvents(IClientAPI client)
  2426. {
  2427. client.OnObjectGroupRequest += m_sceneGraph.HandleObjectGroupUpdate;
  2428. client.OnParcelReturnObjectsRequest += LandChannel.ReturnObjectsInParcel;
  2429. client.OnParcelSetOtherCleanTime += LandChannel.SetParcelOtherCleanTime;
  2430. client.OnParcelBuy += ProcessParcelBuy;
  2431. }
  2432. public virtual void SubscribeToClientGridEvents(IClientAPI client)
  2433. {
  2434. //client.OnNameFromUUIDRequest += HandleUUIDNameRequest;
  2435. client.OnMoneyTransferRequest += ProcessMoneyTransferRequest;
  2436. client.OnAvatarPickerRequest += ProcessAvatarPickerRequest;
  2437. client.OnSetStartLocationRequest += SetHomeRezPoint;
  2438. client.OnRegionHandleRequest += RegionHandleRequest;
  2439. }
  2440. public virtual void SubscribeToClientNetworkEvents(IClientAPI client)
  2441. {
  2442. client.OnNetworkStatsUpdate += StatsReporter.AddPacketsStats;
  2443. client.OnViewerEffect += ProcessViewerEffect;
  2444. }
  2445. /// <summary>
  2446. /// Unsubscribe the client from events.
  2447. /// </summary>
  2448. /// FIXME: Not called anywhere!
  2449. /// <param name="client">The IClientAPI of the client</param>
  2450. public virtual void UnSubscribeToClientEvents(IClientAPI client)
  2451. {
  2452. UnSubscribeToClientTerrainEvents(client);
  2453. UnSubscribeToClientPrimEvents(client);
  2454. UnSubscribeToClientPrimRezEvents(client);
  2455. UnSubscribeToClientInventoryEvents(client);
  2456. UnSubscribeToClientTeleportEvents(client);
  2457. UnSubscribeToClientScriptEvents(client);
  2458. UnSubscribeToClientParcelEvents(client);
  2459. UnSubscribeToClientGridEvents(client);
  2460. UnSubscribeToClientNetworkEvents(client);
  2461. }
  2462. public virtual void UnSubscribeToClientTerrainEvents(IClientAPI client)
  2463. {
  2464. client.OnRegionHandShakeReply -= SendLayerData;
  2465. }
  2466. public virtual void UnSubscribeToClientPrimEvents(IClientAPI client)
  2467. {
  2468. client.OnUpdatePrimGroupPosition -= m_sceneGraph.UpdatePrimGroupPosition;
  2469. client.OnUpdatePrimSinglePosition -= m_sceneGraph.UpdatePrimSinglePosition;
  2470. client.OnUpdatePrimGroupRotation -= m_sceneGraph.UpdatePrimGroupRotation;
  2471. client.OnUpdatePrimGroupMouseRotation -= m_sceneGraph.UpdatePrimGroupRotation;
  2472. client.OnUpdatePrimSingleRotation -= m_sceneGraph.UpdatePrimSingleRotation;
  2473. client.OnUpdatePrimSingleRotationPosition -= m_sceneGraph.UpdatePrimSingleRotationPosition;
  2474. client.OnUpdatePrimScale -= m_sceneGraph.UpdatePrimScale;
  2475. client.OnUpdatePrimGroupScale -= m_sceneGraph.UpdatePrimGroupScale;
  2476. client.OnUpdateExtraParams -= m_sceneGraph.UpdateExtraParam;
  2477. client.OnUpdatePrimShape -= m_sceneGraph.UpdatePrimShape;
  2478. client.OnUpdatePrimTexture -= m_sceneGraph.UpdatePrimTexture;
  2479. client.OnObjectRequest -= RequestPrim;
  2480. client.OnObjectSelect -= SelectPrim;
  2481. client.OnObjectDeselect -= DeselectPrim;
  2482. client.OnGrabUpdate -= m_sceneGraph.MoveObject;
  2483. client.OnSpinStart -= m_sceneGraph.SpinStart;
  2484. client.OnSpinUpdate -= m_sceneGraph.SpinObject;
  2485. client.OnDeRezObject -= DeRezObjects;
  2486. client.OnObjectName -= m_sceneGraph.PrimName;
  2487. client.OnObjectClickAction -= m_sceneGraph.PrimClickAction;
  2488. client.OnObjectMaterial -= m_sceneGraph.PrimMaterial;
  2489. client.OnLinkObjects -= LinkObjects;
  2490. client.OnDelinkObjects -= DelinkObjects;
  2491. client.OnObjectDuplicate -= m_sceneGraph.DuplicateObject;
  2492. client.OnObjectDuplicateOnRay -= doObjectDuplicateOnRay;
  2493. client.OnUpdatePrimFlags -= m_sceneGraph.UpdatePrimFlags;
  2494. client.OnRequestObjectPropertiesFamily -= m_sceneGraph.RequestObjectPropertiesFamily;
  2495. client.OnObjectPermissions -= HandleObjectPermissionsUpdate;
  2496. client.OnGrabObject -= ProcessObjectGrab;
  2497. client.OnDeGrabObject -= ProcessObjectDeGrab;
  2498. client.OnUndo -= m_sceneGraph.HandleUndo;
  2499. client.OnRedo -= m_sceneGraph.HandleRedo;
  2500. client.OnObjectDescription -= m_sceneGraph.PrimDescription;
  2501. client.OnObjectDrop -= m_sceneGraph.DropObject;
  2502. client.OnObjectIncludeInSearch -= m_sceneGraph.MakeObjectSearchable;
  2503. client.OnObjectOwner -= ObjectOwner;
  2504. }
  2505. public virtual void UnSubscribeToClientPrimRezEvents(IClientAPI client)
  2506. {
  2507. client.OnAddPrim -= AddNewPrim;
  2508. client.OnRezObject -= RezObject;
  2509. }
  2510. public virtual void UnSubscribeToClientInventoryEvents(IClientAPI client)
  2511. {
  2512. client.OnCreateNewInventoryFolder -= HandleCreateInventoryFolder;
  2513. client.OnUpdateInventoryFolder -= HandleUpdateInventoryFolder;
  2514. client.OnMoveInventoryFolder -= HandleMoveInventoryFolder; // 2; //!!
  2515. client.OnFetchInventoryDescendents -= HandleFetchInventoryDescendents;
  2516. client.OnPurgeInventoryDescendents -= HandlePurgeInventoryDescendents; // 2; //!!
  2517. client.OnFetchInventory -= m_asyncInventorySender.HandleFetchInventory;
  2518. client.OnUpdateInventoryItem -= UpdateInventoryItemAsset;
  2519. client.OnCopyInventoryItem -= CopyInventoryItem;
  2520. client.OnMoveInventoryItem -= MoveInventoryItem;
  2521. client.OnRemoveInventoryItem -= RemoveInventoryItem;
  2522. client.OnRemoveInventoryFolder -= RemoveInventoryFolder;
  2523. client.OnRezScript -= RezScript;
  2524. client.OnRequestTaskInventory -= RequestTaskInventory;
  2525. client.OnRemoveTaskItem -= RemoveTaskInventory;
  2526. client.OnUpdateTaskInventory -= UpdateTaskInventory;
  2527. client.OnMoveTaskItem -= ClientMoveTaskInventoryItem;
  2528. }
  2529. public virtual void UnSubscribeToClientTeleportEvents(IClientAPI client)
  2530. {
  2531. client.OnTeleportLocationRequest -= RequestTeleportLocation;
  2532. //client.OnTeleportLandmarkRequest -= RequestTeleportLandmark;
  2533. //client.OnTeleportHomeRequest -= TeleportClientHome;
  2534. }
  2535. public virtual void UnSubscribeToClientScriptEvents(IClientAPI client)
  2536. {
  2537. client.OnScriptReset -= ProcessScriptReset;
  2538. client.OnGetScriptRunning -= GetScriptRunning;
  2539. client.OnSetScriptRunning -= SetScriptRunning;
  2540. }
  2541. public virtual void UnSubscribeToClientParcelEvents(IClientAPI client)
  2542. {
  2543. client.OnObjectGroupRequest -= m_sceneGraph.HandleObjectGroupUpdate;
  2544. client.OnParcelReturnObjectsRequest -= LandChannel.ReturnObjectsInParcel;
  2545. client.OnParcelSetOtherCleanTime -= LandChannel.SetParcelOtherCleanTime;
  2546. client.OnParcelBuy -= ProcessParcelBuy;
  2547. }
  2548. public virtual void UnSubscribeToClientGridEvents(IClientAPI client)
  2549. {
  2550. //client.OnNameFromUUIDRequest -= HandleUUIDNameRequest;
  2551. client.OnMoneyTransferRequest -= ProcessMoneyTransferRequest;
  2552. client.OnAvatarPickerRequest -= ProcessAvatarPickerRequest;
  2553. client.OnSetStartLocationRequest -= SetHomeRezPoint;
  2554. client.OnRegionHandleRequest -= RegionHandleRequest;
  2555. }
  2556. public virtual void UnSubscribeToClientNetworkEvents(IClientAPI client)
  2557. {
  2558. client.OnNetworkStatsUpdate -= StatsReporter.AddPacketsStats;
  2559. client.OnViewerEffect -= ProcessViewerEffect;
  2560. }
  2561. /// <summary>
  2562. /// Teleport an avatar to their home region
  2563. /// </summary>
  2564. /// <param name="agentId">The avatar's Unique ID</param>
  2565. /// <param name="client">The IClientAPI for the client</param>
  2566. public virtual void TeleportClientHome(UUID agentId, IClientAPI client)
  2567. {
  2568. if (m_teleportModule != null)
  2569. m_teleportModule.TeleportHome(agentId, client);
  2570. else
  2571. {
  2572. m_log.DebugFormat("[SCENE]: Unable to teleport user home: no AgentTransferModule is active");
  2573. client.SendTeleportFailed("Unable to perform teleports on this simulator.");
  2574. }
  2575. }
  2576. /// <summary>
  2577. /// Duplicates object specified by localID at position raycasted against RayTargetObject using
  2578. /// RayEnd and RayStart to determine what the angle of the ray is
  2579. /// </summary>
  2580. /// <param name="localID">ID of object to duplicate</param>
  2581. /// <param name="dupeFlags"></param>
  2582. /// <param name="AgentID">Agent doing the duplication</param>
  2583. /// <param name="GroupID">Group of new object</param>
  2584. /// <param name="RayTargetObj">The target of the Ray</param>
  2585. /// <param name="RayEnd">The ending of the ray (farthest away point)</param>
  2586. /// <param name="RayStart">The Beginning of the ray (closest point)</param>
  2587. /// <param name="BypassRaycast">Bool to bypass raycasting</param>
  2588. /// <param name="RayEndIsIntersection">The End specified is the place to add the object</param>
  2589. /// <param name="CopyCenters">Position the object at the center of the face that it's colliding with</param>
  2590. /// <param name="CopyRotates">Rotate the object the same as the localID object</param>
  2591. public void doObjectDuplicateOnRay(uint localID, uint dupeFlags, UUID AgentID, UUID GroupID,
  2592. UUID RayTargetObj, Vector3 RayEnd, Vector3 RayStart,
  2593. bool BypassRaycast, bool RayEndIsIntersection, bool CopyCenters, bool CopyRotates)
  2594. {
  2595. Vector3 pos;
  2596. const bool frontFacesOnly = true;
  2597. //m_log.Info("HITTARGET: " + RayTargetObj.ToString() + ", COPYTARGET: " + localID.ToString());
  2598. SceneObjectPart target = GetSceneObjectPart(localID);
  2599. SceneObjectPart target2 = GetSceneObjectPart(RayTargetObj);
  2600. if (target != null && target2 != null)
  2601. {
  2602. Vector3 direction = Vector3.Normalize(RayEnd - RayStart);
  2603. Vector3 AXOrigin = new Vector3(RayStart.X, RayStart.Y, RayStart.Z);
  2604. Vector3 AXdirection = new Vector3(direction.X, direction.Y, direction.Z);
  2605. if (target2.ParentGroup != null)
  2606. {
  2607. pos = target2.AbsolutePosition;
  2608. //m_log.Info("[OBJECT_REZ]: TargetPos: " + pos.ToString() + ", RayStart: " + RayStart.ToString() + ", RayEnd: " + RayEnd.ToString() + ", Volume: " + Util.GetDistanceTo(RayStart,RayEnd).ToString() + ", mag1: " + Util.GetMagnitude(RayStart).ToString() + ", mag2: " + Util.GetMagnitude(RayEnd).ToString());
  2609. // TODO: Raytrace better here
  2610. //EntityIntersection ei = m_sceneGraph.GetClosestIntersectingPrim(new Ray(AXOrigin, AXdirection));
  2611. Ray NewRay = new Ray(AXOrigin, AXdirection);
  2612. // Ray Trace against target here
  2613. EntityIntersection ei = target2.TestIntersectionOBB(NewRay, Quaternion.Identity, frontFacesOnly, CopyCenters);
  2614. // Un-comment out the following line to Get Raytrace results printed to the console.
  2615. //m_log.Info("[RAYTRACERESULTS]: Hit:" + ei.HitTF.ToString() + " Point: " + ei.ipoint.ToString() + " Normal: " + ei.normal.ToString());
  2616. float ScaleOffset = 0.5f;
  2617. // If we hit something
  2618. if (ei.HitTF)
  2619. {
  2620. Vector3 scale = target.Scale;
  2621. Vector3 scaleComponent = new Vector3(ei.AAfaceNormal.X, ei.AAfaceNormal.Y, ei.AAfaceNormal.Z);
  2622. if (scaleComponent.X != 0) ScaleOffset = scale.X;
  2623. if (scaleComponent.Y != 0) ScaleOffset = scale.Y;
  2624. if (scaleComponent.Z != 0) ScaleOffset = scale.Z;
  2625. ScaleOffset = Math.Abs(ScaleOffset);
  2626. Vector3 intersectionpoint = new Vector3(ei.ipoint.X, ei.ipoint.Y, ei.ipoint.Z);
  2627. Vector3 normal = new Vector3(ei.normal.X, ei.normal.Y, ei.normal.Z);
  2628. Vector3 offset = normal * (ScaleOffset / 2f);
  2629. pos = intersectionpoint + offset;
  2630. // stick in offset format from the original prim
  2631. pos = pos - target.ParentGroup.AbsolutePosition;
  2632. if (CopyRotates)
  2633. {
  2634. Quaternion worldRot = target2.GetWorldRotation();
  2635. // SceneObjectGroup obj = m_sceneGraph.DuplicateObject(localID, pos, target.GetEffectiveObjectFlags(), AgentID, GroupID, worldRot);
  2636. m_sceneGraph.DuplicateObject(localID, pos, target.GetEffectiveObjectFlags(), AgentID, GroupID, worldRot);
  2637. //obj.Rotation = worldRot;
  2638. //obj.UpdateGroupRotationR(worldRot);
  2639. }
  2640. else
  2641. {
  2642. m_sceneGraph.DuplicateObject(localID, pos, target.GetEffectiveObjectFlags(), AgentID, GroupID);
  2643. }
  2644. }
  2645. return;
  2646. }
  2647. return;
  2648. }
  2649. }
  2650. /// <summary>
  2651. /// Sets the Home Point. The LoginService uses this to know where to put a user when they log-in
  2652. /// </summary>
  2653. /// <param name="remoteClient"></param>
  2654. /// <param name="regionHandle"></param>
  2655. /// <param name="position"></param>
  2656. /// <param name="lookAt"></param>
  2657. /// <param name="flags"></param>
  2658. public virtual void SetHomeRezPoint(IClientAPI remoteClient, ulong regionHandle, Vector3 position, Vector3 lookAt, uint flags)
  2659. {
  2660. if (GridUserService != null && GridUserService.SetHome(remoteClient.AgentId.ToString(), RegionInfo.RegionID, position, lookAt))
  2661. // FUBAR ALERT: this needs to be "Home position set." so the viewer saves a home-screenshot.
  2662. m_dialogModule.SendAlertToUser(remoteClient, "Home position set.");
  2663. else
  2664. m_dialogModule.SendAlertToUser(remoteClient, "Set Home request Failed.");
  2665. }
  2666. /// <summary>
  2667. /// Get the avatar apperance for the given client.
  2668. /// </summary>
  2669. /// <param name="client"></param>
  2670. /// <param name="appearance"></param>
  2671. public void GetAvatarAppearance(IClientAPI client, out AvatarAppearance appearance)
  2672. {
  2673. AgentCircuitData aCircuit = m_authenticateHandler.GetAgentCircuitData(client.CircuitCode);
  2674. if (aCircuit == null)
  2675. {
  2676. m_log.DebugFormat("[APPEARANCE] Client did not supply a circuit. Non-Linden? Creating default appearance.");
  2677. appearance = new AvatarAppearance();
  2678. return;
  2679. }
  2680. appearance = aCircuit.Appearance;
  2681. if (appearance == null)
  2682. {
  2683. m_log.DebugFormat("[APPEARANCE]: Appearance not found in {0}, returning default", RegionInfo.RegionName);
  2684. appearance = new AvatarAppearance();
  2685. }
  2686. }
  2687. public override void RemoveClient(UUID agentID, bool closeChildAgents)
  2688. {
  2689. CheckHeartbeat();
  2690. bool childagentYN = false;
  2691. ScenePresence avatar = GetScenePresence(agentID);
  2692. if (avatar != null)
  2693. {
  2694. childagentYN = avatar.IsChildAgent;
  2695. if (avatar.ParentID != 0)
  2696. {
  2697. avatar.StandUp();
  2698. }
  2699. try
  2700. {
  2701. m_log.DebugFormat(
  2702. "[SCENE]: Removing {0} agent {1} from region {2}",
  2703. (childagentYN ? "child" : "root"), agentID, RegionInfo.RegionName);
  2704. m_sceneGraph.removeUserCount(!childagentYN);
  2705. // TODO: We shouldn't use closeChildAgents here - it's being used by the NPC module to stop
  2706. // unnecessary operations. This should go away once NPCs have no accompanying IClientAPI
  2707. if (closeChildAgents && CapsModule != null)
  2708. CapsModule.RemoveCaps(agentID);
  2709. // REFACTORING PROBLEM -- well not really a problem, but just to point out that whatever
  2710. // this method is doing is HORRIBLE!!!
  2711. avatar.Scene.NeedSceneCacheClear(avatar.UUID);
  2712. if (closeChildAgents && !avatar.IsChildAgent)
  2713. {
  2714. //List<ulong> childknownRegions = new List<ulong>();
  2715. //List<ulong> ckn = avatar.KnownChildRegionHandles;
  2716. //for (int i = 0; i < ckn.Count; i++)
  2717. //{
  2718. // childknownRegions.Add(ckn[i]);
  2719. //}
  2720. List<ulong> regions = new List<ulong>(avatar.KnownChildRegionHandles);
  2721. regions.Remove(RegionInfo.RegionHandle);
  2722. m_sceneGridService.SendCloseChildAgentConnections(agentID, regions);
  2723. }
  2724. m_eventManager.TriggerClientClosed(agentID, this);
  2725. }
  2726. catch (NullReferenceException)
  2727. {
  2728. // We don't know which count to remove it from
  2729. // Avatar is already disposed :/
  2730. }
  2731. m_eventManager.TriggerOnRemovePresence(agentID);
  2732. if (avatar != null && (!avatar.IsChildAgent) && avatar.PresenceType != PresenceType.Npc)
  2733. avatar.SaveChangedAttachments();
  2734. ForEachClient(
  2735. delegate(IClientAPI client)
  2736. {
  2737. //We can safely ignore null reference exceptions. It means the avatar is dead and cleaned up anyway
  2738. try { client.SendKillObject(avatar.RegionHandle, avatar.LocalId); }
  2739. catch (NullReferenceException) { }
  2740. });
  2741. IAgentAssetTransactions agentTransactions = this.RequestModuleInterface<IAgentAssetTransactions>();
  2742. if (agentTransactions != null)
  2743. {
  2744. agentTransactions.RemoveAgentAssetTransactions(agentID);
  2745. }
  2746. // Remove the avatar from the scene
  2747. m_sceneGraph.RemoveScenePresence(agentID);
  2748. m_clientManager.Remove(agentID);
  2749. try
  2750. {
  2751. avatar.Close();
  2752. }
  2753. catch (NullReferenceException)
  2754. {
  2755. //We can safely ignore null reference exceptions. It means the avatar are dead and cleaned up anyway.
  2756. }
  2757. catch (Exception e)
  2758. {
  2759. m_log.Error("[SCENE] Scene.cs:RemoveClient exception: " + e.ToString());
  2760. }
  2761. m_authenticateHandler.RemoveCircuit(avatar.ControllingClient.CircuitCode);
  2762. CleanDroppedAttachments();
  2763. //m_log.InfoFormat("[SCENE] Memory pre GC {0}", System.GC.GetTotalMemory(false));
  2764. //m_log.InfoFormat("[SCENE] Memory post GC {0}", System.GC.GetTotalMemory(true));
  2765. }
  2766. }
  2767. /// <summary>
  2768. /// Removes region from an avatar's known region list. This coincides with child agents. For each child agent, there will be a known region entry.
  2769. ///
  2770. /// </summary>
  2771. /// <param name="avatarID"></param>
  2772. /// <param name="regionslst"></param>
  2773. public void HandleRemoveKnownRegionsFromAvatar(UUID avatarID, List<ulong> regionslst)
  2774. {
  2775. ScenePresence av = GetScenePresence(avatarID);
  2776. if (av != null)
  2777. {
  2778. lock (av)
  2779. {
  2780. for (int i = 0; i < regionslst.Count; i++)
  2781. {
  2782. av.KnownChildRegionHandles.Remove(regionslst[i]);
  2783. }
  2784. }
  2785. }
  2786. }
  2787. #endregion
  2788. #region Entities
  2789. public void SendKillObject(uint localID)
  2790. {
  2791. SceneObjectPart part = GetSceneObjectPart(localID);
  2792. if (part != null) // It is a prim
  2793. {
  2794. if (part.ParentGroup != null && !part.ParentGroup.IsDeleted) // Valid
  2795. {
  2796. if (part.ParentGroup.RootPart != part) // Child part
  2797. return;
  2798. }
  2799. }
  2800. ForEachClient(delegate(IClientAPI client) { client.SendKillObject(m_regionHandle, localID); });
  2801. }
  2802. #endregion
  2803. #region RegionComms
  2804. /// <summary>
  2805. /// Register the methods that should be invoked when this scene receives various incoming events
  2806. /// </summary>
  2807. public void RegisterCommsEvents()
  2808. {
  2809. m_sceneGridService.OnAvatarCrossingIntoRegion += AgentCrossing;
  2810. m_sceneGridService.OnCloseAgentConnection += IncomingCloseAgent;
  2811. //m_eventManager.OnRegionUp += OtherRegionUp;
  2812. //m_sceneGridService.OnChildAgentUpdate += IncomingChildAgentDataUpdate;
  2813. //m_sceneGridService.OnRemoveKnownRegionFromAvatar += HandleRemoveKnownRegionsFromAvatar;
  2814. m_sceneGridService.OnLogOffUser += HandleLogOffUserFromGrid;
  2815. m_sceneGridService.KiPrimitive += SendKillObject;
  2816. m_sceneGridService.OnGetLandData += GetLandData;
  2817. }
  2818. /// <summary>
  2819. /// Deregister this scene from receiving incoming region events
  2820. /// </summary>
  2821. public void UnRegisterRegionWithComms()
  2822. {
  2823. m_sceneGridService.KiPrimitive -= SendKillObject;
  2824. m_sceneGridService.OnLogOffUser -= HandleLogOffUserFromGrid;
  2825. //m_sceneGridService.OnRemoveKnownRegionFromAvatar -= HandleRemoveKnownRegionsFromAvatar;
  2826. //m_sceneGridService.OnChildAgentUpdate -= IncomingChildAgentDataUpdate;
  2827. //m_eventManager.OnRegionUp -= OtherRegionUp;
  2828. m_sceneGridService.OnAvatarCrossingIntoRegion -= AgentCrossing;
  2829. m_sceneGridService.OnCloseAgentConnection -= IncomingCloseAgent;
  2830. m_sceneGridService.OnGetLandData -= GetLandData;
  2831. // this does nothing; should be removed
  2832. m_sceneGridService.Close();
  2833. if (!GridService.DeregisterRegion(m_regInfo.RegionID))
  2834. m_log.WarnFormat("[SCENE]: Deregister from grid failed for region {0}", m_regInfo.RegionName);
  2835. }
  2836. /// <summary>
  2837. /// Do the work necessary to initiate a new user connection for a particular scene.
  2838. /// At the moment, this consists of setting up the caps infrastructure
  2839. /// The return bool should allow for connections to be refused, but as not all calling paths
  2840. /// take proper notice of it let, we allowed banned users in still.
  2841. /// </summary>
  2842. /// <param name="agent">CircuitData of the agent who is connecting</param>
  2843. /// <param name="reason">Outputs the reason for the false response on this string</param>
  2844. /// <returns>True if the region accepts this agent. False if it does not. False will
  2845. /// also return a reason.</returns>
  2846. public bool NewUserConnection(AgentCircuitData agent, uint teleportFlags, out string reason)
  2847. {
  2848. return NewUserConnection(agent, teleportFlags, out reason, true);
  2849. }
  2850. /// <summary>
  2851. /// Do the work necessary to initiate a new user connection for a particular scene.
  2852. /// At the moment, this consists of setting up the caps infrastructure
  2853. /// The return bool should allow for connections to be refused, but as not all calling paths
  2854. /// take proper notice of it let, we allowed banned users in still.
  2855. /// </summary>
  2856. /// <param name="agent">CircuitData of the agent who is connecting</param>
  2857. /// <param name="reason">Outputs the reason for the false response on this string</param>
  2858. /// <param name="requirePresenceLookup">True for normal presence. False for NPC
  2859. /// or other applications where a full grid/Hypergrid presence may not be required.</param>
  2860. /// <returns>True if the region accepts this agent. False if it does not. False will
  2861. /// also return a reason.</returns>
  2862. public bool NewUserConnection(AgentCircuitData agent, uint teleportFlags, out string reason, bool requirePresenceLookup)
  2863. {
  2864. bool vialogin = ((teleportFlags & (uint)Constants.TeleportFlags.ViaLogin) != 0 ||
  2865. (teleportFlags & (uint)Constants.TeleportFlags.ViaHGLogin) != 0);
  2866. reason = String.Empty;
  2867. //Teleport flags:
  2868. //
  2869. // TeleportFlags.ViaGodlikeLure - Border Crossing
  2870. // TeleportFlags.ViaLogin - Login
  2871. // TeleportFlags.TeleportFlags.ViaLure - Teleport request sent by another user
  2872. // TeleportFlags.ViaLandmark | TeleportFlags.ViaLocation | TeleportFlags.ViaLandmark | TeleportFlags.Default - Regular Teleport
  2873. // Don't disable this log message - it's too helpful
  2874. m_log.DebugFormat(
  2875. "[CONNECTION BEGIN]: Region {0} told of incoming {1} agent {2} {3} {4} (circuit code {5}, teleportflags {6})",
  2876. RegionInfo.RegionName, (agent.child ? "child" : "root"), agent.firstname, agent.lastname,
  2877. agent.AgentID, agent.circuitcode, teleportFlags);
  2878. if (LoginsDisabled)
  2879. {
  2880. reason = "Logins Disabled";
  2881. return false;
  2882. }
  2883. ScenePresence sp = GetScenePresence(agent.AgentID);
  2884. if (sp != null && !sp.IsChildAgent)
  2885. {
  2886. // We have a zombie from a crashed session.
  2887. // Or the same user is trying to be root twice here, won't work.
  2888. // Kill it.
  2889. m_log.DebugFormat("[SCENE]: Zombie scene presence detected for {0} in {1}", agent.AgentID, RegionInfo.RegionName);
  2890. sp.ControllingClient.Close();
  2891. sp = null;
  2892. }
  2893. ILandObject land = LandChannel.GetLandObject(agent.startpos.X, agent.startpos.Y);
  2894. //On login test land permisions
  2895. if (vialogin)
  2896. {
  2897. if (land != null && !TestLandRestrictions(agent, land, out reason))
  2898. {
  2899. return false;
  2900. }
  2901. }
  2902. if (sp == null) // We don't have an [child] agent here already
  2903. {
  2904. if (requirePresenceLookup)
  2905. {
  2906. try
  2907. {
  2908. if (!VerifyUserPresence(agent, out reason))
  2909. return false;
  2910. }
  2911. catch (Exception e)
  2912. {
  2913. m_log.ErrorFormat("[CONNECTION BEGIN]: Exception verifying presence " + e.ToString());
  2914. return false;
  2915. }
  2916. }
  2917. try
  2918. {
  2919. if (!AuthorizeUser(agent, out reason))
  2920. return false;
  2921. }
  2922. catch (Exception e)
  2923. {
  2924. m_log.ErrorFormat("[CONNECTION BEGIN]: Exception authorizing user " + e.ToString());
  2925. return false;
  2926. }
  2927. m_log.InfoFormat(
  2928. "[CONNECTION BEGIN]: Region {0} authenticated and authorized incoming {1} agent {2} {3} {4} (circuit code {5})",
  2929. RegionInfo.RegionName, (agent.child ? "child" : "root"), agent.firstname, agent.lastname,
  2930. agent.AgentID, agent.circuitcode);
  2931. if (CapsModule != null)
  2932. {
  2933. CapsModule.SetAgentCapsSeeds(agent);
  2934. CapsModule.CreateCaps(agent.AgentID);
  2935. }
  2936. }
  2937. else
  2938. {
  2939. // Let the SP know how we got here. This has a lot of interesting
  2940. // uses down the line.
  2941. sp.TeleportFlags = (TeleportFlags)teleportFlags;
  2942. if (sp.IsChildAgent)
  2943. {
  2944. m_log.DebugFormat(
  2945. "[SCENE]: Adjusting known seeds for existing agent {0} in {1}",
  2946. agent.AgentID, RegionInfo.RegionName);
  2947. sp.AdjustKnownSeeds();
  2948. if (CapsModule != null)
  2949. CapsModule.SetAgentCapsSeeds(agent);
  2950. }
  2951. }
  2952. // In all cases, add or update the circuit data with the new agent circuit data and teleport flags
  2953. agent.teleportFlags = teleportFlags;
  2954. m_authenticateHandler.AddNewCircuit(agent.circuitcode, agent);
  2955. if (vialogin)
  2956. {
  2957. CleanDroppedAttachments();
  2958. if (TestBorderCross(agent.startpos, Cardinals.E))
  2959. {
  2960. Border crossedBorder = GetCrossedBorder(agent.startpos, Cardinals.E);
  2961. agent.startpos.X = crossedBorder.BorderLine.Z - 1;
  2962. }
  2963. if (TestBorderCross(agent.startpos, Cardinals.N))
  2964. {
  2965. Border crossedBorder = GetCrossedBorder(agent.startpos, Cardinals.N);
  2966. agent.startpos.Y = crossedBorder.BorderLine.Z - 1;
  2967. }
  2968. //Mitigate http://opensimulator.org/mantis/view.php?id=3522
  2969. // Check if start position is outside of region
  2970. // If it is, check the Z start position also.. if not, leave it alone.
  2971. if (BordersLocked)
  2972. {
  2973. lock (EastBorders)
  2974. {
  2975. if (agent.startpos.X > EastBorders[0].BorderLine.Z)
  2976. {
  2977. m_log.Warn("FIX AGENT POSITION");
  2978. agent.startpos.X = EastBorders[0].BorderLine.Z * 0.5f;
  2979. if (agent.startpos.Z > 720)
  2980. agent.startpos.Z = 720;
  2981. }
  2982. }
  2983. lock (NorthBorders)
  2984. {
  2985. if (agent.startpos.Y > NorthBorders[0].BorderLine.Z)
  2986. {
  2987. m_log.Warn("FIX Agent POSITION");
  2988. agent.startpos.Y = NorthBorders[0].BorderLine.Z * 0.5f;
  2989. if (agent.startpos.Z > 720)
  2990. agent.startpos.Z = 720;
  2991. }
  2992. }
  2993. }
  2994. else
  2995. {
  2996. if (agent.startpos.X > EastBorders[0].BorderLine.Z)
  2997. {
  2998. m_log.Warn("FIX AGENT POSITION");
  2999. agent.startpos.X = EastBorders[0].BorderLine.Z * 0.5f;
  3000. if (agent.startpos.Z > 720)
  3001. agent.startpos.Z = 720;
  3002. }
  3003. if (agent.startpos.Y > NorthBorders[0].BorderLine.Z)
  3004. {
  3005. m_log.Warn("FIX Agent POSITION");
  3006. agent.startpos.Y = NorthBorders[0].BorderLine.Z * 0.5f;
  3007. if (agent.startpos.Z > 720)
  3008. agent.startpos.Z = 720;
  3009. }
  3010. }
  3011. // Honor parcel landing type and position.
  3012. if (land != null)
  3013. {
  3014. if (land.LandData.LandingType == (byte)1 && land.LandData.UserLocation != Vector3.Zero)
  3015. {
  3016. agent.startpos = land.LandData.UserLocation;
  3017. }
  3018. }
  3019. }
  3020. return true;
  3021. }
  3022. private bool TestLandRestrictions(AgentCircuitData agent, ILandObject land, out string reason)
  3023. {
  3024. bool banned = land.IsBannedFromLand(agent.AgentID);
  3025. bool restricted = land.IsRestrictedFromLand(agent.AgentID);
  3026. if (banned || restricted)
  3027. {
  3028. ILandObject nearestParcel = GetNearestAllowedParcel(agent.AgentID, agent.startpos.X, agent.startpos.Y);
  3029. if (nearestParcel != null)
  3030. {
  3031. //Move agent to nearest allowed
  3032. Vector3 newPosition = GetParcelCenterAtGround(nearestParcel);
  3033. agent.startpos.X = newPosition.X;
  3034. agent.startpos.Y = newPosition.Y;
  3035. }
  3036. else
  3037. {
  3038. if (banned)
  3039. {
  3040. reason = "Cannot regioncross into banned parcel.";
  3041. }
  3042. else
  3043. {
  3044. reason = String.Format("Denied access to private region {0}: You are not on the access list for that region.",
  3045. RegionInfo.RegionName);
  3046. }
  3047. return false;
  3048. }
  3049. }
  3050. reason = "";
  3051. return true;
  3052. }
  3053. /// <summary>
  3054. /// Verifies that the user has a presence on the Grid
  3055. /// </summary>
  3056. /// <param name="agent">Circuit Data of the Agent we're verifying</param>
  3057. /// <param name="reason">Outputs the reason for the false response on this string</param>
  3058. /// <returns>True if the user has a session on the grid. False if it does not. False will
  3059. /// also return a reason.</returns>
  3060. public virtual bool VerifyUserPresence(AgentCircuitData agent, out string reason)
  3061. {
  3062. reason = String.Empty;
  3063. IPresenceService presence = RequestModuleInterface<IPresenceService>();
  3064. if (presence == null)
  3065. {
  3066. reason = String.Format("Failed to verify user presence in the grid for {0} {1} in region {2}. Presence service does not exist.", agent.firstname, agent.lastname, RegionInfo.RegionName);
  3067. return false;
  3068. }
  3069. OpenSim.Services.Interfaces.PresenceInfo pinfo = presence.GetAgent(agent.SessionID);
  3070. if (pinfo == null)
  3071. {
  3072. reason = String.Format("Failed to verify user presence in the grid for {0} {1}, access denied to region {2}.", agent.firstname, agent.lastname, RegionInfo.RegionName);
  3073. return false;
  3074. }
  3075. return true;
  3076. }
  3077. /// <summary>
  3078. /// Verify if the user can connect to this region. Checks the banlist and ensures that the region is set for public access
  3079. /// </summary>
  3080. /// <param name="agent">The circuit data for the agent</param>
  3081. /// <param name="reason">outputs the reason to this string</param>
  3082. /// <returns>True if the region accepts this agent. False if it does not. False will
  3083. /// also return a reason.</returns>
  3084. protected virtual bool AuthorizeUser(AgentCircuitData agent, out string reason)
  3085. {
  3086. reason = String.Empty;
  3087. if (!m_strictAccessControl) return true;
  3088. if (Permissions.IsGod(agent.AgentID)) return true;
  3089. if (AuthorizationService != null)
  3090. {
  3091. if (!AuthorizationService.IsAuthorizedForRegion(
  3092. agent.AgentID.ToString(), agent.firstname, agent.lastname, RegionInfo.RegionID.ToString(), out reason))
  3093. {
  3094. m_log.WarnFormat("[CONNECTION BEGIN]: Denied access to: {0} ({1} {2}) at {3} because the user does not have access to the region",
  3095. agent.AgentID, agent.firstname, agent.lastname, RegionInfo.RegionName);
  3096. return false;
  3097. }
  3098. }
  3099. if (m_regInfo.EstateSettings != null)
  3100. {
  3101. if (m_regInfo.EstateSettings.IsBanned(agent.AgentID))
  3102. {
  3103. m_log.WarnFormat("[CONNECTION BEGIN]: Denied access to: {0} ({1} {2}) at {3} because the user is on the banlist",
  3104. agent.AgentID, agent.firstname, agent.lastname, RegionInfo.RegionName);
  3105. reason = String.Format("Denied access to region {0}: You have been banned from that region.",
  3106. RegionInfo.RegionName);
  3107. return false;
  3108. }
  3109. }
  3110. else
  3111. m_log.ErrorFormat("[CONNECTION BEGIN]: Estate Settings is null!");
  3112. IGroupsModule groupsModule =
  3113. RequestModuleInterface<IGroupsModule>();
  3114. List<UUID> agentGroups = new List<UUID>();
  3115. if (groupsModule != null)
  3116. {
  3117. GroupMembershipData[] GroupMembership =
  3118. groupsModule.GetMembershipData(agent.AgentID);
  3119. if (GroupMembership != null)
  3120. {
  3121. for (int i = 0; i < GroupMembership.Length; i++)
  3122. agentGroups.Add(GroupMembership[i].GroupID);
  3123. }
  3124. else
  3125. m_log.ErrorFormat("[CONNECTION BEGIN]: GroupMembership is null!");
  3126. }
  3127. bool groupAccess = false;
  3128. UUID[] estateGroups = m_regInfo.EstateSettings.EstateGroups;
  3129. if (estateGroups != null)
  3130. {
  3131. foreach (UUID group in estateGroups)
  3132. {
  3133. if (agentGroups.Contains(group))
  3134. {
  3135. groupAccess = true;
  3136. break;
  3137. }
  3138. }
  3139. }
  3140. else
  3141. m_log.ErrorFormat("[CONNECTION BEGIN]: EstateGroups is null!");
  3142. if (!m_regInfo.EstateSettings.PublicAccess &&
  3143. !m_regInfo.EstateSettings.HasAccess(agent.AgentID) &&
  3144. !groupAccess)
  3145. {
  3146. m_log.WarnFormat("[CONNECTION BEGIN]: Denied access to: {0} ({1} {2}) at {3} because the user does not have access to the estate",
  3147. agent.AgentID, agent.firstname, agent.lastname, RegionInfo.RegionName);
  3148. reason = String.Format("Denied access to private region {0}: You are not on the access list for that region.",
  3149. RegionInfo.RegionName);
  3150. return false;
  3151. }
  3152. // TODO: estate/region settings are not properly hooked up
  3153. // to ILandObject.isRestrictedFromLand()
  3154. // if (null != LandChannel)
  3155. // {
  3156. // // region seems to have local Id of 1
  3157. // ILandObject land = LandChannel.GetLandObject(1);
  3158. // if (null != land)
  3159. // {
  3160. // if (land.isBannedFromLand(agent.AgentID))
  3161. // {
  3162. // m_log.WarnFormat("[CONNECTION BEGIN]: Denied access to: {0} ({1} {2}) at {3} because the user has been banned from land",
  3163. // agent.AgentID, agent.firstname, agent.lastname, RegionInfo.RegionName);
  3164. // reason = String.Format("Denied access to private region {0}: You are banned from that region.",
  3165. // RegionInfo.RegionName);
  3166. // return false;
  3167. // }
  3168. // if (land.isRestrictedFromLand(agent.AgentID))
  3169. // {
  3170. // m_log.WarnFormat("[CONNECTION BEGIN]: Denied access to: {0} ({1} {2}) at {3} because the user does not have access to the region",
  3171. // agent.AgentID, agent.firstname, agent.lastname, RegionInfo.RegionName);
  3172. // reason = String.Format("Denied access to private region {0}: You are not on the access list for that region.",
  3173. // RegionInfo.RegionName);
  3174. // return false;
  3175. // }
  3176. // }
  3177. // }
  3178. return true;
  3179. }
  3180. /// <summary>
  3181. /// Update an AgentCircuitData object with new information
  3182. /// </summary>
  3183. /// <param name="data">Information to update the AgentCircuitData with</param>
  3184. public void UpdateCircuitData(AgentCircuitData data)
  3185. {
  3186. m_authenticateHandler.UpdateAgentData(data);
  3187. }
  3188. /// <summary>
  3189. /// Change the Circuit Code for the user's Circuit Data
  3190. /// </summary>
  3191. /// <param name="oldcc">The old Circuit Code. Must match a previous circuit code</param>
  3192. /// <param name="newcc">The new Circuit Code. Must not be an already existing circuit code</param>
  3193. /// <returns>True if we successfully changed it. False if we did not</returns>
  3194. public bool ChangeCircuitCode(uint oldcc, uint newcc)
  3195. {
  3196. return m_authenticateHandler.TryChangeCiruitCode(oldcc, newcc);
  3197. }
  3198. /// <summary>
  3199. /// The Grid has requested that we log-off a user. Log them off.
  3200. /// </summary>
  3201. /// <param name="AvatarID">Unique ID of the avatar to log-off</param>
  3202. /// <param name="RegionSecret">SecureSessionID of the user, or the RegionSecret text when logging on to the grid</param>
  3203. /// <param name="message">message to display to the user. Reason for being logged off</param>
  3204. public void HandleLogOffUserFromGrid(UUID AvatarID, UUID RegionSecret, string message)
  3205. {
  3206. ScenePresence loggingOffUser = GetScenePresence(AvatarID);
  3207. if (loggingOffUser != null)
  3208. {
  3209. UUID localRegionSecret = UUID.Zero;
  3210. bool parsedsecret = UUID.TryParse(m_regInfo.regionSecret, out localRegionSecret);
  3211. // Region Secret is used here in case a new sessionid overwrites an old one on the user server.
  3212. // Will update the user server in a few revisions to use it.
  3213. if (RegionSecret == loggingOffUser.ControllingClient.SecureSessionId || (parsedsecret && RegionSecret == localRegionSecret))
  3214. {
  3215. m_sceneGridService.SendCloseChildAgentConnections(loggingOffUser.UUID, new List<ulong>(loggingOffUser.KnownRegions.Keys));
  3216. loggingOffUser.ControllingClient.Kick(message);
  3217. // Give them a second to receive the message!
  3218. Thread.Sleep(1000);
  3219. loggingOffUser.ControllingClient.Close();
  3220. }
  3221. else
  3222. {
  3223. m_log.Info("[USERLOGOFF]: System sending the LogOff user message failed to sucessfully authenticate");
  3224. }
  3225. }
  3226. else
  3227. {
  3228. m_log.InfoFormat("[USERLOGOFF]: Got a logoff request for {0} but the user isn't here. The user might already have been logged out", AvatarID.ToString());
  3229. }
  3230. }
  3231. /// <summary>
  3232. /// Triggered when an agent crosses into this sim. Also happens on initial login.
  3233. /// </summary>
  3234. /// <param name="agentID"></param>
  3235. /// <param name="position"></param>
  3236. /// <param name="isFlying"></param>
  3237. public virtual void AgentCrossing(UUID agentID, Vector3 position, bool isFlying)
  3238. {
  3239. ScenePresence presence = GetScenePresence(agentID);
  3240. if (presence != null)
  3241. {
  3242. try
  3243. {
  3244. presence.MakeRootAgent(position, isFlying);
  3245. }
  3246. catch (Exception e)
  3247. {
  3248. m_log.ErrorFormat("[SCENE]: Unable to do agent crossing, exception {0}", e);
  3249. }
  3250. }
  3251. else
  3252. {
  3253. m_log.ErrorFormat(
  3254. "[SCENE]: Could not find presence for agent {0} crossing into scene {1}",
  3255. agentID, RegionInfo.RegionName);
  3256. }
  3257. }
  3258. /// <summary>
  3259. /// We've got an update about an agent that sees into this region,
  3260. /// send it to ScenePresence for processing It's the full data.
  3261. /// </summary>
  3262. /// <param name="cAgentData">Agent that contains all of the relevant things about an agent.
  3263. /// Appearance, animations, position, etc.</param>
  3264. /// <returns>true if we handled it.</returns>
  3265. public virtual bool IncomingChildAgentDataUpdate(AgentData cAgentData)
  3266. {
  3267. m_log.DebugFormat(
  3268. "[SCENE]: Incoming child agent update for {0} in {1}", cAgentData.AgentID, RegionInfo.RegionName);
  3269. // XPTO: if this agent is not allowed here as root, always return false
  3270. // We have to wait until the viewer contacts this region after receiving EAC.
  3271. // That calls AddNewClient, which finally creates the ScenePresence
  3272. ILandObject nearestParcel = GetNearestAllowedParcel(cAgentData.AgentID, Constants.RegionSize / 2, Constants.RegionSize / 2);
  3273. if (nearestParcel == null)
  3274. {
  3275. m_log.DebugFormat("[SCENE]: Denying root agent entry to {0}: no allowed parcel", cAgentData.AgentID);
  3276. return false;
  3277. }
  3278. int num = m_sceneGraph.GetNumberOfScenePresences();
  3279. if (num >= RegionInfo.RegionSettings.AgentLimit)
  3280. {
  3281. if (!Permissions.IsAdministrator(cAgentData.AgentID))
  3282. return false;
  3283. }
  3284. ScenePresence childAgentUpdate = WaitGetScenePresence(cAgentData.AgentID);
  3285. if (childAgentUpdate != null)
  3286. {
  3287. childAgentUpdate.ChildAgentDataUpdate(cAgentData);
  3288. return true;
  3289. }
  3290. return false;
  3291. }
  3292. /// <summary>
  3293. /// We've got an update about an agent that sees into this region,
  3294. /// send it to ScenePresence for processing It's only positional data
  3295. /// </summary>
  3296. /// <param name="cAgentData">AgentPosition that contains agent positional data so we can know what to send</param>
  3297. /// <returns>true if we handled it.</returns>
  3298. public virtual bool IncomingChildAgentDataUpdate(AgentPosition cAgentData)
  3299. {
  3300. //m_log.Debug(" XXX Scene IncomingChildAgentDataUpdate POSITION in " + RegionInfo.RegionName);
  3301. ScenePresence childAgentUpdate = GetScenePresence(cAgentData.AgentID);
  3302. if (childAgentUpdate != null)
  3303. {
  3304. // I can't imagine *yet* why we would get an update if the agent is a root agent..
  3305. // however to avoid a race condition crossing borders..
  3306. if (childAgentUpdate.IsChildAgent)
  3307. {
  3308. uint rRegionX = (uint)(cAgentData.RegionHandle >> 40);
  3309. uint rRegionY = (((uint)(cAgentData.RegionHandle)) >> 8);
  3310. uint tRegionX = RegionInfo.RegionLocX;
  3311. uint tRegionY = RegionInfo.RegionLocY;
  3312. //Send Data to ScenePresence
  3313. childAgentUpdate.ChildAgentDataUpdate(cAgentData, tRegionX, tRegionY, rRegionX, rRegionY);
  3314. // Not Implemented:
  3315. //TODO: Do we need to pass the message on to one of our neighbors?
  3316. }
  3317. return true;
  3318. }
  3319. return false;
  3320. }
  3321. protected virtual ScenePresence WaitGetScenePresence(UUID agentID)
  3322. {
  3323. int ntimes = 10;
  3324. ScenePresence childAgentUpdate = null;
  3325. while ((childAgentUpdate = GetScenePresence(agentID)) == null && (ntimes-- > 0))
  3326. Thread.Sleep(1000);
  3327. return childAgentUpdate;
  3328. }
  3329. public virtual bool IncomingRetrieveRootAgent(UUID id, out IAgentData agent)
  3330. {
  3331. agent = null;
  3332. ScenePresence sp = GetScenePresence(id);
  3333. if ((sp != null) && (!sp.IsChildAgent))
  3334. {
  3335. sp.IsChildAgent = true;
  3336. return sp.CopyAgent(out agent);
  3337. }
  3338. return false;
  3339. }
  3340. /// <summary>
  3341. /// Tell a single agent to disconnect from the region.
  3342. /// </summary>
  3343. /// <param name="regionHandle"></param>
  3344. /// <param name="agentID"></param>
  3345. public bool IncomingCloseAgent(UUID agentID)
  3346. {
  3347. //m_log.DebugFormat("[SCENE]: Processing incoming close agent for {0}", agentID);
  3348. ScenePresence presence = m_sceneGraph.GetScenePresence(agentID);
  3349. if (presence != null)
  3350. {
  3351. // Nothing is removed here, so down count it as such
  3352. if (presence.IsChildAgent)
  3353. {
  3354. m_sceneGraph.removeUserCount(false);
  3355. }
  3356. else
  3357. {
  3358. m_sceneGraph.removeUserCount(true);
  3359. }
  3360. // Don't do this to root agents on logout, it's not nice for the viewer
  3361. if (presence.IsChildAgent)
  3362. {
  3363. // Tell a single agent to disconnect from the region.
  3364. IEventQueue eq = RequestModuleInterface<IEventQueue>();
  3365. if (eq != null)
  3366. {
  3367. eq.DisableSimulator(RegionInfo.RegionHandle, agentID);
  3368. }
  3369. else
  3370. presence.ControllingClient.SendShutdownConnectionNotice();
  3371. }
  3372. presence.ControllingClient.Close();
  3373. return true;
  3374. }
  3375. // Agent not here
  3376. return false;
  3377. }
  3378. /// <summary>
  3379. /// Tries to teleport agent to another region.
  3380. /// </summary>
  3381. /// <remarks>
  3382. /// The region name must exactly match that given.
  3383. /// </remarks>
  3384. /// <param name="remoteClient"></param>
  3385. /// <param name="regionName"></param>
  3386. /// <param name="position"></param>
  3387. /// <param name="lookAt"></param>
  3388. /// <param name="teleportFlags"></param>
  3389. public void RequestTeleportLocation(IClientAPI remoteClient, string regionName, Vector3 position,
  3390. Vector3 lookat, uint teleportFlags)
  3391. {
  3392. GridRegion region = GridService.GetRegionByName(RegionInfo.ScopeID, regionName);
  3393. if (region == null)
  3394. {
  3395. // can't find the region: Tell viewer and abort
  3396. remoteClient.SendTeleportFailed("The region '" + regionName + "' could not be found.");
  3397. return;
  3398. }
  3399. RequestTeleportLocation(remoteClient, region.RegionHandle, position, lookat, teleportFlags);
  3400. }
  3401. /// <summary>
  3402. /// Tries to teleport agent to other region.
  3403. /// </summary>
  3404. /// <param name="remoteClient"></param>
  3405. /// <param name="regionHandle"></param>
  3406. /// <param name="position"></param>
  3407. /// <param name="lookAt"></param>
  3408. /// <param name="teleportFlags"></param>
  3409. public void RequestTeleportLocation(IClientAPI remoteClient, ulong regionHandle, Vector3 position,
  3410. Vector3 lookAt, uint teleportFlags)
  3411. {
  3412. ScenePresence sp = GetScenePresence(remoteClient.AgentId);
  3413. if (sp != null)
  3414. {
  3415. uint regionX = m_regInfo.RegionLocX;
  3416. uint regionY = m_regInfo.RegionLocY;
  3417. Utils.LongToUInts(regionHandle, out regionX, out regionY);
  3418. int shiftx = (int) regionX - (int) m_regInfo.RegionLocX * (int)Constants.RegionSize;
  3419. int shifty = (int) regionY - (int) m_regInfo.RegionLocY * (int)Constants.RegionSize;
  3420. position.X += shiftx;
  3421. position.Y += shifty;
  3422. bool result = false;
  3423. if (TestBorderCross(position,Cardinals.N))
  3424. result = true;
  3425. if (TestBorderCross(position, Cardinals.S))
  3426. result = true;
  3427. if (TestBorderCross(position, Cardinals.E))
  3428. result = true;
  3429. if (TestBorderCross(position, Cardinals.W))
  3430. result = true;
  3431. // bordercross if position is outside of region
  3432. if (!result)
  3433. {
  3434. regionHandle = m_regInfo.RegionHandle;
  3435. }
  3436. else
  3437. {
  3438. // not in this region, undo the shift!
  3439. position.X -= shiftx;
  3440. position.Y -= shifty;
  3441. }
  3442. if (m_teleportModule != null)
  3443. m_teleportModule.Teleport(sp, regionHandle, position, lookAt, teleportFlags);
  3444. else
  3445. {
  3446. m_log.DebugFormat("[SCENE]: Unable to perform teleports: no AgentTransferModule is active");
  3447. sp.ControllingClient.SendTeleportFailed("Unable to perform teleports on this simulator.");
  3448. }
  3449. }
  3450. }
  3451. public bool CrossAgentToNewRegion(ScenePresence agent, bool isFlying)
  3452. {
  3453. if (m_teleportModule != null)
  3454. return m_teleportModule.Cross(agent, isFlying);
  3455. else
  3456. {
  3457. m_log.DebugFormat("[SCENE]: Unable to cross agent to neighbouring region, because there is no AgentTransferModule");
  3458. }
  3459. return false;
  3460. }
  3461. public void SendOutChildAgentUpdates(AgentPosition cadu, ScenePresence presence)
  3462. {
  3463. m_sceneGridService.SendChildAgentDataUpdate(cadu, presence);
  3464. }
  3465. #endregion
  3466. #region Other Methods
  3467. protected override IConfigSource GetConfig()
  3468. {
  3469. return m_config;
  3470. }
  3471. #endregion
  3472. public void HandleObjectPermissionsUpdate(IClientAPI controller, UUID agentID, UUID sessionID, byte field, uint localId, uint mask, byte set)
  3473. {
  3474. // Check for spoofing.. since this is permissions we're talking about here!
  3475. if ((controller.SessionId == sessionID) && (controller.AgentId == agentID))
  3476. {
  3477. // Tell the object to do permission update
  3478. if (localId != 0)
  3479. {
  3480. SceneObjectGroup chObjectGroup = GetGroupByPrim(localId);
  3481. if (chObjectGroup != null)
  3482. {
  3483. chObjectGroup.UpdatePermissions(agentID, field, localId, mask, set);
  3484. }
  3485. }
  3486. }
  3487. }
  3488. /// <summary>
  3489. /// Causes all clients to get a full object update on all of the objects in the scene.
  3490. /// </summary>
  3491. public void ForceClientUpdate()
  3492. {
  3493. EntityBase[] entityList = GetEntities();
  3494. foreach (EntityBase ent in entityList)
  3495. {
  3496. if (ent is SceneObjectGroup)
  3497. {
  3498. ((SceneObjectGroup)ent).ScheduleGroupForFullUpdate();
  3499. }
  3500. }
  3501. }
  3502. /// <summary>
  3503. /// This is currently only used for scale (to scale to MegaPrim size)
  3504. /// There is a console command that calls this in OpenSimMain
  3505. /// </summary>
  3506. /// <param name="cmdparams"></param>
  3507. public void HandleEditCommand(string[] cmdparams)
  3508. {
  3509. m_log.Debug("Searching for Primitive: '" + cmdparams[2] + "'");
  3510. EntityBase[] entityList = GetEntities();
  3511. foreach (EntityBase ent in entityList)
  3512. {
  3513. if (ent is SceneObjectGroup)
  3514. {
  3515. SceneObjectPart part = ((SceneObjectGroup)ent).GetChildPart(((SceneObjectGroup)ent).UUID);
  3516. if (part != null)
  3517. {
  3518. if (part.Name == cmdparams[2])
  3519. {
  3520. part.Resize(
  3521. new Vector3(Convert.ToSingle(cmdparams[3]), Convert.ToSingle(cmdparams[4]),
  3522. Convert.ToSingle(cmdparams[5])));
  3523. m_log.Debug("Edited scale of Primitive: " + part.Name);
  3524. }
  3525. }
  3526. }
  3527. }
  3528. }
  3529. public override void Show(string[] showParams)
  3530. {
  3531. base.Show(showParams);
  3532. switch (showParams[0])
  3533. {
  3534. case "users":
  3535. m_log.Error("Current Region: " + RegionInfo.RegionName);
  3536. m_log.ErrorFormat("{0,-16}{1,-16}{2,-25}{3,-25}{4,-16}{5,-16}{6,-16}", "Firstname", "Lastname",
  3537. "Agent ID", "Session ID", "Circuit", "IP", "World");
  3538. ForEachScenePresence(delegate(ScenePresence sp)
  3539. {
  3540. m_log.ErrorFormat("{0,-16}{1,-16}{2,-25}{3,-25}{4,-16},{5,-16}{6,-16}",
  3541. sp.Firstname,
  3542. sp.Lastname,
  3543. sp.UUID,
  3544. sp.ControllingClient.AgentId,
  3545. "Unknown",
  3546. "Unknown",
  3547. RegionInfo.RegionName);
  3548. });
  3549. break;
  3550. }
  3551. }
  3552. #region Script Handling Methods
  3553. /// <summary>
  3554. /// Console command handler to send script command to script engine.
  3555. /// </summary>
  3556. /// <param name="args"></param>
  3557. public void SendCommandToPlugins(string[] args)
  3558. {
  3559. m_eventManager.TriggerOnPluginConsole(args);
  3560. }
  3561. public LandData GetLandData(float x, float y)
  3562. {
  3563. return LandChannel.GetLandObject(x, y).LandData;
  3564. }
  3565. public LandData GetLandData(uint x, uint y)
  3566. {
  3567. m_log.DebugFormat("[SCENE]: returning land for {0},{1}", x, y);
  3568. return LandChannel.GetLandObject((int)x, (int)y).LandData;
  3569. }
  3570. #endregion
  3571. #region Script Engine
  3572. private List<ScriptEngineInterface> ScriptEngines = new List<ScriptEngineInterface>();
  3573. public bool DumpAssetsToFile;
  3574. /// <summary>
  3575. ///
  3576. /// </summary>
  3577. /// <param name="scriptEngine"></param>
  3578. public void AddScriptEngine(ScriptEngineInterface scriptEngine)
  3579. {
  3580. ScriptEngines.Add(scriptEngine);
  3581. scriptEngine.InitializeEngine(this);
  3582. }
  3583. private bool ScriptDanger(SceneObjectPart part,Vector3 pos)
  3584. {
  3585. ILandObject parcel = LandChannel.GetLandObject(pos.X, pos.Y);
  3586. if (part != null)
  3587. {
  3588. if (parcel != null)
  3589. {
  3590. if ((parcel.LandData.Flags & (uint)ParcelFlags.AllowOtherScripts) != 0)
  3591. {
  3592. return true;
  3593. }
  3594. else if ((parcel.LandData.Flags & (uint)ParcelFlags.AllowGroupScripts) != 0)
  3595. {
  3596. if (part.OwnerID == parcel.LandData.OwnerID
  3597. || (parcel.LandData.IsGroupOwned && part.GroupID == parcel.LandData.GroupID)
  3598. || Permissions.IsGod(part.OwnerID))
  3599. {
  3600. return true;
  3601. }
  3602. else
  3603. {
  3604. return false;
  3605. }
  3606. }
  3607. else
  3608. {
  3609. if (part.OwnerID == parcel.LandData.OwnerID)
  3610. {
  3611. return true;
  3612. }
  3613. else
  3614. {
  3615. return false;
  3616. }
  3617. }
  3618. }
  3619. else
  3620. {
  3621. if (pos.X > 0f && pos.X < Constants.RegionSize && pos.Y > 0f && pos.Y < Constants.RegionSize)
  3622. {
  3623. // The only time parcel != null when an object is inside a region is when
  3624. // there is nothing behind the landchannel. IE, no land plugin loaded.
  3625. return true;
  3626. }
  3627. else
  3628. {
  3629. // The object is outside of this region. Stop piping events to it.
  3630. return false;
  3631. }
  3632. }
  3633. }
  3634. else
  3635. {
  3636. return false;
  3637. }
  3638. }
  3639. public bool ScriptDanger(uint localID, Vector3 pos)
  3640. {
  3641. SceneObjectPart part = GetSceneObjectPart(localID);
  3642. if (part != null)
  3643. {
  3644. return ScriptDanger(part, pos);
  3645. }
  3646. else
  3647. {
  3648. return false;
  3649. }
  3650. }
  3651. public bool PipeEventsForScript(uint localID)
  3652. {
  3653. SceneObjectPart part = GetSceneObjectPart(localID);
  3654. if (part != null)
  3655. {
  3656. // Changed so that child prims of attachments return ScriptDanger for their parent, so that
  3657. // their scripts will actually run.
  3658. // -- Leaf, Tue Aug 12 14:17:05 EDT 2008
  3659. SceneObjectPart parent = part.ParentGroup.RootPart;
  3660. if (parent != null && part.ParentGroup.IsAttachment)
  3661. return ScriptDanger(parent, parent.GetWorldPosition());
  3662. else
  3663. return ScriptDanger(part, part.GetWorldPosition());
  3664. }
  3665. else
  3666. {
  3667. return false;
  3668. }
  3669. }
  3670. #endregion
  3671. #region SceneGraph wrapper methods
  3672. /// <summary>
  3673. ///
  3674. /// </summary>
  3675. /// <param name="localID"></param>
  3676. /// <returns></returns>
  3677. public UUID ConvertLocalIDToFullID(uint localID)
  3678. {
  3679. return m_sceneGraph.ConvertLocalIDToFullID(localID);
  3680. }
  3681. public void SwapRootAgentCount(bool rootChildChildRootTF)
  3682. {
  3683. m_sceneGraph.SwapRootChildAgent(rootChildChildRootTF);
  3684. }
  3685. public void AddPhysicalPrim(int num)
  3686. {
  3687. m_sceneGraph.AddPhysicalPrim(num);
  3688. }
  3689. public void RemovePhysicalPrim(int num)
  3690. {
  3691. m_sceneGraph.RemovePhysicalPrim(num);
  3692. }
  3693. public int GetRootAgentCount()
  3694. {
  3695. return m_sceneGraph.GetRootAgentCount();
  3696. }
  3697. public int GetChildAgentCount()
  3698. {
  3699. return m_sceneGraph.GetChildAgentCount();
  3700. }
  3701. /// <summary>
  3702. /// Request a scene presence by UUID. Fast, indexed lookup.
  3703. /// </summary>
  3704. /// <param name="agentID"></param>
  3705. /// <returns>null if the presence was not found</returns>
  3706. public ScenePresence GetScenePresence(UUID agentID)
  3707. {
  3708. return m_sceneGraph.GetScenePresence(agentID);
  3709. }
  3710. /// <summary>
  3711. /// Request the scene presence by name.
  3712. /// </summary>
  3713. /// <param name="firstName"></param>
  3714. /// <param name="lastName"></param>
  3715. /// <returns>null if the presence was not found</returns>
  3716. public ScenePresence GetScenePresence(string firstName, string lastName)
  3717. {
  3718. return m_sceneGraph.GetScenePresence(firstName, lastName);
  3719. }
  3720. /// <summary>
  3721. /// Request the scene presence by localID.
  3722. /// </summary>
  3723. /// <param name="localID"></param>
  3724. /// <returns>null if the presence was not found</returns>
  3725. public ScenePresence GetScenePresence(uint localID)
  3726. {
  3727. return m_sceneGraph.GetScenePresence(localID);
  3728. }
  3729. public override bool PresenceChildStatus(UUID avatarID)
  3730. {
  3731. ScenePresence cp = GetScenePresence(avatarID);
  3732. // FIXME: This is really crap - some logout code is relying on a NullReferenceException to halt its processing
  3733. // This needs to be fixed properly by cleaning up the logout code.
  3734. //if (cp != null)
  3735. // return cp.IsChildAgent;
  3736. //return false;
  3737. return cp.IsChildAgent;
  3738. }
  3739. /// <summary>
  3740. /// Performs action on all scene presences.
  3741. /// </summary>
  3742. /// <param name="action"></param>
  3743. public void ForEachScenePresence(Action<ScenePresence> action)
  3744. {
  3745. if (m_sceneGraph != null)
  3746. {
  3747. m_sceneGraph.ForEachScenePresence(action);
  3748. }
  3749. }
  3750. /// <summary>
  3751. /// Perform the given action for each object
  3752. /// </summary>
  3753. /// <param name="action"></param>
  3754. // public void ForEachObject(Action<SceneObjectGroup> action)
  3755. // {
  3756. // List<SceneObjectGroup> presenceList;
  3757. //
  3758. // lock (m_sceneObjects)
  3759. // {
  3760. // presenceList = new List<SceneObjectGroup>(m_sceneObjects.Values);
  3761. // }
  3762. //
  3763. // foreach (SceneObjectGroup presence in presenceList)
  3764. // {
  3765. // action(presence);
  3766. // }
  3767. // }
  3768. /// <summary>
  3769. /// Get a group via its UUID
  3770. /// </summary>
  3771. /// <param name="fullID"></param>
  3772. /// <returns>null if no group with that name exists</returns>
  3773. public SceneObjectGroup GetSceneObjectGroup(UUID fullID)
  3774. {
  3775. return m_sceneGraph.GetSceneObjectGroup(fullID);
  3776. }
  3777. /// <summary>
  3778. /// Get a group by name from the scene (will return the first
  3779. /// found, if there are more than one prim with the same name)
  3780. /// </summary>
  3781. /// <param name="name"></param>
  3782. /// <returns>null if no group with that name exists</returns>
  3783. public SceneObjectGroup GetSceneObjectGroup(string name)
  3784. {
  3785. return m_sceneGraph.GetSceneObjectGroup(name);
  3786. }
  3787. /// <summary>
  3788. /// Get a prim by name from the scene (will return the first
  3789. /// found, if there are more than one prim with the same name)
  3790. /// </summary>
  3791. /// <param name="name"></param>
  3792. /// <returns></returns>
  3793. public SceneObjectPart GetSceneObjectPart(string name)
  3794. {
  3795. return m_sceneGraph.GetSceneObjectPart(name);
  3796. }
  3797. /// <summary>
  3798. /// Get a prim via its local id
  3799. /// </summary>
  3800. /// <param name="localID"></param>
  3801. /// <returns></returns>
  3802. public SceneObjectPart GetSceneObjectPart(uint localID)
  3803. {
  3804. return m_sceneGraph.GetSceneObjectPart(localID);
  3805. }
  3806. /// <summary>
  3807. /// Get a prim via its UUID
  3808. /// </summary>
  3809. /// <param name="fullID"></param>
  3810. /// <returns></returns>
  3811. public SceneObjectPart GetSceneObjectPart(UUID fullID)
  3812. {
  3813. return m_sceneGraph.GetSceneObjectPart(fullID);
  3814. }
  3815. /// <summary>
  3816. /// Get a scene object group that contains the prim with the given local id
  3817. /// </summary>
  3818. /// <param name="localID"></param>
  3819. /// <returns>null if no scene object group containing that prim is found</returns>
  3820. public SceneObjectGroup GetGroupByPrim(uint localID)
  3821. {
  3822. return m_sceneGraph.GetGroupByPrim(localID);
  3823. }
  3824. public override bool TryGetScenePresence(UUID avatarId, out ScenePresence avatar)
  3825. {
  3826. return m_sceneGraph.TryGetScenePresence(avatarId, out avatar);
  3827. }
  3828. public bool TryGetAvatarByName(string avatarName, out ScenePresence avatar)
  3829. {
  3830. return m_sceneGraph.TryGetAvatarByName(avatarName, out avatar);
  3831. }
  3832. public void ForEachClient(Action<IClientAPI> action)
  3833. {
  3834. m_clientManager.ForEachSync(action);
  3835. }
  3836. public bool TryGetClient(UUID avatarID, out IClientAPI client)
  3837. {
  3838. return m_clientManager.TryGetValue(avatarID, out client);
  3839. }
  3840. public bool TryGetClient(System.Net.IPEndPoint remoteEndPoint, out IClientAPI client)
  3841. {
  3842. return m_clientManager.TryGetValue(remoteEndPoint, out client);
  3843. }
  3844. public void ForEachSOG(Action<SceneObjectGroup> action)
  3845. {
  3846. m_sceneGraph.ForEachSOG(action);
  3847. }
  3848. /// <summary>
  3849. /// Returns a list of the entities in the scene. This is a new list so operations perform on the list itself
  3850. /// will not affect the original list of objects in the scene.
  3851. /// </summary>
  3852. /// <returns></returns>
  3853. public EntityBase[] GetEntities()
  3854. {
  3855. return m_sceneGraph.GetEntities();
  3856. }
  3857. #endregion
  3858. public void RegionHandleRequest(IClientAPI client, UUID regionID)
  3859. {
  3860. ulong handle = 0;
  3861. if (regionID == RegionInfo.RegionID)
  3862. handle = RegionInfo.RegionHandle;
  3863. else
  3864. {
  3865. GridRegion r = GridService.GetRegionByUUID(UUID.Zero, regionID);
  3866. if (r != null)
  3867. handle = r.RegionHandle;
  3868. }
  3869. if (handle != 0)
  3870. client.SendRegionHandle(regionID, handle);
  3871. }
  3872. public bool NeedSceneCacheClear(UUID agentID)
  3873. {
  3874. IInventoryTransferModule inv = RequestModuleInterface<IInventoryTransferModule>();
  3875. if (inv == null)
  3876. return true;
  3877. return inv.NeedSceneCacheClear(agentID, this);
  3878. }
  3879. public void CleanTempObjects()
  3880. {
  3881. EntityBase[] entities = GetEntities();
  3882. foreach (EntityBase obj in entities)
  3883. {
  3884. if (obj is SceneObjectGroup)
  3885. {
  3886. SceneObjectGroup grp = (SceneObjectGroup)obj;
  3887. if (!grp.IsDeleted)
  3888. {
  3889. if ((grp.RootPart.Flags & PrimFlags.TemporaryOnRez) != 0)
  3890. {
  3891. if (grp.RootPart.Expires <= DateTime.Now)
  3892. DeleteSceneObject(grp, false);
  3893. }
  3894. }
  3895. }
  3896. }
  3897. }
  3898. public void DeleteFromStorage(UUID uuid)
  3899. {
  3900. SimulationDataService.RemoveObject(uuid, m_regInfo.RegionID);
  3901. }
  3902. public int GetHealth()
  3903. {
  3904. // Returns:
  3905. // 1 = sim is up and accepting http requests. The heartbeat has
  3906. // stopped and the sim is probably locked up, but a remote
  3907. // admin restart may succeed
  3908. //
  3909. // 2 = Sim is up and the heartbeat is running. The sim is likely
  3910. // usable for people within and logins _may_ work
  3911. //
  3912. // 3 = We have seen a new user enter within the past 4 minutes
  3913. // which can be seen as positive confirmation of sim health
  3914. //
  3915. int health=1; // Start at 1, means we're up
  3916. if ((Util.EnvironmentTickCountSubtract(m_lastUpdate)) < 1000)
  3917. health+=1;
  3918. else
  3919. return health;
  3920. // A login in the last 4 mins? We can't be doing too badly
  3921. //
  3922. if ((Util.EnvironmentTickCountSubtract(m_LastLogin)) < 240000)
  3923. health++;
  3924. else
  3925. return health;
  3926. CheckHeartbeat();
  3927. return health;
  3928. }
  3929. // This callback allows the PhysicsScene to call back to its caller (the SceneGraph) and
  3930. // update non-physical objects like the joint proxy objects that represent the position
  3931. // of the joints in the scene.
  3932. // This routine is normally called from within a lock (OdeLock) from within the OdePhysicsScene
  3933. // WARNING: be careful of deadlocks here if you manipulate the scene. Remember you are being called
  3934. // from within the OdePhysicsScene.
  3935. protected internal void jointMoved(PhysicsJoint joint)
  3936. {
  3937. // m_parentScene.PhysicsScene.DumpJointInfo(); // non-thread-locked version; we should already be in a lock (OdeLock) when this callback is invoked
  3938. SceneObjectPart jointProxyObject = GetSceneObjectPart(joint.ObjectNameInScene);
  3939. if (jointProxyObject == null)
  3940. {
  3941. jointErrorMessage(joint, "WARNING, joint proxy not found, name " + joint.ObjectNameInScene);
  3942. return;
  3943. }
  3944. // now update the joint proxy object in the scene to have the position of the joint as returned by the physics engine
  3945. SceneObjectPart trackedBody = GetSceneObjectPart(joint.TrackedBodyName); // FIXME: causes a sequential lookup
  3946. if (trackedBody == null) return; // the actor may have been deleted but the joint still lingers around a few frames waiting for deletion. during this time, trackedBody is NULL to prevent further motion of the joint proxy.
  3947. jointProxyObject.Velocity = trackedBody.Velocity;
  3948. jointProxyObject.AngularVelocity = trackedBody.AngularVelocity;
  3949. switch (joint.Type)
  3950. {
  3951. case PhysicsJointType.Ball:
  3952. {
  3953. Vector3 jointAnchor = PhysicsScene.GetJointAnchor(joint);
  3954. Vector3 proxyPos = new Vector3(jointAnchor.X, jointAnchor.Y, jointAnchor.Z);
  3955. jointProxyObject.ParentGroup.UpdateGroupPosition(proxyPos); // schedules the entire group for a terse update
  3956. }
  3957. break;
  3958. case PhysicsJointType.Hinge:
  3959. {
  3960. Vector3 jointAnchor = PhysicsScene.GetJointAnchor(joint);
  3961. // Normally, we would just ask the physics scene to return the axis for the joint.
  3962. // Unfortunately, ODE sometimes returns <0,0,0> for the joint axis, which should
  3963. // never occur. Therefore we cannot rely on ODE to always return a correct joint axis.
  3964. // Therefore the following call does not always work:
  3965. //PhysicsVector phyJointAxis = _PhyScene.GetJointAxis(joint);
  3966. // instead we compute the joint orientation by saving the original joint orientation
  3967. // relative to one of the jointed bodies, and applying this transformation
  3968. // to the current position of the jointed bodies (the tracked body) to compute the
  3969. // current joint orientation.
  3970. if (joint.TrackedBodyName == null)
  3971. {
  3972. jointErrorMessage(joint, "joint.TrackedBodyName is null, joint " + joint.ObjectNameInScene);
  3973. }
  3974. Vector3 proxyPos = new Vector3(jointAnchor.X, jointAnchor.Y, jointAnchor.Z);
  3975. Quaternion q = trackedBody.RotationOffset * joint.LocalRotation;
  3976. jointProxyObject.ParentGroup.UpdateGroupPosition(proxyPos); // schedules the entire group for a terse update
  3977. jointProxyObject.ParentGroup.UpdateGroupRotationR(q); // schedules the entire group for a terse update
  3978. }
  3979. break;
  3980. }
  3981. }
  3982. // This callback allows the PhysicsScene to call back to its caller (the SceneGraph) and
  3983. // update non-physical objects like the joint proxy objects that represent the position
  3984. // of the joints in the scene.
  3985. // This routine is normally called from within a lock (OdeLock) from within the OdePhysicsScene
  3986. // WARNING: be careful of deadlocks here if you manipulate the scene. Remember you are being called
  3987. // from within the OdePhysicsScene.
  3988. protected internal void jointDeactivated(PhysicsJoint joint)
  3989. {
  3990. //m_log.Debug("[NINJA] SceneGraph.jointDeactivated, joint:" + joint.ObjectNameInScene);
  3991. SceneObjectPart jointProxyObject = GetSceneObjectPart(joint.ObjectNameInScene);
  3992. if (jointProxyObject == null)
  3993. {
  3994. jointErrorMessage(joint, "WARNING, trying to deactivate (stop interpolation of) joint proxy, but not found, name " + joint.ObjectNameInScene);
  3995. return;
  3996. }
  3997. // turn the proxy non-physical, which also stops its client-side interpolation
  3998. bool wasUsingPhysics = ((jointProxyObject.Flags & PrimFlags.Physics) != 0);
  3999. if (wasUsingPhysics)
  4000. {
  4001. jointProxyObject.UpdatePrimFlags(false, false, true, false); // FIXME: possible deadlock here; check to make sure all the scene alterations set into motion here won't deadlock
  4002. }
  4003. }
  4004. // This callback allows the PhysicsScene to call back to its caller (the SceneGraph) and
  4005. // alert the user of errors by using the debug channel in the same way that scripts alert
  4006. // the user of compile errors.
  4007. // This routine is normally called from within a lock (OdeLock) from within the OdePhysicsScene
  4008. // WARNING: be careful of deadlocks here if you manipulate the scene. Remember you are being called
  4009. // from within the OdePhysicsScene.
  4010. public void jointErrorMessage(PhysicsJoint joint, string message)
  4011. {
  4012. if (joint != null)
  4013. {
  4014. if (joint.ErrorMessageCount > PhysicsJoint.maxErrorMessages)
  4015. return;
  4016. SceneObjectPart jointProxyObject = GetSceneObjectPart(joint.ObjectNameInScene);
  4017. if (jointProxyObject != null)
  4018. {
  4019. SimChat(Utils.StringToBytes("[NINJA]: " + message),
  4020. ChatTypeEnum.DebugChannel,
  4021. 2147483647,
  4022. jointProxyObject.AbsolutePosition,
  4023. jointProxyObject.Name,
  4024. jointProxyObject.UUID,
  4025. false);
  4026. joint.ErrorMessageCount++;
  4027. if (joint.ErrorMessageCount > PhysicsJoint.maxErrorMessages)
  4028. {
  4029. SimChat(Utils.StringToBytes("[NINJA]: Too many messages for this joint, suppressing further messages."),
  4030. ChatTypeEnum.DebugChannel,
  4031. 2147483647,
  4032. jointProxyObject.AbsolutePosition,
  4033. jointProxyObject.Name,
  4034. jointProxyObject.UUID,
  4035. false);
  4036. }
  4037. }
  4038. else
  4039. {
  4040. // couldn't find the joint proxy object; the error message is silently suppressed
  4041. }
  4042. }
  4043. }
  4044. public Scene ConsoleScene()
  4045. {
  4046. if (MainConsole.Instance == null)
  4047. return null;
  4048. if (MainConsole.Instance.ConsoleScene is Scene)
  4049. return (Scene)MainConsole.Instance.ConsoleScene;
  4050. return null;
  4051. }
  4052. public float GetGroundHeight(float x, float y)
  4053. {
  4054. if (x < 0)
  4055. x = 0;
  4056. if (x >= Heightmap.Width)
  4057. x = Heightmap.Width - 1;
  4058. if (y < 0)
  4059. y = 0;
  4060. if (y >= Heightmap.Height)
  4061. y = Heightmap.Height - 1;
  4062. Vector3 p0 = new Vector3(x, y, (float)Heightmap[(int)x, (int)y]);
  4063. Vector3 p1 = new Vector3(p0);
  4064. Vector3 p2 = new Vector3(p0);
  4065. p1.X += 1.0f;
  4066. if (p1.X < Heightmap.Width)
  4067. p1.Z = (float)Heightmap[(int)p1.X, (int)p1.Y];
  4068. p2.Y += 1.0f;
  4069. if (p2.Y < Heightmap.Height)
  4070. p2.Z = (float)Heightmap[(int)p2.X, (int)p2.Y];
  4071. Vector3 v0 = new Vector3(p1.X - p0.X, p1.Y - p0.Y, p1.Z - p0.Z);
  4072. Vector3 v1 = new Vector3(p2.X - p0.X, p2.Y - p0.Y, p2.Z - p0.Z);
  4073. v0.Normalize();
  4074. v1.Normalize();
  4075. Vector3 vsn = new Vector3();
  4076. vsn.X = (v0.Y * v1.Z) - (v0.Z * v1.Y);
  4077. vsn.Y = (v0.Z * v1.X) - (v0.X * v1.Z);
  4078. vsn.Z = (v0.X * v1.Y) - (v0.Y * v1.X);
  4079. vsn.Normalize();
  4080. float xdiff = x - (float)((int)x);
  4081. float ydiff = y - (float)((int)y);
  4082. return (((vsn.X * xdiff) + (vsn.Y * ydiff)) / (-1 * vsn.Z)) + p0.Z;
  4083. }
  4084. private void CheckHeartbeat()
  4085. {
  4086. if (m_firstHeartbeat)
  4087. return;
  4088. if (Util.EnvironmentTickCountSubtract(m_lastUpdate) > 2000)
  4089. StartTimer();
  4090. }
  4091. public override ISceneObject DeserializeObject(string representation)
  4092. {
  4093. return SceneObjectSerializer.FromXml2Format(representation);
  4094. }
  4095. public override bool AllowScriptCrossings
  4096. {
  4097. get { return m_allowScriptCrossings; }
  4098. }
  4099. public Vector3? GetNearestAllowedPosition(ScenePresence avatar)
  4100. {
  4101. //simulate to make sure we have pretty up to date positions
  4102. PhysicsScene.Simulate(0);
  4103. ILandObject nearestParcel = GetNearestAllowedParcel(avatar.UUID, avatar.AbsolutePosition.X, avatar.AbsolutePosition.Y);
  4104. if (nearestParcel != null)
  4105. {
  4106. Vector3 dir = Vector3.Normalize(Vector3.Multiply(avatar.Velocity, -1));
  4107. //Try to get a location that feels like where they came from
  4108. Vector3? nearestPoint = GetNearestPointInParcelAlongDirectionFromPoint(avatar.AbsolutePosition, dir, nearestParcel);
  4109. if (nearestPoint != null)
  4110. {
  4111. Debug.WriteLine("Found a sane previous position based on velocity, sending them to: " + nearestPoint.ToString());
  4112. return nearestPoint.Value;
  4113. }
  4114. //Sometimes velocity might be zero (local teleport), so try finding point along path from avatar to center of nearest parcel
  4115. Vector3 directionToParcelCenter = Vector3.Subtract(GetParcelCenterAtGround(nearestParcel), avatar.AbsolutePosition);
  4116. dir = Vector3.Normalize(directionToParcelCenter);
  4117. nearestPoint = GetNearestPointInParcelAlongDirectionFromPoint(avatar.AbsolutePosition, dir, nearestParcel);
  4118. if (nearestPoint != null)
  4119. {
  4120. Debug.WriteLine("They had a zero velocity, sending them to: " + nearestPoint.ToString());
  4121. return nearestPoint.Value;
  4122. }
  4123. //Ultimate backup if we have no idea where they are
  4124. Debug.WriteLine("Have no idea where they are, sending them to: " + avatar.lastKnownAllowedPosition.ToString());
  4125. return avatar.lastKnownAllowedPosition;
  4126. }
  4127. //Go to the edge, this happens in teleporting to a region with no available parcels
  4128. Vector3 nearestRegionEdgePoint = GetNearestRegionEdgePosition(avatar);
  4129. //Debug.WriteLine("They are really in a place they don't belong, sending them to: " + nearestRegionEdgePoint.ToString());
  4130. return nearestRegionEdgePoint;
  4131. }
  4132. private Vector3 GetParcelCenterAtGround(ILandObject parcel)
  4133. {
  4134. Vector2 center = GetParcelCenter(parcel);
  4135. return GetPositionAtGround(center.X, center.Y);
  4136. }
  4137. private Vector3? GetNearestPointInParcelAlongDirectionFromPoint(Vector3 pos, Vector3 direction, ILandObject parcel)
  4138. {
  4139. Vector3 unitDirection = Vector3.Normalize(direction);
  4140. //Making distance to search go through some sane limit of distance
  4141. for (float distance = 0; distance < Constants.RegionSize * 2; distance += .5f)
  4142. {
  4143. Vector3 testPos = Vector3.Add(pos, Vector3.Multiply(unitDirection, distance));
  4144. if (parcel.ContainsPoint((int)testPos.X, (int)testPos.Y))
  4145. {
  4146. return testPos;
  4147. }
  4148. }
  4149. return null;
  4150. }
  4151. public ILandObject GetNearestAllowedParcel(UUID avatarId, float x, float y)
  4152. {
  4153. List<ILandObject> all = AllParcels();
  4154. float minParcelDistance = float.MaxValue;
  4155. ILandObject nearestParcel = null;
  4156. foreach (var parcel in all)
  4157. {
  4158. if (!parcel.IsEitherBannedOrRestricted(avatarId))
  4159. {
  4160. float parcelDistance = GetParcelDistancefromPoint(parcel, x, y);
  4161. if (parcelDistance < minParcelDistance)
  4162. {
  4163. minParcelDistance = parcelDistance;
  4164. nearestParcel = parcel;
  4165. }
  4166. }
  4167. }
  4168. return nearestParcel;
  4169. }
  4170. private List<ILandObject> AllParcels()
  4171. {
  4172. return LandChannel.AllParcels();
  4173. }
  4174. private float GetParcelDistancefromPoint(ILandObject parcel, float x, float y)
  4175. {
  4176. return Vector2.Distance(new Vector2(x, y), GetParcelCenter(parcel));
  4177. }
  4178. //calculate the average center point of a parcel
  4179. private Vector2 GetParcelCenter(ILandObject parcel)
  4180. {
  4181. int count = 0;
  4182. int avgx = 0;
  4183. int avgy = 0;
  4184. for (int x = 0; x < Constants.RegionSize; x++)
  4185. {
  4186. for (int y = 0; y < Constants.RegionSize; y++)
  4187. {
  4188. //Just keep a running average as we check if all the points are inside or not
  4189. if (parcel.ContainsPoint(x, y))
  4190. {
  4191. if (count == 0)
  4192. {
  4193. avgx = x;
  4194. avgy = y;
  4195. }
  4196. else
  4197. {
  4198. avgx = (avgx * count + x) / (count + 1);
  4199. avgy = (avgy * count + y) / (count + 1);
  4200. }
  4201. count += 1;
  4202. }
  4203. }
  4204. }
  4205. return new Vector2(avgx, avgy);
  4206. }
  4207. private Vector3 GetNearestRegionEdgePosition(ScenePresence avatar)
  4208. {
  4209. float xdistance = avatar.AbsolutePosition.X < Constants.RegionSize / 2 ? avatar.AbsolutePosition.X : Constants.RegionSize - avatar.AbsolutePosition.X;
  4210. float ydistance = avatar.AbsolutePosition.Y < Constants.RegionSize / 2 ? avatar.AbsolutePosition.Y : Constants.RegionSize - avatar.AbsolutePosition.Y;
  4211. //find out what vertical edge to go to
  4212. if (xdistance < ydistance)
  4213. {
  4214. if (avatar.AbsolutePosition.X < Constants.RegionSize / 2)
  4215. {
  4216. return GetPositionAtAvatarHeightOrGroundHeight(avatar, 0.0f, avatar.AbsolutePosition.Y);
  4217. }
  4218. else
  4219. {
  4220. return GetPositionAtAvatarHeightOrGroundHeight(avatar, Constants.RegionSize, avatar.AbsolutePosition.Y);
  4221. }
  4222. }
  4223. //find out what horizontal edge to go to
  4224. else
  4225. {
  4226. if (avatar.AbsolutePosition.Y < Constants.RegionSize / 2)
  4227. {
  4228. return GetPositionAtAvatarHeightOrGroundHeight(avatar, avatar.AbsolutePosition.X, 0.0f);
  4229. }
  4230. else
  4231. {
  4232. return GetPositionAtAvatarHeightOrGroundHeight(avatar, avatar.AbsolutePosition.X, Constants.RegionSize);
  4233. }
  4234. }
  4235. }
  4236. private Vector3 GetPositionAtAvatarHeightOrGroundHeight(ScenePresence avatar, float x, float y)
  4237. {
  4238. Vector3 ground = GetPositionAtGround(x, y);
  4239. if (avatar.AbsolutePosition.Z > ground.Z)
  4240. {
  4241. ground.Z = avatar.AbsolutePosition.Z;
  4242. }
  4243. return ground;
  4244. }
  4245. private Vector3 GetPositionAtGround(float x, float y)
  4246. {
  4247. return new Vector3(x, y, GetGroundHeight(x, y));
  4248. }
  4249. public List<UUID> GetEstateRegions(int estateID)
  4250. {
  4251. IEstateDataService estateDataService = EstateDataService;
  4252. if (estateDataService == null)
  4253. return new List<UUID>(0);
  4254. return estateDataService.GetRegions(estateID);
  4255. }
  4256. public void ReloadEstateData()
  4257. {
  4258. IEstateDataService estateDataService = EstateDataService;
  4259. if (estateDataService != null)
  4260. {
  4261. m_regInfo.EstateSettings = estateDataService.LoadEstateSettings(m_regInfo.RegionID, false);
  4262. TriggerEstateSunUpdate();
  4263. }
  4264. }
  4265. public void TriggerEstateSunUpdate()
  4266. {
  4267. float sun;
  4268. if (RegionInfo.RegionSettings.UseEstateSun)
  4269. {
  4270. sun = (float)RegionInfo.EstateSettings.SunPosition;
  4271. if (RegionInfo.EstateSettings.UseGlobalTime)
  4272. {
  4273. sun = EventManager.GetCurrentTimeAsSunLindenHour() - 6.0f;
  4274. }
  4275. //
  4276. EventManager.TriggerEstateToolsSunUpdate(
  4277. RegionInfo.RegionHandle,
  4278. RegionInfo.EstateSettings.FixedSun,
  4279. RegionInfo.RegionSettings.UseEstateSun,
  4280. sun);
  4281. }
  4282. else
  4283. {
  4284. // Use the Sun Position from the Region Settings
  4285. sun = (float)RegionInfo.RegionSettings.SunPosition - 6.0f;
  4286. EventManager.TriggerEstateToolsSunUpdate(
  4287. RegionInfo.RegionHandle,
  4288. RegionInfo.RegionSettings.FixedSun,
  4289. RegionInfo.RegionSettings.UseEstateSun,
  4290. sun);
  4291. }
  4292. }
  4293. private void HandleDeleteObject(string module, string[] cmd)
  4294. {
  4295. if (cmd.Length < 3)
  4296. return;
  4297. string mode = cmd[2];
  4298. string o = "";
  4299. if (mode != "outside")
  4300. {
  4301. if (cmd.Length < 4)
  4302. return;
  4303. o = cmd[3];
  4304. }
  4305. List<SceneObjectGroup> deletes = new List<SceneObjectGroup>();
  4306. UUID match;
  4307. switch (mode)
  4308. {
  4309. case "owner":
  4310. if (!UUID.TryParse(o, out match))
  4311. return;
  4312. ForEachSOG(delegate (SceneObjectGroup g)
  4313. {
  4314. if (g.OwnerID == match && !g.IsAttachment)
  4315. deletes.Add(g);
  4316. });
  4317. break;
  4318. case "creator":
  4319. if (!UUID.TryParse(o, out match))
  4320. return;
  4321. ForEachSOG(delegate (SceneObjectGroup g)
  4322. {
  4323. if (g.RootPart.CreatorID == match && !g.IsAttachment)
  4324. deletes.Add(g);
  4325. });
  4326. break;
  4327. case "uuid":
  4328. if (!UUID.TryParse(o, out match))
  4329. return;
  4330. ForEachSOG(delegate (SceneObjectGroup g)
  4331. {
  4332. if (g.UUID == match && !g.IsAttachment)
  4333. deletes.Add(g);
  4334. });
  4335. break;
  4336. case "name":
  4337. ForEachSOG(delegate (SceneObjectGroup g)
  4338. {
  4339. if (g.RootPart.Name == o && !g.IsAttachment)
  4340. deletes.Add(g);
  4341. });
  4342. break;
  4343. case "outside":
  4344. ForEachSOG(delegate (SceneObjectGroup g)
  4345. {
  4346. SceneObjectPart rootPart = g.RootPart;
  4347. bool delete = false;
  4348. if (rootPart.GroupPosition.Z < 0.0 || rootPart.GroupPosition.Z > 10000.0)
  4349. {
  4350. delete = true;
  4351. } else {
  4352. ILandObject parcel = LandChannel.GetLandObject(rootPart.GroupPosition.X, rootPart.GroupPosition.Y);
  4353. if (parcel == null || parcel.LandData.Name == "NO LAND")
  4354. delete = true;
  4355. }
  4356. if (delete && !g.IsAttachment && !deletes.Contains(g))
  4357. deletes.Add(g);
  4358. });
  4359. break;
  4360. }
  4361. foreach (SceneObjectGroup g in deletes)
  4362. {
  4363. m_log.InfoFormat("[SCENE]: Deleting object {0}", g.UUID);
  4364. DeleteSceneObject(g, false);
  4365. }
  4366. }
  4367. private void HandleReloadEstate(string module, string[] cmd)
  4368. {
  4369. if (MainConsole.Instance.ConsoleScene == null ||
  4370. (MainConsole.Instance.ConsoleScene is Scene &&
  4371. (Scene)MainConsole.Instance.ConsoleScene == this))
  4372. {
  4373. ReloadEstateData();
  4374. }
  4375. }
  4376. /// <summary>
  4377. /// Get the volume of space that will encompass all the given objects.
  4378. /// </summary>
  4379. /// <param name="objects"></param>
  4380. /// <param name="minX"></param>
  4381. /// <param name="maxX"></param>
  4382. /// <param name="minY"></param>
  4383. /// <param name="maxY"></param>
  4384. /// <param name="minZ"></param>
  4385. /// <param name="maxZ"></param>
  4386. /// <returns></returns>
  4387. public static Vector3[] GetCombinedBoundingBox(
  4388. List<SceneObjectGroup> objects,
  4389. out float minX, out float maxX, out float minY, out float maxY, out float minZ, out float maxZ)
  4390. {
  4391. minX = 256;
  4392. maxX = -256;
  4393. minY = 256;
  4394. maxY = -256;
  4395. minZ = 8192;
  4396. maxZ = -256;
  4397. List<Vector3> offsets = new List<Vector3>();
  4398. foreach (SceneObjectGroup g in objects)
  4399. {
  4400. float ominX, ominY, ominZ, omaxX, omaxY, omaxZ;
  4401. Vector3 vec = g.AbsolutePosition;
  4402. g.GetAxisAlignedBoundingBoxRaw(out ominX, out omaxX, out ominY, out omaxY, out ominZ, out omaxZ);
  4403. // m_log.DebugFormat(
  4404. // "[SCENE]: For {0} found AxisAlignedBoundingBoxRaw {1}, {2}",
  4405. // g.Name, new Vector3(ominX, ominY, ominZ), new Vector3(omaxX, omaxY, omaxZ));
  4406. ominX += vec.X;
  4407. omaxX += vec.X;
  4408. ominY += vec.Y;
  4409. omaxY += vec.Y;
  4410. ominZ += vec.Z;
  4411. omaxZ += vec.Z;
  4412. if (minX > ominX)
  4413. minX = ominX;
  4414. if (minY > ominY)
  4415. minY = ominY;
  4416. if (minZ > ominZ)
  4417. minZ = ominZ;
  4418. if (maxX < omaxX)
  4419. maxX = omaxX;
  4420. if (maxY < omaxY)
  4421. maxY = omaxY;
  4422. if (maxZ < omaxZ)
  4423. maxZ = omaxZ;
  4424. }
  4425. foreach (SceneObjectGroup g in objects)
  4426. {
  4427. Vector3 vec = g.AbsolutePosition;
  4428. vec.X -= minX;
  4429. vec.Y -= minY;
  4430. vec.Z -= minZ;
  4431. offsets.Add(vec);
  4432. }
  4433. return offsets.ToArray();
  4434. }
  4435. public void RegenerateMaptile(object sender, ElapsedEventArgs e)
  4436. {
  4437. IWorldMapModule mapModule = RequestModuleInterface<IWorldMapModule>();
  4438. if (mapModule != null)
  4439. {
  4440. mapModule.GenerateMaptile();
  4441. string error = GridService.RegisterRegion(RegionInfo.ScopeID, new GridRegion(RegionInfo));
  4442. if (error != String.Empty)
  4443. throw new Exception(error);
  4444. }
  4445. }
  4446. public void CleanDroppedAttachments()
  4447. {
  4448. List<SceneObjectGroup> objectsToDelete =
  4449. new List<SceneObjectGroup>();
  4450. lock (m_cleaningAttachments)
  4451. {
  4452. ForEachSOG(delegate (SceneObjectGroup grp)
  4453. {
  4454. if (grp.RootPart.Shape.PCode == 0 && grp.RootPart.Shape.State != 0 && (!objectsToDelete.Contains(grp)))
  4455. {
  4456. UUID agentID = grp.OwnerID;
  4457. if (agentID == UUID.Zero)
  4458. {
  4459. objectsToDelete.Add(grp);
  4460. return;
  4461. }
  4462. ScenePresence sp = GetScenePresence(agentID);
  4463. if (sp == null)
  4464. {
  4465. objectsToDelete.Add(grp);
  4466. return;
  4467. }
  4468. }
  4469. });
  4470. }
  4471. foreach (SceneObjectGroup grp in objectsToDelete)
  4472. {
  4473. m_log.InfoFormat("[SCENE]: Deleting dropped attachment {0} of user {1}", grp.UUID, grp.OwnerID);
  4474. DeleteSceneObject(grp, true);
  4475. }
  4476. }
  4477. // This method is called across the simulation connector to
  4478. // determine if a given agent is allowed in this region
  4479. // AS A ROOT AGENT. Returning false here will prevent them
  4480. // from logging into the region, teleporting into the region
  4481. // or corssing the broder walking, but will NOT prevent
  4482. // child agent creation, thereby emulating the SL behavior.
  4483. public bool QueryAccess(UUID agentID, Vector3 position, out string reason)
  4484. {
  4485. int num = m_sceneGraph.GetNumberOfScenePresences();
  4486. if (num >= RegionInfo.RegionSettings.AgentLimit)
  4487. {
  4488. if (!Permissions.IsAdministrator(agentID))
  4489. {
  4490. reason = "The region is full";
  4491. return false;
  4492. }
  4493. }
  4494. reason = String.Empty;
  4495. return true;
  4496. }
  4497. }
  4498. }