1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690769176927693769476957696769776987699770077017702770377047705770677077708770977107711771277137714771577167717771877197720772177227723772477257726772777287729773077317732773377347735773677377738773977407741774277437744774577467747774877497750775177527753775477557756775777587759776077617762776377647765776677677768776977707771777277737774777577767777777877797780778177827783778477857786778777887789779077917792779377947795779677977798779978007801780278037804780578067807780878097810781178127813781478157816781778187819782078217822782378247825782678277828782978307831783278337834783578367837783878397840784178427843784478457846784778487849785078517852785378547855785678577858785978607861786278637864786578667867786878697870787178727873787478757876787778787879788078817882788378847885788678877888788978907891789278937894789578967897789878997900790179027903790479057906790779087909791079117912791379147915791679177918791979207921792279237924792579267927792879297930793179327933793479357936793779387939794079417942794379447945794679477948794979507951795279537954795579567957795879597960796179627963796479657966796779687969797079717972797379747975797679777978797979807981798279837984798579867987798879897990799179927993799479957996799779987999800080018002800380048005800680078008800980108011801280138014801580168017801880198020802180228023802480258026802780288029803080318032803380348035803680378038803980408041804280438044804580468047804880498050805180528053805480558056805780588059806080618062806380648065806680678068806980708071807280738074807580768077807880798080808180828083808480858086808780888089809080918092809380948095809680978098809981008101810281038104810581068107810881098110811181128113811481158116811781188119812081218122812381248125812681278128812981308131813281338134813581368137813881398140814181428143814481458146814781488149815081518152815381548155815681578158815981608161816281638164816581668167816881698170817181728173817481758176817781788179818081818182818381848185818681878188818981908191819281938194819581968197819881998200820182028203820482058206820782088209821082118212821382148215821682178218821982208221822282238224822582268227822882298230823182328233823482358236823782388239824082418242824382448245824682478248824982508251825282538254825582568257825882598260826182628263826482658266826782688269827082718272827382748275827682778278827982808281828282838284828582868287828882898290829182928293829482958296829782988299830083018302830383048305830683078308830983108311831283138314831583168317831883198320832183228323832483258326832783288329833083318332833383348335833683378338833983408341834283438344834583468347834883498350835183528353835483558356835783588359836083618362836383648365836683678368836983708371837283738374837583768377837883798380838183828383838483858386838783888389839083918392839383948395839683978398839984008401840284038404840584068407840884098410841184128413841484158416841784188419842084218422842384248425842684278428842984308431843284338434843584368437843884398440844184428443844484458446844784488449845084518452845384548455845684578458845984608461846284638464846584668467846884698470847184728473847484758476847784788479848084818482848384848485848684878488848984908491849284938494849584968497849884998500850185028503850485058506850785088509851085118512851385148515851685178518851985208521852285238524852585268527852885298530853185328533853485358536853785388539854085418542854385448545854685478548854985508551855285538554855585568557855885598560856185628563856485658566856785688569857085718572857385748575857685778578857985808581858285838584858585868587858885898590859185928593859485958596859785988599860086018602860386048605860686078608860986108611861286138614861586168617861886198620862186228623862486258626862786288629863086318632863386348635863686378638863986408641864286438644864586468647864886498650865186528653865486558656865786588659866086618662866386648665866686678668866986708671867286738674867586768677867886798680868186828683868486858686868786888689869086918692869386948695869686978698869987008701870287038704870587068707870887098710871187128713871487158716871787188719872087218722872387248725872687278728872987308731873287338734873587368737873887398740874187428743874487458746874787488749875087518752875387548755875687578758875987608761876287638764876587668767876887698770877187728773877487758776877787788779878087818782878387848785878687878788878987908791879287938794879587968797879887998800880188028803880488058806880788088809881088118812881388148815881688178818881988208821882288238824882588268827882888298830883188328833883488358836883788388839884088418842884388448845884688478848884988508851885288538854885588568857885888598860886188628863886488658866886788688869887088718872887388748875887688778878887988808881888288838884888588868887888888898890889188928893889488958896889788988899890089018902890389048905890689078908890989108911891289138914891589168917891889198920892189228923892489258926892789288929893089318932893389348935893689378938893989408941894289438944894589468947894889498950895189528953895489558956895789588959896089618962896389648965896689678968896989708971897289738974897589768977897889798980898189828983898489858986898789888989899089918992899389948995899689978998899990009001900290039004900590069007900890099010901190129013901490159016901790189019902090219022902390249025902690279028902990309031903290339034903590369037903890399040904190429043904490459046904790489049905090519052905390549055905690579058905990609061906290639064906590669067906890699070907190729073907490759076907790789079908090819082908390849085908690879088908990909091909290939094909590969097909890999100910191029103910491059106910791089109911091119112911391149115911691179118911991209121912291239124912591269127912891299130913191329133913491359136913791389139914091419142914391449145914691479148914991509151915291539154915591569157915891599160916191629163916491659166916791689169917091719172917391749175917691779178917991809181918291839184918591869187918891899190919191929193919491959196919791989199920092019202920392049205920692079208920992109211921292139214921592169217921892199220922192229223922492259226922792289229923092319232923392349235923692379238923992409241924292439244924592469247924892499250925192529253925492559256925792589259926092619262926392649265926692679268926992709271927292739274927592769277927892799280928192829283928492859286928792889289929092919292929392949295929692979298929993009301930293039304930593069307930893099310931193129313931493159316931793189319932093219322932393249325932693279328932993309331933293339334933593369337933893399340934193429343934493459346934793489349935093519352935393549355935693579358935993609361936293639364936593669367936893699370937193729373937493759376937793789379938093819382938393849385938693879388938993909391939293939394939593969397939893999400940194029403940494059406940794089409941094119412941394149415941694179418941994209421942294239424942594269427942894299430943194329433943494359436943794389439944094419442944394449445944694479448944994509451945294539454945594569457945894599460946194629463946494659466946794689469947094719472947394749475947694779478947994809481948294839484948594869487948894899490949194929493949494959496949794989499950095019502950395049505950695079508950995109511951295139514951595169517951895199520952195229523952495259526952795289529953095319532953395349535953695379538953995409541954295439544954595469547954895499550955195529553955495559556955795589559956095619562956395649565956695679568956995709571957295739574957595769577957895799580958195829583958495859586958795889589959095919592959395949595959695979598959996009601960296039604960596069607960896099610961196129613961496159616961796189619962096219622962396249625962696279628962996309631963296339634963596369637963896399640964196429643964496459646964796489649965096519652965396549655965696579658965996609661966296639664966596669667966896699670967196729673967496759676967796789679968096819682968396849685968696879688968996909691969296939694969596969697969896999700970197029703970497059706970797089709971097119712971397149715971697179718971997209721972297239724972597269727972897299730973197329733973497359736973797389739974097419742974397449745974697479748974997509751975297539754975597569757975897599760976197629763976497659766976797689769977097719772977397749775977697779778977997809781978297839784978597869787978897899790979197929793979497959796979797989799980098019802980398049805980698079808980998109811981298139814981598169817981898199820982198229823982498259826982798289829983098319832983398349835983698379838983998409841984298439844984598469847984898499850985198529853985498559856985798589859986098619862986398649865986698679868986998709871987298739874987598769877987898799880988198829883988498859886988798889889989098919892989398949895989698979898989999009901990299039904990599069907990899099910991199129913991499159916991799189919992099219922992399249925992699279928992999309931993299339934993599369937993899399940994199429943994499459946994799489949995099519952995399549955995699579958995999609961996299639964996599669967996899699970997199729973997499759976997799789979998099819982998399849985998699879988998999909991999299939994999599969997999899991000010001100021000310004100051000610007100081000910010100111001210013100141001510016100171001810019100201002110022100231002410025100261002710028100291003010031100321003310034100351003610037100381003910040100411004210043100441004510046100471004810049100501005110052100531005410055100561005710058100591006010061100621006310064100651006610067100681006910070100711007210073100741007510076100771007810079100801008110082100831008410085100861008710088100891009010091100921009310094100951009610097100981009910100101011010210103101041010510106101071010810109101101011110112101131011410115101161011710118101191012010121101221012310124101251012610127101281012910130101311013210133101341013510136101371013810139101401014110142101431014410145101461014710148101491015010151101521015310154101551015610157101581015910160101611016210163101641016510166101671016810169101701017110172101731017410175101761017710178101791018010181101821018310184101851018610187101881018910190101911019210193101941019510196101971019810199102001020110202102031020410205102061020710208102091021010211102121021310214102151021610217102181021910220102211022210223102241022510226102271022810229102301023110232102331023410235102361023710238102391024010241102421024310244102451024610247102481024910250102511025210253102541025510256102571025810259102601026110262102631026410265102661026710268102691027010271102721027310274102751027610277102781027910280102811028210283102841028510286102871028810289102901029110292102931029410295102961029710298102991030010301103021030310304103051030610307103081030910310103111031210313103141031510316103171031810319103201032110322103231032410325103261032710328103291033010331103321033310334103351033610337103381033910340103411034210343103441034510346103471034810349103501035110352103531035410355103561035710358103591036010361103621036310364103651036610367103681036910370103711037210373103741037510376103771037810379103801038110382103831038410385103861038710388103891039010391103921039310394103951039610397103981039910400104011040210403104041040510406104071040810409104101041110412104131041410415104161041710418104191042010421104221042310424104251042610427104281042910430104311043210433104341043510436104371043810439104401044110442104431044410445104461044710448104491045010451104521045310454104551045610457104581045910460104611046210463104641046510466104671046810469104701047110472104731047410475104761047710478104791048010481104821048310484104851048610487104881048910490104911049210493104941049510496104971049810499105001050110502105031050410505105061050710508105091051010511105121051310514105151051610517105181051910520105211052210523105241052510526105271052810529105301053110532105331053410535105361053710538105391054010541105421054310544105451054610547105481054910550105511055210553105541055510556105571055810559105601056110562105631056410565105661056710568105691057010571105721057310574105751057610577105781057910580105811058210583105841058510586105871058810589105901059110592105931059410595105961059710598105991060010601106021060310604106051060610607106081060910610106111061210613106141061510616106171061810619106201062110622106231062410625106261062710628106291063010631106321063310634106351063610637106381063910640106411064210643106441064510646106471064810649106501065110652106531065410655106561065710658106591066010661106621066310664106651066610667106681066910670106711067210673106741067510676106771067810679106801068110682106831068410685106861068710688106891069010691106921069310694106951069610697106981069910700107011070210703107041070510706107071070810709107101071110712107131071410715107161071710718107191072010721107221072310724107251072610727107281072910730107311073210733107341073510736107371073810739107401074110742107431074410745107461074710748107491075010751107521075310754107551075610757107581075910760107611076210763107641076510766107671076810769107701077110772107731077410775107761077710778107791078010781107821078310784107851078610787107881078910790107911079210793107941079510796107971079810799108001080110802108031080410805108061080710808108091081010811108121081310814108151081610817108181081910820108211082210823108241082510826108271082810829108301083110832108331083410835108361083710838108391084010841108421084310844108451084610847108481084910850108511085210853108541085510856108571085810859108601086110862108631086410865108661086710868108691087010871108721087310874108751087610877108781087910880108811088210883108841088510886108871088810889108901089110892108931089410895108961089710898108991090010901109021090310904109051090610907109081090910910109111091210913109141091510916109171091810919109201092110922109231092410925109261092710928109291093010931109321093310934109351093610937109381093910940109411094210943109441094510946109471094810949109501095110952109531095410955109561095710958109591096010961109621096310964109651096610967109681096910970109711097210973109741097510976109771097810979109801098110982109831098410985109861098710988109891099010991109921099310994109951099610997109981099911000110011100211003110041100511006110071100811009110101101111012110131101411015110161101711018110191102011021110221102311024110251102611027110281102911030110311103211033110341103511036110371103811039110401104111042110431104411045110461104711048110491105011051110521105311054110551105611057110581105911060110611106211063110641106511066110671106811069110701107111072110731107411075110761107711078110791108011081110821108311084110851108611087110881108911090110911109211093110941109511096110971109811099111001110111102111031110411105111061110711108111091111011111111121111311114111151111611117111181111911120111211112211123111241112511126111271112811129111301113111132111331113411135111361113711138111391114011141111421114311144111451114611147111481114911150111511115211153111541115511156111571115811159111601116111162111631116411165111661116711168111691117011171111721117311174111751117611177111781117911180111811118211183111841118511186111871118811189111901119111192111931119411195111961119711198111991120011201112021120311204112051120611207112081120911210112111121211213112141121511216112171121811219112201122111222112231122411225112261122711228112291123011231112321123311234112351123611237112381123911240112411124211243112441124511246112471124811249112501125111252112531125411255112561125711258112591126011261112621126311264112651126611267112681126911270112711127211273112741127511276112771127811279112801128111282112831128411285112861128711288112891129011291112921129311294112951129611297112981129911300113011130211303113041130511306113071130811309113101131111312113131131411315113161131711318113191132011321113221132311324113251132611327113281132911330113311133211333113341133511336113371133811339113401134111342113431134411345113461134711348113491135011351113521135311354113551135611357113581135911360113611136211363113641136511366113671136811369113701137111372113731137411375113761137711378113791138011381113821138311384113851138611387113881138911390113911139211393113941139511396113971139811399114001140111402114031140411405114061140711408114091141011411114121141311414114151141611417114181141911420114211142211423114241142511426114271142811429114301143111432114331143411435114361143711438114391144011441114421144311444114451144611447114481144911450114511145211453114541145511456114571145811459114601146111462114631146411465114661146711468114691147011471114721147311474114751147611477114781147911480114811148211483114841148511486114871148811489114901149111492114931149411495114961149711498114991150011501115021150311504115051150611507115081150911510115111151211513115141151511516115171151811519115201152111522115231152411525115261152711528115291153011531115321153311534115351153611537115381153911540115411154211543115441154511546115471154811549115501155111552115531155411555115561155711558115591156011561115621156311564115651156611567115681156911570115711157211573115741157511576115771157811579115801158111582115831158411585115861158711588115891159011591115921159311594115951159611597115981159911600116011160211603116041160511606116071160811609116101161111612116131161411615116161161711618116191162011621116221162311624116251162611627116281162911630116311163211633116341163511636116371163811639116401164111642116431164411645116461164711648116491165011651116521165311654116551165611657116581165911660116611166211663116641166511666116671166811669116701167111672116731167411675116761167711678116791168011681116821168311684116851168611687116881168911690116911169211693116941169511696116971169811699117001170111702117031170411705117061170711708117091171011711117121171311714117151171611717117181171911720117211172211723117241172511726117271172811729117301173111732117331173411735117361173711738117391174011741117421174311744117451174611747117481174911750117511175211753117541175511756117571175811759117601176111762117631176411765117661176711768117691177011771117721177311774117751177611777117781177911780117811178211783117841178511786117871178811789117901179111792117931179411795117961179711798117991180011801118021180311804118051180611807118081180911810118111181211813118141181511816118171181811819118201182111822118231182411825118261182711828118291183011831118321183311834118351183611837118381183911840118411184211843118441184511846118471184811849118501185111852118531185411855118561185711858118591186011861118621186311864118651186611867118681186911870118711187211873118741187511876118771187811879118801188111882118831188411885118861188711888118891189011891118921189311894118951189611897118981189911900119011190211903119041190511906119071190811909119101191111912119131191411915119161191711918119191192011921119221192311924119251192611927119281192911930119311193211933119341193511936119371193811939119401194111942119431194411945119461194711948119491195011951119521195311954119551195611957119581195911960119611196211963119641196511966119671196811969119701197111972119731197411975119761197711978119791198011981119821198311984119851198611987119881198911990119911199211993119941199511996119971199811999120001200112002120031200412005120061200712008120091201012011120121201312014120151201612017120181201912020120211202212023120241202512026120271202812029120301203112032120331203412035120361203712038120391204012041120421204312044120451204612047120481204912050120511205212053120541205512056120571205812059120601206112062120631206412065120661206712068120691207012071120721207312074120751207612077120781207912080120811208212083120841208512086120871208812089120901209112092120931209412095120961209712098120991210012101121021210312104121051210612107121081210912110121111211212113121141211512116121171211812119121201212112122121231212412125121261212712128121291213012131121321213312134121351213612137121381213912140121411214212143121441214512146121471214812149121501215112152121531215412155121561215712158121591216012161121621216312164121651216612167121681216912170121711217212173121741217512176121771217812179121801218112182121831218412185121861218712188121891219012191121921219312194121951219612197121981219912200122011220212203122041220512206122071220812209122101221112212122131221412215122161221712218122191222012221122221222312224122251222612227122281222912230122311223212233122341223512236122371223812239122401224112242122431224412245122461224712248122491225012251122521225312254122551225612257122581225912260122611226212263122641226512266122671226812269122701227112272122731227412275122761227712278122791228012281122821228312284122851228612287122881228912290122911229212293122941229512296122971229812299123001230112302123031230412305123061230712308123091231012311123121231312314123151231612317123181231912320123211232212323123241232512326123271232812329123301233112332123331233412335123361233712338123391234012341123421234312344123451234612347123481234912350123511235212353123541235512356123571235812359123601236112362123631236412365123661236712368123691237012371123721237312374123751237612377123781237912380123811238212383123841238512386123871238812389123901239112392123931239412395123961239712398123991240012401124021240312404124051240612407124081240912410124111241212413124141241512416124171241812419124201242112422124231242412425124261242712428124291243012431124321243312434124351243612437124381243912440124411244212443124441244512446124471244812449124501245112452124531245412455124561245712458124591246012461124621246312464124651246612467124681246912470124711247212473124741247512476124771247812479124801248112482124831248412485124861248712488124891249012491124921249312494124951249612497124981249912500125011250212503125041250512506125071250812509125101251112512125131251412515125161251712518125191252012521125221252312524125251252612527125281252912530125311253212533125341253512536125371253812539125401254112542125431254412545125461254712548125491255012551125521255312554125551255612557125581255912560125611256212563125641256512566125671256812569125701257112572125731257412575125761257712578125791258012581125821258312584125851258612587125881258912590125911259212593125941259512596125971259812599126001260112602126031260412605126061260712608126091261012611126121261312614126151261612617126181261912620126211262212623126241262512626126271262812629126301263112632126331263412635126361263712638126391264012641126421264312644126451264612647126481264912650126511265212653126541265512656126571265812659126601266112662126631266412665126661266712668126691267012671126721267312674126751267612677126781267912680126811268212683126841268512686126871268812689126901269112692126931269412695126961269712698126991270012701127021270312704127051270612707127081270912710127111271212713127141271512716127171271812719127201272112722127231272412725127261272712728127291273012731127321273312734127351273612737127381273912740127411274212743127441274512746127471274812749127501275112752127531275412755127561275712758127591276012761127621276312764127651276612767127681276912770127711277212773127741277512776127771277812779127801278112782127831278412785127861278712788127891279012791127921279312794127951279612797127981279912800128011280212803128041280512806128071280812809128101281112812128131281412815128161281712818128191282012821128221282312824128251282612827128281282912830128311283212833128341283512836128371283812839128401284112842128431284412845128461284712848128491285012851128521285312854128551285612857128581285912860128611286212863128641286512866128671286812869128701287112872128731287412875128761287712878128791288012881128821288312884128851288612887128881288912890128911289212893128941289512896128971289812899129001290112902129031290412905129061290712908129091291012911129121291312914129151291612917129181291912920129211292212923129241292512926129271292812929129301293112932129331293412935129361293712938129391294012941129421294312944129451294612947129481294912950129511295212953129541295512956129571295812959129601296112962129631296412965129661296712968129691297012971129721297312974129751297612977129781297912980129811298212983129841298512986129871298812989129901299112992129931299412995129961299712998129991300013001130021300313004130051300613007130081300913010130111301213013130141301513016130171301813019130201302113022130231302413025130261302713028130291303013031130321303313034130351303613037130381303913040130411304213043130441304513046130471304813049130501305113052130531305413055130561305713058130591306013061130621306313064130651306613067130681306913070130711307213073130741307513076130771307813079130801308113082130831308413085130861308713088130891309013091130921309313094130951309613097130981309913100131011310213103131041310513106131071310813109131101311113112131131311413115131161311713118131191312013121131221312313124131251312613127131281312913130131311313213133131341313513136131371313813139131401314113142131431314413145131461314713148131491315013151131521315313154131551315613157131581315913160131611316213163131641316513166131671316813169131701317113172131731317413175131761317713178131791318013181131821318313184131851318613187131881318913190131911319213193131941319513196131971319813199132001320113202132031320413205132061320713208132091321013211132121321313214132151321613217132181321913220132211322213223132241322513226132271322813229132301323113232132331323413235132361323713238132391324013241132421324313244132451324613247132481324913250132511325213253132541325513256132571325813259132601326113262132631326413265132661326713268132691327013271132721327313274132751327613277132781327913280132811328213283132841328513286132871328813289132901329113292132931329413295132961329713298132991330013301133021330313304133051330613307133081330913310133111331213313133141331513316133171331813319133201332113322133231332413325133261332713328133291333013331133321333313334133351333613337133381333913340133411334213343133441334513346133471334813349133501335113352133531335413355133561335713358133591336013361133621336313364133651336613367133681336913370133711337213373133741337513376133771337813379133801338113382133831338413385133861338713388133891339013391133921339313394133951339613397133981339913400134011340213403134041340513406134071340813409134101341113412134131341413415134161341713418134191342013421134221342313424134251342613427134281342913430134311343213433134341343513436134371343813439134401344113442134431344413445134461344713448134491345013451134521345313454134551345613457134581345913460134611346213463134641346513466134671346813469134701347113472134731347413475134761347713478134791348013481134821348313484134851348613487134881348913490134911349213493134941349513496134971349813499135001350113502135031350413505135061350713508135091351013511135121351313514135151351613517135181351913520135211352213523135241352513526135271352813529135301353113532135331353413535135361353713538135391354013541135421354313544135451354613547135481354913550135511355213553135541355513556135571355813559135601356113562135631356413565135661356713568135691357013571135721357313574135751357613577135781357913580135811358213583135841358513586135871358813589135901359113592135931359413595135961359713598135991360013601136021360313604136051360613607136081360913610136111361213613136141361513616136171361813619136201362113622136231362413625136261362713628136291363013631136321363313634136351363613637136381363913640136411364213643136441364513646136471364813649136501365113652136531365413655136561365713658136591366013661136621366313664136651366613667136681366913670136711367213673136741367513676136771367813679136801368113682136831368413685136861368713688136891369013691136921369313694136951369613697136981369913700137011370213703137041370513706137071370813709137101371113712137131371413715137161371713718137191372013721137221372313724137251372613727137281372913730137311373213733137341373513736137371373813739137401374113742137431374413745137461374713748137491375013751137521375313754137551375613757137581375913760137611376213763137641376513766137671376813769137701377113772137731377413775137761377713778137791378013781137821378313784137851378613787137881378913790137911379213793137941379513796137971379813799138001380113802138031380413805138061380713808138091381013811138121381313814138151381613817138181381913820138211382213823138241382513826138271382813829138301383113832138331383413835138361383713838138391384013841138421384313844138451384613847138481384913850138511385213853138541385513856138571385813859138601386113862138631386413865138661386713868138691387013871138721387313874138751387613877138781387913880138811388213883138841388513886138871388813889138901389113892138931389413895138961389713898138991390013901139021390313904139051390613907139081390913910139111391213913139141391513916139171391813919139201392113922139231392413925139261392713928139291393013931139321393313934139351393613937139381393913940139411394213943139441394513946139471394813949139501395113952139531395413955139561395713958139591396013961139621396313964139651396613967139681396913970139711397213973139741397513976139771397813979139801398113982139831398413985139861398713988139891399013991139921399313994139951399613997139981399914000140011400214003140041400514006140071400814009140101401114012140131401414015140161401714018140191402014021140221402314024140251402614027140281402914030140311403214033140341403514036140371403814039140401404114042140431404414045140461404714048140491405014051140521405314054140551405614057140581405914060140611406214063140641406514066140671406814069140701407114072140731407414075140761407714078140791408014081140821408314084140851408614087140881408914090140911409214093140941409514096140971409814099141001410114102141031410414105141061410714108141091411014111141121411314114141151411614117141181411914120141211412214123141241412514126141271412814129141301413114132141331413414135141361413714138141391414014141141421414314144141451414614147141481414914150141511415214153141541415514156141571415814159141601416114162141631416414165141661416714168141691417014171141721417314174141751417614177141781417914180141811418214183141841418514186141871418814189141901419114192141931419414195141961419714198141991420014201142021420314204142051420614207142081420914210142111421214213142141421514216142171421814219142201422114222142231422414225142261422714228142291423014231142321423314234142351423614237142381423914240142411424214243142441424514246142471424814249142501425114252142531425414255142561425714258142591426014261142621426314264142651426614267142681426914270142711427214273142741427514276142771427814279142801428114282142831428414285142861428714288142891429014291142921429314294142951429614297142981429914300143011430214303143041430514306143071430814309143101431114312143131431414315143161431714318143191432014321143221432314324143251432614327143281432914330143311433214333143341433514336143371433814339143401434114342143431434414345143461434714348143491435014351143521435314354143551435614357143581435914360143611436214363143641436514366143671436814369143701437114372143731437414375143761437714378143791438014381143821438314384143851438614387143881438914390143911439214393143941439514396143971439814399144001440114402144031440414405144061440714408144091441014411144121441314414144151441614417144181441914420144211442214423144241442514426144271442814429144301443114432144331443414435144361443714438144391444014441144421444314444144451444614447144481444914450144511445214453144541445514456144571445814459144601446114462144631446414465144661446714468144691447014471144721447314474144751447614477144781447914480144811448214483144841448514486144871448814489144901449114492144931449414495144961449714498144991450014501145021450314504145051450614507145081450914510145111451214513145141451514516145171451814519145201452114522145231452414525145261452714528145291453014531145321453314534145351453614537145381453914540145411454214543145441454514546145471454814549145501455114552145531455414555145561455714558145591456014561145621456314564145651456614567145681456914570145711457214573145741457514576145771457814579145801458114582145831458414585145861458714588145891459014591145921459314594145951459614597145981459914600146011460214603146041460514606146071460814609146101461114612146131461414615146161461714618146191462014621146221462314624146251462614627146281462914630146311463214633146341463514636146371463814639146401464114642146431464414645146461464714648146491465014651146521465314654146551465614657146581465914660146611466214663146641466514666146671466814669146701467114672146731467414675146761467714678146791468014681146821468314684146851468614687146881468914690146911469214693146941469514696146971469814699147001470114702147031470414705147061470714708147091471014711147121471314714147151471614717147181471914720147211472214723147241472514726147271472814729147301473114732147331473414735147361473714738147391474014741147421474314744147451474614747147481474914750147511475214753147541475514756147571475814759147601476114762147631476414765147661476714768147691477014771147721477314774147751477614777147781477914780147811478214783147841478514786147871478814789147901479114792147931479414795147961479714798147991480014801148021480314804148051480614807148081480914810148111481214813148141481514816148171481814819148201482114822148231482414825148261482714828148291483014831148321483314834148351483614837148381483914840148411484214843148441484514846148471484814849148501485114852148531485414855148561485714858148591486014861148621486314864148651486614867148681486914870148711487214873148741487514876148771487814879148801488114882148831488414885148861488714888148891489014891148921489314894148951489614897148981489914900149011490214903149041490514906149071490814909149101491114912149131491414915149161491714918149191492014921149221492314924149251492614927149281492914930149311493214933149341493514936149371493814939149401494114942149431494414945149461494714948149491495014951149521495314954149551495614957149581495914960149611496214963149641496514966149671496814969149701497114972149731497414975149761497714978149791498014981149821498314984149851498614987149881498914990149911499214993149941499514996149971499814999150001500115002150031500415005150061500715008150091501015011150121501315014150151501615017150181501915020150211502215023150241502515026150271502815029150301503115032150331503415035150361503715038150391504015041150421504315044150451504615047150481504915050150511505215053150541505515056150571505815059150601506115062150631506415065150661506715068150691507015071150721507315074150751507615077150781507915080150811508215083150841508515086150871508815089150901509115092150931509415095150961509715098150991510015101151021510315104151051510615107151081510915110151111511215113151141511515116151171511815119151201512115122151231512415125151261512715128151291513015131151321513315134151351513615137151381513915140151411514215143151441514515146151471514815149151501515115152151531515415155151561515715158151591516015161151621516315164151651516615167151681516915170151711517215173151741517515176151771517815179151801518115182151831518415185151861518715188151891519015191151921519315194151951519615197151981519915200152011520215203152041520515206152071520815209152101521115212152131521415215152161521715218152191522015221152221522315224152251522615227152281522915230152311523215233152341523515236152371523815239152401524115242152431524415245152461524715248152491525015251152521525315254152551525615257152581525915260152611526215263152641526515266152671526815269152701527115272152731527415275152761527715278152791528015281152821528315284152851528615287152881528915290152911529215293152941529515296152971529815299153001530115302153031530415305 |
- /*
- * Asterisk -- An open source telephony toolkit.
- *
- * Copyright (C) 1999 - 2006, Digium, Inc.
- *
- * Mark Spencer <markster@digium.com>
- *
- * See http://www.asterisk.org for more information about
- * the Asterisk project. Please do not directly contact
- * any of the maintainers of this project for assistance;
- * the project provides a web site, mailing lists and IRC
- * channels for your use.
- *
- * This program is free software, distributed under the terms of
- * the GNU General Public License Version 2. See the LICENSE file
- * at the top of the source tree.
- */
- /*! \file
- *
- * \brief Implementation of Inter-Asterisk eXchange Version 2
- * as specified in RFC 5456
- *
- * \author Mark Spencer <markster@digium.com>
- *
- * \par See also
- * \arg \ref Config_iax
- *
- * \ingroup channel_drivers
- *
- * \todo Implement musicclass settings for IAX2 devices
- */
- /*** MODULEINFO
- <use type="external">crypto</use>
- <support_level>core</support_level>
- ***/
- #include "asterisk.h"
- ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
- #include <sys/mman.h>
- #include <dirent.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include <netinet/in_systm.h>
- #include <netinet/ip.h>
- #include <sys/time.h>
- #include <sys/signal.h>
- #include <signal.h>
- #include <strings.h>
- #include <netdb.h>
- #include <fcntl.h>
- #include <sys/stat.h>
- #include <regex.h>
- #include "asterisk/paths.h" /* need ast_config_AST_DATA_DIR for firmware */
- #include "asterisk/lock.h"
- #include "asterisk/frame.h"
- #include "asterisk/channel.h"
- #include "asterisk/module.h"
- #include "asterisk/pbx.h"
- #include "asterisk/sched.h"
- #include "asterisk/io.h"
- #include "asterisk/config.h"
- #include "asterisk/cli.h"
- #include "asterisk/translate.h"
- #include "asterisk/md5.h"
- #include "asterisk/cdr.h"
- #include "asterisk/crypto.h"
- #include "asterisk/acl.h"
- #include "asterisk/manager.h"
- #include "asterisk/callerid.h"
- #include "asterisk/app.h"
- #include "asterisk/astdb.h"
- #include "asterisk/musiconhold.h"
- #include "asterisk/features.h"
- #include "asterisk/utils.h"
- #include "asterisk/causes.h"
- #include "asterisk/localtime.h"
- #include "asterisk/dnsmgr.h"
- #include "asterisk/devicestate.h"
- #include "asterisk/netsock.h"
- #include "asterisk/stringfields.h"
- #include "asterisk/linkedlists.h"
- #include "asterisk/event.h"
- #include "asterisk/astobj2.h"
- #include "asterisk/timing.h"
- #include "asterisk/taskprocessor.h"
- #include "asterisk/test.h"
- #include "asterisk/data.h"
- #include "asterisk/netsock2.h"
- #include "iax2.h"
- #include "iax2-parser.h"
- #include "iax2-provision.h"
- #include "jitterbuf.h"
- /*** DOCUMENTATION
- <application name="IAX2Provision" language="en_US">
- <synopsis>
- Provision a calling IAXy with a given template.
- </synopsis>
- <syntax>
- <parameter name="template">
- <para>If not specified, defaults to <literal>default</literal>.</para>
- </parameter>
- </syntax>
- <description>
- <para>Provisions the calling IAXy (assuming the calling entity is in fact an IAXy) with the
- given <replaceable>template</replaceable>. Returns <literal>-1</literal> on error
- or <literal>0</literal> on success.</para>
- </description>
- </application>
- <function name="IAXPEER" language="en_US">
- <synopsis>
- Gets IAX peer information.
- </synopsis>
- <syntax>
- <parameter name="peername" required="true">
- <enumlist>
- <enum name="CURRENTCHANNEL">
- <para>If <replaceable>peername</replaceable> is specified to this value, return the IP address of the
- endpoint of the current channel</para>
- </enum>
- </enumlist>
- </parameter>
- <parameter name="item">
- <para>If <replaceable>peername</replaceable> is specified, valid items are:</para>
- <enumlist>
- <enum name="ip">
- <para>(default) The IP address.</para>
- </enum>
- <enum name="status">
- <para>The peer's status (if <literal>qualify=yes</literal>)</para>
- </enum>
- <enum name="mailbox">
- <para>The configured mailbox.</para>
- </enum>
- <enum name="context">
- <para>The configured context.</para>
- </enum>
- <enum name="expire">
- <para>The epoch time of the next expire.</para>
- </enum>
- <enum name="dynamic">
- <para>Is it dynamic? (yes/no).</para>
- </enum>
- <enum name="callerid_name">
- <para>The configured Caller ID name.</para>
- </enum>
- <enum name="callerid_num">
- <para>The configured Caller ID number.</para>
- </enum>
- <enum name="codecs">
- <para>The configured codecs.</para>
- </enum>
- <enum name="codec[x]">
- <para>Preferred codec index number <replaceable>x</replaceable> (beginning
- with <literal>0</literal>)</para>
- </enum>
- </enumlist>
- </parameter>
- </syntax>
- <description>
- <para>Gets information associated with the specified IAX2 peer.</para>
- </description>
- <see-also>
- <ref type="function">SIPPEER</ref>
- </see-also>
- </function>
- <function name="IAXVAR" language="en_US">
- <synopsis>
- Sets or retrieves a remote variable.
- </synopsis>
- <syntax>
- <parameter name="varname" required="true" />
- </syntax>
- <description>
- <para>Gets or sets a variable that is sent to a remote IAX2 peer during call setup.</para>
- </description>
- </function>
- <manager name="IAXpeers" language="en_US">
- <synopsis>
- List IAX peers.
- </synopsis>
- <syntax>
- <xi:include xpointer="xpointer(/docs/manager[@name='Login']/syntax/parameter[@name='ActionID'])" />
- </syntax>
- <description>
- </description>
- </manager>
- <manager name="IAXpeerlist" language="en_US">
- <synopsis>
- List IAX Peers.
- </synopsis>
- <syntax>
- <xi:include xpointer="xpointer(/docs/manager[@name='Login']/syntax/parameter[@name='ActionID'])" />
- </syntax>
- <description>
- <para>List all the IAX peers.</para>
- </description>
- </manager>
- <manager name="IAXnetstats" language="en_US">
- <synopsis>
- Show IAX Netstats.
- </synopsis>
- <syntax />
- <description>
- <para>Show IAX channels network statistics.</para>
- </description>
- </manager>
- <manager name="IAXregistry" language="en_US">
- <synopsis>
- Show IAX registrations.
- </synopsis>
- <syntax>
- <xi:include xpointer="xpointer(/docs/manager[@name='Login']/syntax/parameter[@name='ActionID'])" />
- </syntax>
- <description>
- <para>Show IAX registrations.</para>
- </description>
- </manager>
- ***/
- /* Define SCHED_MULTITHREADED to run the scheduler in a special
- multithreaded mode. */
- #define SCHED_MULTITHREADED
- /* Define DEBUG_SCHED_MULTITHREADED to keep track of where each
- thread is actually doing. */
- #define DEBUG_SCHED_MULTITHREAD
- #ifdef SO_NO_CHECK
- static int nochecksums = 0;
- #endif
- #define PTR_TO_CALLNO(a) ((unsigned short)(unsigned long)(a))
- #define CALLNO_TO_PTR(a) ((void *)(unsigned long)(a))
- #define DEFAULT_THREAD_COUNT 10
- #define DEFAULT_MAX_THREAD_COUNT 100
- #define DEFAULT_RETRY_TIME 1000
- #define MEMORY_SIZE 100
- #define DEFAULT_DROP 3
- #define DEBUG_SUPPORT
- #define MIN_REUSE_TIME 60 /* Don't reuse a call number within 60 seconds */
- /* Sample over last 100 units to determine historic jitter */
- #define GAMMA (0.01)
- static struct ast_codec_pref prefs;
- static const char tdesc[] = "Inter Asterisk eXchange Driver (Ver 2)";
- /*! \brief Maximum transmission unit for the UDP packet in the trunk not to be
- fragmented. This is based on 1516 - ethernet - ip - udp - iax minus one g711 frame = 1240 */
- #define MAX_TRUNK_MTU 1240
- static int global_max_trunk_mtu; /*!< Maximum MTU, 0 if not used */
- static int trunk_timed, trunk_untimed, trunk_maxmtu, trunk_nmaxmtu ; /*!< Trunk MTU statistics */
- #define DEFAULT_CONTEXT "default"
- static char default_parkinglot[AST_MAX_CONTEXT];
- static char language[MAX_LANGUAGE] = "";
- static char regcontext[AST_MAX_CONTEXT] = "";
- static struct ast_event_sub *network_change_event_subscription; /*!< subscription id for network change events */
- static struct ast_event_sub *acl_change_event_subscription; /*!< subscription id for ACL change events */
- static int network_change_event_sched_id = -1;
- static int maxauthreq = 3;
- static int max_retries = 4;
- static int ping_time = 21;
- static int lagrq_time = 10;
- static int maxjitterbuffer=1000;
- static int resyncthreshold=1000;
- static int maxjitterinterps=10;
- static int jittertargetextra = 40; /* number of milliseconds the new jitter buffer adds on to its size */
- #define MAX_TRUNKDATA 640 * 200 /*!< 40ms, uncompressed linear * 200 channels */
- static int trunkfreq = 20;
- static int trunkmaxsize = MAX_TRUNKDATA;
- static int authdebug = 0;
- static int autokill = 0;
- static int iaxcompat = 0;
- static int last_authmethod = 0;
- static int iaxdefaultdpcache=10 * 60; /* Cache dialplan entries for 10 minutes by default */
- static int iaxdefaulttimeout = 5; /* Default to wait no more than 5 seconds for a reply to come back */
- static struct {
- unsigned int tos;
- unsigned int cos;
- } qos = { 0, 0 };
- static int min_reg_expire;
- static int max_reg_expire;
- static int srvlookup = 0;
- static struct ast_timer *timer; /* Timer for trunking */
- static struct ast_netsock_list *netsock;
- static struct ast_netsock_list *outsock; /*!< used if sourceaddress specified and bindaddr == INADDR_ANY */
- static int defaultsockfd = -1;
- static int (*iax2_regfunk)(const char *username, int onoff) = NULL;
- /* Ethernet, etc */
- #define IAX_CAPABILITY_FULLBANDWIDTH 0xFFFF
- /* T1, maybe ISDN */
- #define IAX_CAPABILITY_MEDBANDWIDTH (IAX_CAPABILITY_FULLBANDWIDTH & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_SLINEAR) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_SLINEAR16) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_SIREN7) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_SIREN14) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_G719) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_ULAW) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_ALAW) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_G722))
- /* A modem */
- #define IAX_CAPABILITY_LOWBANDWIDTH (IAX_CAPABILITY_MEDBANDWIDTH & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_G726) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_G726_AAL2) & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_ADPCM))
- #define IAX_CAPABILITY_LOWFREE (IAX_CAPABILITY_LOWBANDWIDTH & \
- ~ast_format_id_to_old_bitfield(AST_FORMAT_G723_1))
- #define DEFAULT_MAXMS 2000 /* Must be faster than 2 seconds by default */
- #define DEFAULT_FREQ_OK 60 * 1000 /* How often to check for the host to be up */
- #define DEFAULT_FREQ_NOTOK 10 * 1000 /* How often to check, if the host is down... */
- /* if a pvt has encryption setup done and is running on the call */
- #define IAX_CALLENCRYPTED(pvt) \
- (ast_test_flag64(pvt, IAX_ENCRYPTED) && ast_test_flag64(pvt, IAX_KEYPOPULATED))
- #define IAX_DEBUGDIGEST(msg, key) do { \
- int idx; \
- char digest[33] = ""; \
- \
- if (!iaxdebug) \
- break; \
- \
- for (idx = 0; idx < 16; idx++) \
- sprintf(digest + (idx << 1), "%02hhx", (unsigned char) key[idx]); \
- \
- ast_log(LOG_NOTICE, msg " IAX_COMMAND_RTKEY to rotate key to '%s'\n", digest); \
- } while(0)
- static struct io_context *io;
- static struct ast_sched_context *sched;
- #define DONT_RESCHEDULE -2
- static iax2_format iax2_capability = IAX_CAPABILITY_FULLBANDWIDTH;
- static int iaxdebug = 0;
- static int iaxtrunkdebug = 0;
- static int test_losspct = 0;
- #ifdef IAXTESTS
- static int test_late = 0;
- static int test_resync = 0;
- static int test_jit = 0;
- static int test_jitpct = 0;
- #endif /* IAXTESTS */
- static char accountcode[AST_MAX_ACCOUNT_CODE];
- static char mohinterpret[MAX_MUSICCLASS];
- static char mohsuggest[MAX_MUSICCLASS];
- static int amaflags = 0;
- static int adsi = 0;
- static int delayreject = 0;
- static int iax2_encryption = 0;
- static struct ast_flags64 globalflags = { 0 };
- static pthread_t netthreadid = AST_PTHREADT_NULL;
- enum iax2_state {
- IAX_STATE_STARTED = (1 << 0),
- IAX_STATE_AUTHENTICATED = (1 << 1),
- IAX_STATE_TBD = (1 << 2),
- };
- struct iax2_context {
- char context[AST_MAX_CONTEXT];
- struct iax2_context *next;
- };
- #define IAX_HASCALLERID (uint64_t)(1 << 0) /*!< CallerID has been specified */
- #define IAX_DELME (uint64_t)(1 << 1) /*!< Needs to be deleted */
- #define IAX_TEMPONLY (uint64_t)(1 << 2) /*!< Temporary (realtime) */
- #define IAX_TRUNK (uint64_t)(1 << 3) /*!< Treat as a trunk */
- #define IAX_NOTRANSFER (uint64_t)(1 << 4) /*!< Don't native bridge */
- #define IAX_USEJITTERBUF (uint64_t)(1 << 5) /*!< Use jitter buffer */
- #define IAX_DYNAMIC (uint64_t)(1 << 6) /*!< dynamic peer */
- #define IAX_SENDANI (uint64_t)(1 << 7) /*!< Send ANI along with CallerID */
- #define IAX_RTSAVE_SYSNAME (uint64_t)(1 << 8) /*!< Save Systname on Realtime Updates */
- #define IAX_ALREADYGONE (uint64_t)(1 << 9) /*!< Already disconnected */
- #define IAX_PROVISION (uint64_t)(1 << 10) /*!< This is a provisioning request */
- #define IAX_QUELCH (uint64_t)(1 << 11) /*!< Whether or not we quelch audio */
- #define IAX_ENCRYPTED (uint64_t)(1 << 12) /*!< Whether we should assume encrypted tx/rx */
- #define IAX_KEYPOPULATED (uint64_t)(1 << 13) /*!< Whether we have a key populated */
- #define IAX_CODEC_USER_FIRST (uint64_t)(1 << 14) /*!< are we willing to let the other guy choose the codec? */
- #define IAX_CODEC_NOPREFS (uint64_t)(1 << 15) /*!< Force old behaviour by turning off prefs */
- #define IAX_CODEC_NOCAP (uint64_t)(1 << 16) /*!< only consider requested format and ignore capabilities*/
- #define IAX_RTCACHEFRIENDS (uint64_t)(1 << 17) /*!< let realtime stay till your reload */
- #define IAX_RTUPDATE (uint64_t)(1 << 18) /*!< Send a realtime update */
- #define IAX_RTAUTOCLEAR (uint64_t)(1 << 19) /*!< erase me on expire */
- #define IAX_FORCEJITTERBUF (uint64_t)(1 << 20) /*!< Force jitterbuffer, even when bridged to a channel that can take jitter */
- #define IAX_RTIGNOREREGEXPIRE (uint64_t)(1 << 21) /*!< When using realtime, ignore registration expiration */
- #define IAX_TRUNKTIMESTAMPS (uint64_t)(1 << 22) /*!< Send trunk timestamps */
- #define IAX_TRANSFERMEDIA (uint64_t)(1 << 23) /*!< When doing IAX2 transfers, transfer media only */
- #define IAX_MAXAUTHREQ (uint64_t)(1 << 24) /*!< Maximum outstanding AUTHREQ restriction is in place */
- #define IAX_DELAYPBXSTART (uint64_t)(1 << 25) /*!< Don't start a PBX on the channel until the peer sends us a response, so that we've achieved a three-way handshake with them before sending voice or anything else */
- #define IAX_ALLOWFWDOWNLOAD (uint64_t)(1 << 26) /*!< Allow the FWDOWNL command? */
- #define IAX_IMMEDIATE (uint64_t)(1 << 27) /*!< Allow immediate off-hook to extension s */
- #define IAX_SENDCONNECTEDLINE (uint64_t)(1 << 28) /*!< Allow sending of connected line updates */
- #define IAX_RECVCONNECTEDLINE (uint64_t)(1 << 29) /*!< Allow receiving of connected line updates */
- #define IAX_FORCE_ENCRYPT (uint64_t)(1 << 30) /*!< Forces call encryption, if encryption not possible hangup */
- #define IAX_SHRINKCALLERID (uint64_t)(1 << 31) /*!< Turn on and off caller id shrinking */
- static int global_rtautoclear = 120;
- static int reload_config(int forced_reload);
- /*!
- * \brief Call token validation settings.
- */
- enum calltoken_peer_enum {
- /*! \brief Default calltoken required unless the ip is in the ignorelist */
- CALLTOKEN_DEFAULT = 0,
- /*! \brief Require call token validation. */
- CALLTOKEN_YES = 1,
- /*! \brief Require call token validation after a successful registration
- * using call token validation occurs. */
- CALLTOKEN_AUTO = 2,
- /*! \brief Do not require call token validation. */
- CALLTOKEN_NO = 3,
- };
- struct iax2_user {
- AST_DECLARE_STRING_FIELDS(
- AST_STRING_FIELD(name);
- AST_STRING_FIELD(secret);
- AST_STRING_FIELD(dbsecret);
- AST_STRING_FIELD(accountcode);
- AST_STRING_FIELD(mohinterpret);
- AST_STRING_FIELD(mohsuggest);
- AST_STRING_FIELD(inkeys); /*!< Key(s) this user can use to authenticate to us */
- AST_STRING_FIELD(language);
- AST_STRING_FIELD(cid_num);
- AST_STRING_FIELD(cid_name);
- AST_STRING_FIELD(parkinglot); /*!< Default parkinglot for device */
- );
- int authmethods;
- int encmethods;
- int amaflags;
- int adsi;
- uint64_t flags;
- iax2_format capability;
- int maxauthreq; /*!< Maximum allowed outstanding AUTHREQs */
- int curauthreq; /*!< Current number of outstanding AUTHREQs */
- struct ast_codec_pref prefs;
- struct ast_acl_list *acl;
- struct iax2_context *contexts;
- struct ast_variable *vars;
- enum calltoken_peer_enum calltoken_required; /*!< Is calltoken validation required or not, can be YES, NO, or AUTO */
- };
- struct iax2_peer {
- AST_DECLARE_STRING_FIELDS(
- AST_STRING_FIELD(name);
- AST_STRING_FIELD(username);
- AST_STRING_FIELD(description); /*!< Description of the peer */
- AST_STRING_FIELD(secret);
- AST_STRING_FIELD(dbsecret);
- AST_STRING_FIELD(outkey); /*!< What key we use to talk to this peer */
- AST_STRING_FIELD(regexten); /*!< Extension to register (if regcontext is used) */
- AST_STRING_FIELD(context); /*!< For transfers only */
- AST_STRING_FIELD(peercontext); /*!< Context to pass to peer */
- AST_STRING_FIELD(mailbox); /*!< Mailbox */
- AST_STRING_FIELD(mohinterpret);
- AST_STRING_FIELD(mohsuggest);
- AST_STRING_FIELD(inkeys); /*!< Key(s) this peer can use to authenticate to us */
- /* Suggested caller id if registering */
- AST_STRING_FIELD(cid_num); /*!< Default context (for transfer really) */
- AST_STRING_FIELD(cid_name); /*!< Default context (for transfer really) */
- AST_STRING_FIELD(zonetag); /*!< Time Zone */
- AST_STRING_FIELD(parkinglot); /*!< Default parkinglot for device */
- );
- struct ast_codec_pref prefs;
- struct ast_dnsmgr_entry *dnsmgr; /*!< DNS refresh manager */
- struct ast_sockaddr addr;
- int formats;
- int sockfd; /*!< Socket to use for transmission */
- struct in_addr mask;
- int adsi;
- uint64_t flags;
- /* Dynamic Registration fields */
- struct sockaddr_in defaddr; /*!< Default address if there is one */
- int authmethods; /*!< Authentication methods (IAX_AUTH_*) */
- int encmethods; /*!< Encryption methods (IAX_ENCRYPT_*) */
- int expire; /*!< Schedule entry for expiry */
- int expiry; /*!< How soon to expire */
- iax2_format capability; /*!< Capability */
- /* Qualification */
- int callno; /*!< Call number of POKE request */
- int pokeexpire; /*!< Scheduled qualification-related task (ie iax2_poke_peer_s or iax2_poke_noanswer) */
- int lastms; /*!< How long last response took (in ms), or -1 for no response */
- int maxms; /*!< Max ms we will accept for the host to be up, 0 to not monitor */
- int pokefreqok; /*!< How often to check if the host is up */
- int pokefreqnotok; /*!< How often to check when the host has been determined to be down */
- int historicms; /*!< How long recent average responses took */
- int smoothing; /*!< Sample over how many units to determine historic ms */
- uint16_t maxcallno; /*!< Max call number limit for this peer. Set on registration */
- struct ast_event_sub *mwi_event_sub;
- struct ast_acl_list *acl;
- enum calltoken_peer_enum calltoken_required; /*!< Is calltoken validation required or not, can be YES, NO, or AUTO */
- };
- #define IAX2_TRUNK_PREFACE (sizeof(struct iax_frame) + sizeof(struct ast_iax2_meta_hdr) + sizeof(struct ast_iax2_meta_trunk_hdr))
- struct iax2_trunk_peer {
- ast_mutex_t lock;
- int sockfd;
- struct sockaddr_in addr;
- struct timeval txtrunktime; /*!< Transmit trunktime */
- struct timeval rxtrunktime; /*!< Receive trunktime */
- struct timeval lasttxtime; /*!< Last transmitted trunktime */
- struct timeval trunkact; /*!< Last trunk activity */
- unsigned int lastsent; /*!< Last sent time */
- /* Trunk data and length */
- unsigned char *trunkdata;
- unsigned int trunkdatalen;
- unsigned int trunkdataalloc;
- int trunkmaxmtu;
- int trunkerror;
- int calls;
- AST_LIST_ENTRY(iax2_trunk_peer) list;
- };
- static AST_LIST_HEAD_STATIC(tpeers, iax2_trunk_peer);
- struct iax_firmware {
- AST_LIST_ENTRY(iax_firmware) list;
- int fd;
- int mmaplen;
- int dead;
- struct ast_iax2_firmware_header *fwh;
- unsigned char *buf;
- };
- enum iax_reg_state {
- REG_STATE_UNREGISTERED = 0,
- REG_STATE_REGSENT,
- REG_STATE_AUTHSENT,
- REG_STATE_REGISTERED,
- REG_STATE_REJECTED,
- REG_STATE_TIMEOUT,
- REG_STATE_NOAUTH
- };
- enum iax_transfer_state {
- TRANSFER_NONE = 0,
- TRANSFER_BEGIN,
- TRANSFER_READY,
- TRANSFER_RELEASED,
- TRANSFER_PASSTHROUGH,
- TRANSFER_MBEGIN,
- TRANSFER_MREADY,
- TRANSFER_MRELEASED,
- TRANSFER_MPASSTHROUGH,
- TRANSFER_MEDIA,
- TRANSFER_MEDIAPASS
- };
- struct iax2_registry {
- struct ast_sockaddr addr; /*!< Who we connect to for registration purposes */
- char username[80];
- char secret[80]; /*!< Password or key name in []'s */
- int expire; /*!< Sched ID of expiration */
- int refresh; /*!< How often to refresh */
- enum iax_reg_state regstate;
- int messages; /*!< Message count, low 8 bits = new, high 8 bits = old */
- int callno; /*!< Associated call number if applicable */
- struct sockaddr_in us; /*!< Who the server thinks we are */
- struct ast_dnsmgr_entry *dnsmgr; /*!< DNS refresh manager */
- AST_LIST_ENTRY(iax2_registry) entry;
- int port;
- char hostname[];
- };
- static AST_LIST_HEAD_STATIC(registrations, iax2_registry);
- /* Don't retry more frequently than every 10 ms, or less frequently than every 5 seconds */
- #define MIN_RETRY_TIME 100
- #define MAX_RETRY_TIME 10000
- #define MAX_JITTER_BUFFER 50
- #define MIN_JITTER_BUFFER 10
- #define DEFAULT_TRUNKDATA 640 * 10 /*!< 40ms, uncompressed linear * 10 channels */
- #define MAX_TIMESTAMP_SKEW 160 /*!< maximum difference between actual and predicted ts for sending */
- /* If consecutive voice frame timestamps jump by more than this many milliseconds, then jitter buffer will resync */
- #define TS_GAP_FOR_JB_RESYNC 5000
- /* used for first_iax_message and last_iax_message. If this bit is set it was TX, else RX */
- #define MARK_IAX_SUBCLASS_TX 0x8000
- static int iaxthreadcount = DEFAULT_THREAD_COUNT;
- static int iaxmaxthreadcount = DEFAULT_MAX_THREAD_COUNT;
- static int iaxdynamicthreadcount = 0;
- static int iaxdynamicthreadnum = 0;
- static int iaxactivethreadcount = 0;
- struct iax_rr {
- int jitter;
- int losspct;
- int losscnt;
- int packets;
- int delay;
- int dropped;
- int ooo;
- };
- struct iax2_pvt_ref;
- struct chan_iax2_pvt {
- /*! Socket to send/receive on for this call */
- int sockfd;
- /*! ast_callid bound to dialog */
- struct ast_callid *callid;
- /*! Last received voice format */
- iax2_format voiceformat;
- /*! Last received video format */
- iax2_format videoformat;
- /*! Last sent voice format */
- iax2_format svoiceformat;
- /*! Last sent video format */
- iax2_format svideoformat;
- /*! What we are capable of sending */
- iax2_format capability;
- /*! Last received timestamp */
- unsigned int last;
- /*! Last sent timestamp - never send the same timestamp twice in a single call */
- unsigned int lastsent;
- /*! Timestamp of the last video frame sent */
- unsigned int lastvsent;
- /*! Next outgoing timestamp if everything is good */
- unsigned int nextpred;
- /*! iax frame subclass that began iax2_pvt entry. 0x8000 bit is set on TX */
- int first_iax_message;
- /*! Last iax frame subclass sent or received for a iax2_pvt. 0x8000 bit is set on TX */
- int last_iax_message;
- /*! True if the last voice we transmitted was not silence/CNG */
- unsigned int notsilenttx:1;
- /*! Ping time */
- unsigned int pingtime;
- /*! Max time for initial response */
- int maxtime;
- /*! Peer Address */
- struct sockaddr_in addr;
- /*! Actual used codec preferences */
- struct ast_codec_pref prefs;
- /*! Requested codec preferences */
- struct ast_codec_pref rprefs;
- /*! Our call number */
- unsigned short callno;
- /*! Our callno_entry entry */
- struct callno_entry *callno_entry;
- /*! Peer callno */
- unsigned short peercallno;
- /*! Negotiated format, this is only used to remember what format was
- chosen for an unauthenticated call so that the channel can get
- created later using the right format */
- iax2_format chosenformat;
- /*! Peer selected format */
- iax2_format peerformat;
- /*! Peer capability */
- iax2_format peercapability;
- /*! timeval that we base our transmission on */
- struct timeval offset;
- /*! timeval that we base our delivery on */
- struct timeval rxcore;
- /*! The jitterbuffer */
- jitterbuf *jb;
- /*! active jb read scheduler id */
- int jbid;
- /*! LAG */
- int lag;
- /*! Error, as discovered by the manager */
- int error;
- /*! Owner if we have one */
- struct ast_channel *owner;
- /*! What's our state? */
- struct ast_flags state;
- /*! Expiry (optional) */
- int expiry;
- /*! Next outgoing sequence number */
- unsigned char oseqno;
- /*! Next sequence number they have not yet acknowledged */
- unsigned char rseqno;
- /*! Next incoming sequence number */
- unsigned char iseqno;
- /*! Last incoming sequence number we have acknowledged */
- unsigned char aseqno;
- AST_DECLARE_STRING_FIELDS(
- /*! Peer name */
- AST_STRING_FIELD(peer);
- /*! Default Context */
- AST_STRING_FIELD(context);
- /*! Caller ID if available */
- AST_STRING_FIELD(cid_num);
- AST_STRING_FIELD(cid_name);
- /*! Hidden Caller ID (i.e. ANI) if appropriate */
- AST_STRING_FIELD(ani);
- /*! DNID */
- AST_STRING_FIELD(dnid);
- /*! RDNIS */
- AST_STRING_FIELD(rdnis);
- /*! Requested Extension */
- AST_STRING_FIELD(exten);
- /*! Expected Username */
- AST_STRING_FIELD(username);
- /*! Expected Secret */
- AST_STRING_FIELD(secret);
- /*! MD5 challenge */
- AST_STRING_FIELD(challenge);
- /*! Public keys permitted keys for incoming authentication */
- AST_STRING_FIELD(inkeys);
- /*! Private key for outgoing authentication */
- AST_STRING_FIELD(outkey);
- /*! Preferred language */
- AST_STRING_FIELD(language);
- /*! Hostname/peername for naming purposes */
- AST_STRING_FIELD(host);
- AST_STRING_FIELD(dproot);
- AST_STRING_FIELD(accountcode);
- AST_STRING_FIELD(mohinterpret);
- AST_STRING_FIELD(mohsuggest);
- /*! received OSP token */
- AST_STRING_FIELD(osptoken);
- /*! Default parkinglot */
- AST_STRING_FIELD(parkinglot);
- );
- /*! AUTHREJ all AUTHREP frames */
- int authrej;
- /*! permitted authentication methods */
- int authmethods;
- /*! permitted encryption methods */
- int encmethods;
- /*! Encryption AES-128 Key */
- ast_aes_encrypt_key ecx;
- /*! Decryption AES-128 Key corresponding to ecx */
- ast_aes_decrypt_key mydcx;
- /*! Decryption AES-128 Key used to decrypt peer frames */
- ast_aes_decrypt_key dcx;
- /*! scheduler id associated with iax_key_rotate
- * for encrypted calls*/
- int keyrotateid;
- /*! 32 bytes of semi-random data */
- unsigned char semirand[32];
- /*! Associated registry */
- struct iax2_registry *reg;
- /*! Associated peer for poking */
- struct iax2_peer *peerpoke;
- /*! IAX_ flags */
- uint64_t flags;
- int adsi;
- /*! Transferring status */
- enum iax_transfer_state transferring;
- /*! Transfer identifier */
- int transferid;
- /*! Who we are IAX transferring to */
- struct sockaddr_in transfer;
- /*! What's the new call number for the transfer */
- unsigned short transfercallno;
- /*! Transfer encrypt AES-128 Key */
- ast_aes_encrypt_key tdcx;
- /*! Status of knowledge of peer ADSI capability */
- int peeradsicpe;
- /*! Callno of native bridge peer. (Valid if nonzero) */
- unsigned short bridgecallno;
- int pingid; /*!< Transmit PING request */
- int lagid; /*!< Retransmit lag request */
- int autoid; /*!< Auto hangup for Dialplan requestor */
- int authid; /*!< Authentication rejection ID */
- int authfail; /*!< Reason to report failure */
- int initid; /*!< Initial peer auto-congest ID (based on qualified peers) */
- int calling_ton;
- int calling_tns;
- int calling_pres;
- int amaflags;
- AST_LIST_HEAD_NOLOCK(, iax2_dpcache) dpentries;
- /*! variables inherited from the user definition */
- struct ast_variable *vars;
- /*! variables transmitted in a NEW packet */
- struct ast_variable *iaxvars;
- /*! last received remote rr */
- struct iax_rr remote_rr;
- /*! Current base time: (just for stats) */
- int min;
- /*! Dropped frame count: (just for stats) */
- int frames_dropped;
- /*! received frame count: (just for stats) */
- int frames_received;
- /*! num bytes used for calltoken ie, even an empty ie should contain 2 */
- unsigned char calltoken_ie_len;
- /*! hold all signaling frames from the pbx thread until we have a destination callno */
- char hold_signaling;
- /*! frame queue for signaling frames from pbx thread waiting for destination callno */
- AST_LIST_HEAD_NOLOCK(signaling_queue, signaling_queue_entry) signaling_queue;
- };
- struct signaling_queue_entry {
- struct ast_frame f;
- AST_LIST_ENTRY(signaling_queue_entry) next;
- };
- /*! table of available call numbers */
- static struct ao2_container *callno_pool;
- /*! table of available trunk call numbers */
- static struct ao2_container *callno_pool_trunk;
- static const unsigned int CALLNO_POOL_BUCKETS = 2699;
- /*!
- * \brief a list of frames that may need to be retransmitted
- *
- * \note The contents of this list do not need to be explicitly destroyed
- * on module unload. This is because all active calls are destroyed, and
- * all frames in this queue will get destroyed as a part of that process.
- *
- * \note Contents protected by the iaxsl[] locks
- */
- static AST_LIST_HEAD_NOLOCK(, iax_frame) frame_queue[IAX_MAX_CALLS];
- static struct ast_taskprocessor *transmit_processor;
- static int randomcalltokendata;
- static const time_t MAX_CALLTOKEN_DELAY = 10;
- /*!
- * This module will get much higher performance when doing a lot of
- * user and peer lookups if the number of buckets is increased from 1.
- * However, to maintain old behavior for Asterisk 1.4, these are set to
- * 1 by default. When using multiple buckets, search order through these
- * containers is considered random, so you will not be able to depend on
- * the order the entires are specified in iax.conf for matching order. */
- #ifdef LOW_MEMORY
- #define MAX_PEER_BUCKETS 17
- #else
- #define MAX_PEER_BUCKETS 563
- #endif
- static struct ao2_container *peers;
- #define MAX_USER_BUCKETS MAX_PEER_BUCKETS
- static struct ao2_container *users;
- /*! Table containing peercnt objects for every ip address consuming a callno */
- static struct ao2_container *peercnts;
- /*! Table containing custom callno limit rules for a range of ip addresses. */
- static struct ao2_container *callno_limits;
- /*! Table containing ip addresses not requiring calltoken validation */
- static struct ao2_container *calltoken_ignores;
- static uint16_t DEFAULT_MAXCALLNO_LIMIT = 2048;
- static uint16_t DEFAULT_MAXCALLNO_LIMIT_NONVAL = 8192;
- static uint16_t global_maxcallno;
- /*! Total num of call numbers allowed to be allocated without calltoken validation */
- static uint16_t global_maxcallno_nonval;
- static uint16_t total_nonval_callno_used = 0;
- /*! peer connection private, keeps track of all the call numbers
- * consumed by a single ip address */
- struct peercnt {
- /*! ip address consuming call numbers */
- unsigned long addr;
- /*! Number of call numbers currently used by this ip address */
- uint16_t cur;
- /*! Max call numbers allowed for this ip address */
- uint16_t limit;
- /*! Specifies whether limit is set by a registration or not, if so normal
- * limit setting rules do not apply to this address. */
- unsigned char reg;
- };
- /*! used by both callno_limits and calltoken_ignores containers */
- struct addr_range {
- /*! ip address range for custom callno limit rule */
- struct ast_ha ha;
- /*! callno limit for this ip address range, only used in callno_limits container */
- uint16_t limit;
- /*! delete me marker for reloads */
- unsigned char delme;
- };
- struct callno_entry {
- /*! callno used for this entry */
- uint16_t callno;
- /*! was this callno calltoken validated or not */
- unsigned char validated;
- };
- static AST_LIST_HEAD_STATIC(firmwares, iax_firmware);
- enum {
- /*! Extension exists */
- CACHE_FLAG_EXISTS = (1 << 0),
- /*! Extension is nonexistent */
- CACHE_FLAG_NONEXISTENT = (1 << 1),
- /*! Extension can exist */
- CACHE_FLAG_CANEXIST = (1 << 2),
- /*! Waiting to hear back response */
- CACHE_FLAG_PENDING = (1 << 3),
- /*! Timed out */
- CACHE_FLAG_TIMEOUT = (1 << 4),
- /*! Request transmitted */
- CACHE_FLAG_TRANSMITTED = (1 << 5),
- /*! Timeout */
- CACHE_FLAG_UNKNOWN = (1 << 6),
- /*! Matchmore */
- CACHE_FLAG_MATCHMORE = (1 << 7),
- };
- struct iax2_dpcache {
- char peercontext[AST_MAX_CONTEXT];
- char exten[AST_MAX_EXTENSION];
- struct timeval orig;
- struct timeval expiry;
- int flags;
- unsigned short callno;
- int waiters[256];
- AST_LIST_ENTRY(iax2_dpcache) cache_list;
- AST_LIST_ENTRY(iax2_dpcache) peer_list;
- };
- static AST_LIST_HEAD_STATIC(dpcache, iax2_dpcache);
- static void reg_source_db(struct iax2_peer *p);
- static struct iax2_peer *realtime_peer(const char *peername, struct sockaddr_in *sin);
- static struct iax2_user *realtime_user(const char *username, struct sockaddr_in *sin);
- static int ast_cli_netstats(struct mansession *s, int fd, int limit_fmt);
- static char *complete_iax2_peers(const char *line, const char *word, int pos, int state, uint64_t flags);
- static char *complete_iax2_unregister(const char *line, const char *word, int pos, int state);
- enum iax2_thread_iostate {
- IAX_IOSTATE_IDLE,
- IAX_IOSTATE_READY,
- IAX_IOSTATE_PROCESSING,
- IAX_IOSTATE_SCHEDREADY,
- };
- enum iax2_thread_type {
- IAX_THREAD_TYPE_POOL,
- IAX_THREAD_TYPE_DYNAMIC,
- };
- struct iax2_pkt_buf {
- AST_LIST_ENTRY(iax2_pkt_buf) entry;
- size_t len;
- unsigned char buf[1];
- };
- struct iax2_thread {
- AST_LIST_ENTRY(iax2_thread) list;
- enum iax2_thread_type type;
- enum iax2_thread_iostate iostate;
- #ifdef SCHED_MULTITHREADED
- void (*schedfunc)(const void *);
- const void *scheddata;
- #endif
- #ifdef DEBUG_SCHED_MULTITHREAD
- char curfunc[80];
- #endif
- int actions;
- pthread_t threadid;
- int threadnum;
- struct sockaddr_in iosin;
- unsigned char readbuf[4096];
- unsigned char *buf;
- ssize_t buf_len;
- size_t buf_size;
- int iofd;
- time_t checktime;
- ast_mutex_t lock;
- ast_cond_t cond;
- ast_mutex_t init_lock;
- ast_cond_t init_cond;
- /*! if this thread is processing a full frame,
- some information about that frame will be stored
- here, so we can avoid dispatching any more full
- frames for that callno to other threads */
- struct {
- unsigned short callno;
- struct sockaddr_in sin;
- unsigned char type;
- unsigned char csub;
- } ffinfo;
- /*! Queued up full frames for processing. If more full frames arrive for
- * a call which this thread is already processing a full frame for, they
- * are queued up here. */
- AST_LIST_HEAD_NOLOCK(, iax2_pkt_buf) full_frames;
- unsigned char stop;
- };
- /* Thread lists */
- static AST_LIST_HEAD_STATIC(idle_list, iax2_thread);
- static AST_LIST_HEAD_STATIC(active_list, iax2_thread);
- static AST_LIST_HEAD_STATIC(dynamic_list, iax2_thread);
- static void *iax2_process_thread(void *data);
- static void iax2_destroy(int callno);
- static void signal_condition(ast_mutex_t *lock, ast_cond_t *cond)
- {
- ast_mutex_lock(lock);
- ast_cond_signal(cond);
- ast_mutex_unlock(lock);
- }
- /*!
- * \brief an array of iax2 pvt structures
- *
- * The container for active chan_iax2_pvt structures is implemented as an
- * array for extremely quick direct access to the correct pvt structure
- * based on the local call number. The local call number is used as the
- * index into the array where the associated pvt structure is stored.
- */
- static struct chan_iax2_pvt *iaxs[IAX_MAX_CALLS];
- static struct ast_callid *iax_pvt_callid_get(int callno)
- {
- if (iaxs[callno]->callid) {
- return ast_callid_ref(iaxs[callno]->callid);
- }
- return NULL;
- }
- static void iax_pvt_callid_set(int callno, struct ast_callid *callid)
- {
- if (iaxs[callno]->callid) {
- ast_callid_unref(iaxs[callno]->callid);
- }
- ast_callid_ref(callid);
- iaxs[callno]->callid = callid;
- }
- static void iax_pvt_callid_new(int callno)
- {
- struct ast_callid *callid = ast_create_callid();
- char buffer[AST_CALLID_BUFFER_LENGTH];
- ast_callid_strnprint(buffer, sizeof(buffer), callid);
- iax_pvt_callid_set(callno, callid);
- ast_callid_unref(callid);
- }
- /*!
- * \brief Another container of iax2_pvt structures
- *
- * Active IAX2 pvt structs are also stored in this container, if they are a part
- * of an active call where we know the remote side's call number. The reason
- * for this is that incoming media frames do not contain our call number. So,
- * instead of having to iterate the entire iaxs array, we use this container to
- * look up calls where the remote side is using a given call number.
- */
- static struct ao2_container *iax_peercallno_pvts;
- /*!
- * \brief chan_iax2_pvt structure locks
- *
- * These locks are used when accessing a pvt structure in the iaxs array.
- * The index used here is the same as used in the iaxs array. It is the
- * local call number for the associated pvt struct.
- */
- static ast_mutex_t iaxsl[ARRAY_LEN(iaxs)];
- /*!
- * * \brief Another container of iax2_pvt structures
- *
- * Active IAX2 pvt stucts used during transfering a call are stored here.
- */
- static struct ao2_container *iax_transfercallno_pvts;
- /* Flag to use with trunk calls, keeping these calls high up. It halves our effective use
- but keeps the division between trunked and non-trunked better. */
- #define TRUNK_CALL_START (IAX_MAX_CALLS / 2)
- /* Debug routines... */
- static struct sockaddr_in debugaddr;
- static void iax_outputframe(struct iax_frame *f, struct ast_iax2_full_hdr *fhi, int rx, struct sockaddr_in *sin, int datalen)
- {
- if (iaxdebug ||
- (sin && debugaddr.sin_addr.s_addr &&
- (!ntohs(debugaddr.sin_port) ||
- debugaddr.sin_port == sin->sin_port) &&
- debugaddr.sin_addr.s_addr == sin->sin_addr.s_addr)) {
- if (iaxdebug) {
- iax_showframe(f, fhi, rx, sin, datalen);
- } else {
- iaxdebug = 1;
- iax_showframe(f, fhi, rx, sin, datalen);
- iaxdebug = 0;
- }
- }
- }
- static void iax_debug_output(const char *data)
- {
- if (iaxdebug)
- ast_verbose("%s", data);
- }
- static void iax_error_output(const char *data)
- {
- ast_log(LOG_WARNING, "%s", data);
- }
- static void __attribute__((format(printf, 1, 2))) jb_error_output(const char *fmt, ...)
- {
- va_list args;
- char buf[1024];
- va_start(args, fmt);
- vsnprintf(buf, sizeof(buf), fmt, args);
- va_end(args);
- ast_log(LOG_ERROR, "%s", buf);
- }
- static void __attribute__((format(printf, 1, 2))) jb_warning_output(const char *fmt, ...)
- {
- va_list args;
- char buf[1024];
- va_start(args, fmt);
- vsnprintf(buf, sizeof(buf), fmt, args);
- va_end(args);
- ast_log(LOG_WARNING, "%s", buf);
- }
- static void __attribute__((format(printf, 1, 2))) jb_debug_output(const char *fmt, ...)
- {
- va_list args;
- char buf[1024];
- va_start(args, fmt);
- vsnprintf(buf, sizeof(buf), fmt, args);
- va_end(args);
- ast_verbose("%s", buf);
- }
- static enum ast_bridge_result iax2_bridge(struct ast_channel *c0, struct ast_channel *c1, int flags, struct ast_frame **fo, struct ast_channel **rc, int timeoutms);
- static int expire_registry(const void *data);
- static int iax2_answer(struct ast_channel *c);
- static int iax2_call(struct ast_channel *c, const char *dest, int timeout);
- static int iax2_devicestate(const char *data);
- static int iax2_digit_begin(struct ast_channel *c, char digit);
- static int iax2_digit_end(struct ast_channel *c, char digit, unsigned int duration);
- static int iax2_do_register(struct iax2_registry *reg);
- static int iax2_fixup(struct ast_channel *oldchannel, struct ast_channel *newchan);
- static int iax2_hangup(struct ast_channel *c);
- static int iax2_indicate(struct ast_channel *c, int condition, const void *data, size_t datalen);
- static int iax2_poke_peer(struct iax2_peer *peer, int heldcall);
- static int iax2_provision(struct sockaddr_in *end, int sockfd, const char *dest, const char *template, int force);
- static int iax2_send(struct chan_iax2_pvt *pvt, struct ast_frame *f, unsigned int ts, int seqno, int now, int transfer, int final);
- static int iax2_sendhtml(struct ast_channel *c, int subclass, const char *data, int datalen);
- static int iax2_sendimage(struct ast_channel *c, struct ast_frame *img);
- static int iax2_sendtext(struct ast_channel *c, const char *text);
- static int iax2_setoption(struct ast_channel *c, int option, void *data, int datalen);
- static int iax2_queryoption(struct ast_channel *c, int option, void *data, int *datalen);
- static int iax2_transfer(struct ast_channel *c, const char *dest);
- static int iax2_write(struct ast_channel *c, struct ast_frame *f);
- static int iax2_sched_add(struct ast_sched_context *sched, int when, ast_sched_cb callback, const void *data);
- static int send_trunk(struct iax2_trunk_peer *tpeer, struct timeval *now);
- static int send_command(struct chan_iax2_pvt *, char, int, unsigned int, const unsigned char *, int, int);
- static int send_command_final(struct chan_iax2_pvt *, char, int, unsigned int, const unsigned char *, int, int);
- static int send_command_immediate(struct chan_iax2_pvt *, char, int, unsigned int, const unsigned char *, int, int);
- static int send_command_locked(unsigned short callno, char, int, unsigned int, const unsigned char *, int, int);
- static int send_command_transfer(struct chan_iax2_pvt *, char, int, unsigned int, const unsigned char *, int);
- static struct ast_channel *iax2_request(const char *type, struct ast_format_cap *cap, const struct ast_channel *requestor, const char *data, int *cause);
- static struct ast_frame *iax2_read(struct ast_channel *c);
- static struct iax2_peer *build_peer(const char *name, struct ast_variable *v, struct ast_variable *alt, int temponly);
- static struct iax2_user *build_user(const char *name, struct ast_variable *v, struct ast_variable *alt, int temponly);
- static void realtime_update_peer(const char *peername, struct ast_sockaddr *sockaddr, time_t regtime);
- static void *iax2_dup_variable_datastore(void *);
- static void prune_peers(void);
- static void prune_users(void);
- static void iax2_free_variable_datastore(void *);
- static int acf_channel_read(struct ast_channel *chan, const char *funcname, char *preparse, char *buf, size_t buflen);
- static int decode_frame(ast_aes_decrypt_key *dcx, struct ast_iax2_full_hdr *fh, struct ast_frame *f, int *datalen);
- static int encrypt_frame(ast_aes_encrypt_key *ecx, struct ast_iax2_full_hdr *fh, unsigned char *poo, int *datalen);
- static void build_ecx_key(const unsigned char *digest, struct chan_iax2_pvt *pvt);
- static void build_rand_pad(unsigned char *buf, ssize_t len);
- static struct callno_entry *get_unused_callno(int trunk, int validated);
- static int replace_callno(const void *obj);
- static void sched_delay_remove(struct sockaddr_in *sin, struct callno_entry *callno_entry);
- static void network_change_event_cb(const struct ast_event *, void *);
- static void acl_change_event_cb(const struct ast_event *, void *);
- static struct ast_channel_tech iax2_tech = {
- .type = "IAX2",
- .description = tdesc,
- .properties = AST_CHAN_TP_WANTSJITTER,
- .requester = iax2_request,
- .devicestate = iax2_devicestate,
- .send_digit_begin = iax2_digit_begin,
- .send_digit_end = iax2_digit_end,
- .send_text = iax2_sendtext,
- .send_image = iax2_sendimage,
- .send_html = iax2_sendhtml,
- .call = iax2_call,
- .hangup = iax2_hangup,
- .answer = iax2_answer,
- .read = iax2_read,
- .write = iax2_write,
- .write_video = iax2_write,
- .indicate = iax2_indicate,
- .setoption = iax2_setoption,
- .queryoption = iax2_queryoption,
- .bridge = iax2_bridge,
- .transfer = iax2_transfer,
- .fixup = iax2_fixup,
- .func_channel_read = acf_channel_read,
- };
- /*!
- * \internal
- * \brief Obtain the owner channel lock if the owner exists.
- *
- * \param callno IAX2 call id.
- *
- * \note Assumes the iaxsl[callno] lock is already obtained.
- *
- * \note
- * IMPORTANT NOTE!!! Any time this function is used, even if
- * iaxs[callno] was valid before calling it, it may no longer be
- * valid after calling it. This function may unlock and lock
- * the mutex associated with this callno, meaning that another
- * thread may grab it and destroy the call.
- *
- * \return Nothing
- */
- static void iax2_lock_owner(int callno)
- {
- for (;;) {
- if (!iaxs[callno] || !iaxs[callno]->owner) {
- /* There is no owner lock to get. */
- break;
- }
- if (!ast_channel_trylock(iaxs[callno]->owner)) {
- /* We got the lock */
- break;
- }
- /* Avoid deadlock by pausing and trying again */
- DEADLOCK_AVOIDANCE(&iaxsl[callno]);
- }
- }
- /*!
- * \internal
- * \brief Check if a control subtype is allowed on the wire.
- *
- * \param subtype Control frame subtype to check if allowed to/from the wire.
- *
- * \retval non-zero if allowed.
- */
- static int iax2_is_control_frame_allowed(int subtype)
- {
- enum ast_control_frame_type control = subtype;
- int is_allowed;
- /*
- * Note: If we compare the enumeration type, which does not have any
- * negative constants, the compiler may optimize this code away.
- * Therefore, we must perform an integer comparison here.
- */
- if (subtype == -1) {
- return -1;
- }
- /* Default to not allowing control frames to pass. */
- is_allowed = 0;
- /*
- * The switch default is not present in order to take advantage
- * of the compiler complaining of a missing enum case.
- */
- switch (control) {
- /*
- * These control frames make sense to send/receive across the link.
- */
- case AST_CONTROL_HANGUP:
- case AST_CONTROL_RING:
- case AST_CONTROL_RINGING:
- case AST_CONTROL_ANSWER:
- case AST_CONTROL_BUSY:
- case AST_CONTROL_TAKEOFFHOOK:
- case AST_CONTROL_OFFHOOK:
- case AST_CONTROL_CONGESTION:
- case AST_CONTROL_FLASH:
- case AST_CONTROL_WINK:
- case AST_CONTROL_OPTION:
- case AST_CONTROL_RADIO_KEY:
- case AST_CONTROL_RADIO_UNKEY:
- case AST_CONTROL_PROGRESS:
- case AST_CONTROL_PROCEEDING:
- case AST_CONTROL_HOLD:
- case AST_CONTROL_UNHOLD:
- case AST_CONTROL_VIDUPDATE:
- case AST_CONTROL_CONNECTED_LINE:
- case AST_CONTROL_REDIRECTING:
- case AST_CONTROL_T38_PARAMETERS:
- case AST_CONTROL_AOC:
- case AST_CONTROL_INCOMPLETE:
- case AST_CONTROL_MCID:
- is_allowed = -1;
- break;
- /*
- * These control frames do not make sense to send/receive across the link.
- */
- case _XXX_AST_CONTROL_T38:
- /* The control value is deprecated in favor of AST_CONTROL_T38_PARAMETERS. */
- case AST_CONTROL_SRCUPDATE:
- /* Across an IAX link the source is still the same. */
- case AST_CONTROL_TRANSFER:
- /* A success/fail status report from calling ast_transfer() on this machine. */
- case AST_CONTROL_CC:
- /* The payload contains pointers that are valid for the sending machine only. */
- case AST_CONTROL_SRCCHANGE:
- /* Across an IAX link the source is still the same. */
- case AST_CONTROL_READ_ACTION:
- /* The action can only be done by the sending machine. */
- case AST_CONTROL_END_OF_Q:
- /* This frame would cause the call to unexpectedly hangup. */
- case AST_CONTROL_UPDATE_RTP_PEER:
- /* Only meaningful across a bridge on this machine for direct-media exchange. */
- case AST_CONTROL_PVT_CAUSE_CODE:
- /* Intended only for the sending machine's local channel structure. */
- break;
- }
- return is_allowed;
- }
- static void mwi_event_cb(const struct ast_event *event, void *userdata)
- {
- /* The MWI subscriptions exist just so the core knows we care about those
- * mailboxes. However, we just grab the events out of the cache when it
- * is time to send MWI, since it is only sent with a REGACK. */
- }
- static void network_change_event_subscribe(void)
- {
- if (!network_change_event_subscription) {
- network_change_event_subscription = ast_event_subscribe(AST_EVENT_NETWORK_CHANGE,
- network_change_event_cb, "IAX2 Network Change", NULL, AST_EVENT_IE_END);
- }
- }
- static void network_change_event_unsubscribe(void)
- {
- if (network_change_event_subscription) {
- network_change_event_subscription = ast_event_unsubscribe(network_change_event_subscription);
- }
- }
- static void acl_change_event_subscribe(void)
- {
- if (!acl_change_event_subscription) {
- acl_change_event_subscription = ast_event_subscribe(AST_EVENT_ACL_CHANGE,
- acl_change_event_cb, "IAX2 ACL Change", NULL, AST_EVENT_IE_END);
- }
- }
- static void acl_change_event_unsubscribe(void)
- {
- if (acl_change_event_subscription) {
- acl_change_event_subscription = ast_event_unsubscribe(acl_change_event_subscription);
- }
- }
- static int network_change_event_sched_cb(const void *data)
- {
- struct iax2_registry *reg;
- network_change_event_sched_id = -1;
- AST_LIST_LOCK(®istrations);
- AST_LIST_TRAVERSE(®istrations, reg, entry) {
- iax2_do_register(reg);
- }
- AST_LIST_UNLOCK(®istrations);
- return 0;
- }
- static void network_change_event_cb(const struct ast_event *event, void *userdata)
- {
- ast_debug(1, "IAX, got a network change event, renewing all IAX registrations.\n");
- if (network_change_event_sched_id == -1) {
- network_change_event_sched_id = iax2_sched_add(sched, 1000, network_change_event_sched_cb, NULL);
- }
- }
- static void acl_change_event_cb(const struct ast_event *event, void *userdata)
- {
- ast_log(LOG_NOTICE, "Reloading chan_iax2 in response to ACL change event.\n");
- reload_config(1);
- }
- /*! \brief Send manager event at call setup to link between Asterisk channel name
- and IAX2 call identifiers */
- static void iax2_ami_channelupdate(struct chan_iax2_pvt *pvt)
- {
- manager_event(EVENT_FLAG_SYSTEM, "ChannelUpdate",
- "Channel: %s\r\nChanneltype: IAX2\r\nIAX2-callno-local: %d\r\nIAX2-callno-remote: %d\r\nIAX2-peer: %s\r\n",
- pvt->owner ? ast_channel_name(pvt->owner) : "",
- pvt->callno, pvt->peercallno, pvt->peer ? pvt->peer : "");
- }
- static const struct ast_datastore_info iax2_variable_datastore_info = {
- .type = "IAX2_VARIABLE",
- .duplicate = iax2_dup_variable_datastore,
- .destroy = iax2_free_variable_datastore,
- };
- static void *iax2_dup_variable_datastore(void *old)
- {
- AST_LIST_HEAD(, ast_var_t) *oldlist = old, *newlist;
- struct ast_var_t *oldvar, *newvar;
- newlist = ast_calloc(sizeof(*newlist), 1);
- if (!newlist) {
- ast_log(LOG_ERROR, "Unable to duplicate iax2 variables\n");
- return NULL;
- }
- AST_LIST_HEAD_INIT(newlist);
- AST_LIST_LOCK(oldlist);
- AST_LIST_TRAVERSE(oldlist, oldvar, entries) {
- newvar = ast_var_assign(ast_var_name(oldvar), ast_var_value(oldvar));
- if (newvar)
- AST_LIST_INSERT_TAIL(newlist, newvar, entries);
- else
- ast_log(LOG_ERROR, "Unable to duplicate iax2 variable '%s'\n", ast_var_name(oldvar));
- }
- AST_LIST_UNLOCK(oldlist);
- return newlist;
- }
- static void iax2_free_variable_datastore(void *old)
- {
- AST_LIST_HEAD(, ast_var_t) *oldlist = old;
- struct ast_var_t *oldvar;
- AST_LIST_LOCK(oldlist);
- while ((oldvar = AST_LIST_REMOVE_HEAD(oldlist, entries))) {
- ast_free(oldvar);
- }
- AST_LIST_UNLOCK(oldlist);
- AST_LIST_HEAD_DESTROY(oldlist);
- ast_free(oldlist);
- }
- /* WARNING: insert_idle_thread should only ever be called within the
- * context of an iax2_process_thread() thread.
- */
- static void insert_idle_thread(struct iax2_thread *thread)
- {
- if (thread->type == IAX_THREAD_TYPE_DYNAMIC) {
- AST_LIST_LOCK(&dynamic_list);
- AST_LIST_INSERT_TAIL(&dynamic_list, thread, list);
- AST_LIST_UNLOCK(&dynamic_list);
- } else {
- AST_LIST_LOCK(&idle_list);
- AST_LIST_INSERT_TAIL(&idle_list, thread, list);
- AST_LIST_UNLOCK(&idle_list);
- }
- return;
- }
- static struct iax2_thread *find_idle_thread(void)
- {
- struct iax2_thread *thread = NULL;
- /* Pop the head of the idle list off */
- AST_LIST_LOCK(&idle_list);
- thread = AST_LIST_REMOVE_HEAD(&idle_list, list);
- AST_LIST_UNLOCK(&idle_list);
- /* If we popped a thread off the idle list, just return it */
- if (thread) {
- memset(&thread->ffinfo, 0, sizeof(thread->ffinfo));
- return thread;
- }
- /* Pop the head of the dynamic list off */
- AST_LIST_LOCK(&dynamic_list);
- thread = AST_LIST_REMOVE_HEAD(&dynamic_list, list);
- AST_LIST_UNLOCK(&dynamic_list);
- /* If we popped a thread off the dynamic list, just return it */
- if (thread) {
- memset(&thread->ffinfo, 0, sizeof(thread->ffinfo));
- return thread;
- }
- /* If we can't create a new dynamic thread for any reason, return no thread at all */
- if (iaxdynamicthreadcount >= iaxmaxthreadcount || !(thread = ast_calloc(1, sizeof(*thread))))
- return NULL;
- /* Set default values */
- ast_atomic_fetchadd_int(&iaxdynamicthreadcount, 1);
- thread->threadnum = ast_atomic_fetchadd_int(&iaxdynamicthreadnum, 1);
- thread->type = IAX_THREAD_TYPE_DYNAMIC;
- /* Initialize lock and condition */
- ast_mutex_init(&thread->lock);
- ast_cond_init(&thread->cond, NULL);
- ast_mutex_init(&thread->init_lock);
- ast_cond_init(&thread->init_cond, NULL);
- ast_mutex_lock(&thread->init_lock);
- /* Create thread and send it on it's way */
- if (ast_pthread_create_background(&thread->threadid, NULL, iax2_process_thread, thread)) {
- ast_cond_destroy(&thread->cond);
- ast_mutex_destroy(&thread->lock);
- ast_mutex_unlock(&thread->init_lock);
- ast_cond_destroy(&thread->init_cond);
- ast_mutex_destroy(&thread->init_lock);
- ast_free(thread);
- return NULL;
- }
- /* this thread is not processing a full frame (since it is idle),
- so ensure that the field for the full frame call number is empty */
- memset(&thread->ffinfo, 0, sizeof(thread->ffinfo));
- /* Wait for the thread to be ready before returning it to the caller */
- ast_cond_wait(&thread->init_cond, &thread->init_lock);
- /* Done with init_lock */
- ast_mutex_unlock(&thread->init_lock);
- return thread;
- }
- #ifdef SCHED_MULTITHREADED
- static int __schedule_action(void (*func)(const void *data), const void *data, const char *funcname)
- {
- struct iax2_thread *thread;
- static time_t lasterror;
- time_t t;
- thread = find_idle_thread();
- if (thread != NULL) {
- thread->schedfunc = func;
- thread->scheddata = data;
- thread->iostate = IAX_IOSTATE_SCHEDREADY;
- #ifdef DEBUG_SCHED_MULTITHREAD
- ast_copy_string(thread->curfunc, funcname, sizeof(thread->curfunc));
- #endif
- signal_condition(&thread->lock, &thread->cond);
- return 0;
- }
- time(&t);
- if (t != lasterror) {
- lasterror = t;
- ast_debug(1, "Out of idle IAX2 threads for scheduling! (%s)\n", funcname);
- }
- return -1;
- }
- #define schedule_action(func, data) __schedule_action(func, data, __PRETTY_FUNCTION__)
- #endif
- static int iax2_sched_replace(int id, struct ast_sched_context *con, int when,
- ast_sched_cb callback, const void *data)
- {
- return ast_sched_replace(id, con, when, callback, data);
- }
- static int iax2_sched_add(struct ast_sched_context *con, int when,
- ast_sched_cb callback, const void *data)
- {
- return ast_sched_add(con, when, callback, data);
- }
- static int send_ping(const void *data);
- static void __send_ping(const void *data)
- {
- int callno = (long) data;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- if (iaxs[callno]->peercallno) {
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_PING, 0, NULL, 0, -1);
- if (iaxs[callno]->pingid != DONT_RESCHEDULE) {
- iaxs[callno]->pingid = iax2_sched_add(sched, ping_time * 1000, send_ping, data);
- }
- }
- } else {
- ast_debug(1, "I was supposed to send a PING with callno %d, but no such call exists.\n", callno);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int send_ping(const void *data)
- {
- int callno = (long) data;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno] && iaxs[callno]->pingid != DONT_RESCHEDULE) {
- iaxs[callno]->pingid = -1;
- }
- ast_mutex_unlock(&iaxsl[callno]);
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__send_ping, data))
- #endif
- __send_ping(data);
- return 0;
- }
- static void encmethods_to_str(int e, struct ast_str **buf)
- {
- ast_str_set(buf, 0, "(");
- if (e & IAX_ENCRYPT_AES128) {
- ast_str_append(buf, 0, "aes128");
- }
- if (e & IAX_ENCRYPT_KEYROTATE) {
- ast_str_append(buf, 0, ",keyrotate");
- }
- if (ast_str_strlen(*buf) > 1) {
- ast_str_append(buf, 0, ")");
- } else {
- ast_str_set(buf, 0, "No");
- }
- }
- static int get_encrypt_methods(const char *s)
- {
- int e;
- if (!strcasecmp(s, "aes128"))
- e = IAX_ENCRYPT_AES128 | IAX_ENCRYPT_KEYROTATE;
- else if (ast_true(s))
- e = IAX_ENCRYPT_AES128 | IAX_ENCRYPT_KEYROTATE;
- else
- e = 0;
- return e;
- }
- static int send_lagrq(const void *data);
- static void __send_lagrq(const void *data)
- {
- int callno = (long) data;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- if (iaxs[callno]->peercallno) {
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_LAGRQ, 0, NULL, 0, -1);
- if (iaxs[callno]->lagid != DONT_RESCHEDULE) {
- iaxs[callno]->lagid = iax2_sched_add(sched, lagrq_time * 1000, send_lagrq, data);
- }
- }
- } else {
- ast_debug(1, "I was supposed to send a LAGRQ with callno %d, but no such call exists.\n", callno);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int send_lagrq(const void *data)
- {
- int callno = (long) data;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno] && iaxs[callno]->lagid != DONT_RESCHEDULE) {
- iaxs[callno]->lagid = -1;
- }
- ast_mutex_unlock(&iaxsl[callno]);
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__send_lagrq, data))
- #endif
- __send_lagrq(data);
- return 0;
- }
- static unsigned char compress_subclass(iax2_format subclass)
- {
- int x;
- int power=-1;
- /* If it's 64 or smaller, just return it */
- if (subclass < IAX_FLAG_SC_LOG)
- return subclass;
- /* Otherwise find its power */
- for (x = 0; x < IAX_MAX_SHIFT; x++) {
- if (subclass & (1LL << x)) {
- if (power > -1) {
- ast_log(LOG_WARNING, "Can't compress subclass %lld\n", (long long) subclass);
- return 0;
- } else
- power = x;
- }
- }
- return power | IAX_FLAG_SC_LOG;
- }
- static iax2_format uncompress_subclass(unsigned char csub)
- {
- /* If the SC_LOG flag is set, return 2^csub otherwise csub */
- if (csub & IAX_FLAG_SC_LOG) {
- /* special case for 'compressed' -1 */
- if (csub == 0xff)
- return -1;
- else
- return 1LL << (csub & ~IAX_FLAG_SC_LOG & IAX_MAX_SHIFT);
- }
- else
- return csub;
- }
- static iax2_format iax2_codec_choose(struct ast_codec_pref *pref, iax2_format formats, int find_best)
- {
- struct ast_format_cap *cap;
- struct ast_format tmpfmt;
- iax2_format format = 0;
- if ((cap = ast_format_cap_alloc_nolock())) {
- ast_format_clear(&tmpfmt);
- ast_format_cap_from_old_bitfield(cap, formats);
- ast_codec_choose(pref, cap, find_best, &tmpfmt);
- format = ast_format_to_old_bitfield(&tmpfmt);
- cap = ast_format_cap_destroy(cap);
- }
- return format;
- }
- static iax2_format iax2_best_codec(iax2_format formats)
- {
- struct ast_format_cap *cap = ast_format_cap_alloc_nolock();
- struct ast_format tmpfmt;
- if (!cap) {
- return 0;
- }
- ast_format_clear(&tmpfmt);
- ast_format_cap_from_old_bitfield(cap, formats);
- ast_best_codec(cap, &tmpfmt);
- cap = ast_format_cap_destroy(cap);
- return ast_format_to_old_bitfield(&tmpfmt);
- }
- const char *iax2_getformatname(iax2_format format)
- {
- struct ast_format tmpfmt;
- if (!(ast_format_from_old_bitfield(&tmpfmt, format))) {
- return "Unknown";
- }
- return ast_getformatname(&tmpfmt);
- }
- static char *iax2_getformatname_multiple(char *codec_buf, size_t len, iax2_format format)
- {
- struct ast_format_cap *cap = ast_format_cap_alloc_nolock();
- if (!cap) {
- return "(Nothing)";
- }
- ast_format_cap_from_old_bitfield(cap, format);
- ast_getformatname_multiple(codec_buf, len, cap);
- cap = ast_format_cap_destroy(cap);
- return codec_buf;
- }
- static int iax2_parse_allow_disallow(struct ast_codec_pref *pref, iax2_format *formats, const char *list, int allowing)
- {
- int res;
- struct ast_format_cap *cap = ast_format_cap_alloc_nolock();
- if (!cap) {
- return 1;
- }
- ast_format_cap_from_old_bitfield(cap, *formats);
- res = ast_parse_allow_disallow(pref, cap, list, allowing);
- *formats = ast_format_cap_to_old_bitfield(cap);
- cap = ast_format_cap_destroy(cap);
- return res;
- }
- static int iax2_data_add_codecs(struct ast_data *root, const char *node_name, iax2_format formats)
- {
- int res;
- struct ast_format_cap *cap = ast_format_cap_alloc_nolock();
- if (!cap) {
- return -1;
- }
- ast_format_cap_from_old_bitfield(cap, formats);
- res = ast_data_add_codecs(root, node_name, cap);
- cap = ast_format_cap_destroy(cap);
- return res;
- }
- /*!
- * \note The only member of the peer passed here guaranteed to be set is the name field
- */
- static int peer_hash_cb(const void *obj, const int flags)
- {
- const struct iax2_peer *peer = obj;
- const char *name = obj;
- return ast_str_hash(flags & OBJ_KEY ? name : peer->name);
- }
- /*!
- * \note The only member of the peer passed here guaranteed to be set is the name field
- */
- static int peer_cmp_cb(void *obj, void *arg, int flags)
- {
- struct iax2_peer *peer = obj, *peer2 = arg;
- const char *name = arg;
- return !strcmp(peer->name, flags & OBJ_KEY ? name : peer2->name) ?
- CMP_MATCH | CMP_STOP : 0;
- }
- /*!
- * \note The only member of the user passed here guaranteed to be set is the name field
- */
- static int user_hash_cb(const void *obj, const int flags)
- {
- const struct iax2_user *user = obj;
- const char *name = obj;
- return ast_str_hash(flags & OBJ_KEY ? name : user->name);
- }
- /*!
- * \note The only member of the user passed here guaranteed to be set is the name field
- */
- static int user_cmp_cb(void *obj, void *arg, int flags)
- {
- struct iax2_user *user = obj, *user2 = arg;
- const char *name = arg;
- return !strcmp(user->name, flags & OBJ_KEY ? name : user2->name) ?
- CMP_MATCH | CMP_STOP : 0;
- }
- /*!
- * \note This funtion calls realtime_peer -> reg_source_db -> iax2_poke_peer -> find_callno,
- * so do not call it with a pvt lock held.
- */
- static struct iax2_peer *find_peer(const char *name, int realtime)
- {
- struct iax2_peer *peer = NULL;
- peer = ao2_find(peers, name, OBJ_KEY);
- /* Now go for realtime if applicable */
- if(!peer && realtime)
- peer = realtime_peer(name, NULL);
- return peer;
- }
- static struct iax2_peer *peer_ref(struct iax2_peer *peer)
- {
- ao2_ref(peer, +1);
- return peer;
- }
- static inline struct iax2_peer *peer_unref(struct iax2_peer *peer)
- {
- ao2_ref(peer, -1);
- return NULL;
- }
- static struct iax2_user *find_user(const char *name)
- {
- return ao2_find(users, name, OBJ_KEY);
- }
- static inline struct iax2_user *user_unref(struct iax2_user *user)
- {
- ao2_ref(user, -1);
- return NULL;
- }
- static int iax2_getpeername(struct sockaddr_in sin, char *host, int len)
- {
- struct iax2_peer *peer = NULL;
- int res = 0;
- struct ao2_iterator i;
- i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- struct sockaddr_in peer_addr;
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- if ((peer_addr.sin_addr.s_addr == sin.sin_addr.s_addr) &&
- (peer_addr.sin_port == sin.sin_port)) {
- ast_copy_string(host, peer->name, len);
- peer_unref(peer);
- res = 1;
- break;
- }
- peer_unref(peer);
- }
- ao2_iterator_destroy(&i);
- if (!peer) {
- peer = realtime_peer(NULL, &sin);
- if (peer) {
- ast_copy_string(host, peer->name, len);
- peer_unref(peer);
- res = 1;
- }
- }
- return res;
- }
- /*!\note Assumes the lock on the pvt is already held, when
- * iax2_destroy_helper() is called. */
- static void iax2_destroy_helper(struct chan_iax2_pvt *pvt)
- {
- /* Decrement AUTHREQ count if needed */
- if (ast_test_flag64(pvt, IAX_MAXAUTHREQ)) {
- struct iax2_user *user;
- user = ao2_find(users, pvt->username, OBJ_KEY);
- if (user) {
- ast_atomic_fetchadd_int(&user->curauthreq, -1);
- user_unref(user);
- }
- ast_clear_flag64(pvt, IAX_MAXAUTHREQ);
- }
- /* No more pings or lagrq's */
- AST_SCHED_DEL_SPINLOCK(sched, pvt->pingid, &iaxsl[pvt->callno]);
- pvt->pingid = DONT_RESCHEDULE;
- AST_SCHED_DEL_SPINLOCK(sched, pvt->lagid, &iaxsl[pvt->callno]);
- pvt->lagid = DONT_RESCHEDULE;
- AST_SCHED_DEL(sched, pvt->autoid);
- AST_SCHED_DEL(sched, pvt->authid);
- AST_SCHED_DEL(sched, pvt->initid);
- AST_SCHED_DEL(sched, pvt->jbid);
- AST_SCHED_DEL(sched, pvt->keyrotateid);
- }
- static void iax2_frame_free(struct iax_frame *fr)
- {
- AST_SCHED_DEL(sched, fr->retrans);
- iax_frame_free(fr);
- }
- static int scheduled_destroy(const void *vid)
- {
- unsigned short callno = PTR_TO_CALLNO(vid);
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- ast_debug(1, "Really destroying %d now...\n", callno);
- iax2_destroy(callno);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- return 0;
- }
- static void free_signaling_queue_entry(struct signaling_queue_entry *s)
- {
- if (s->f.datalen) {
- ast_free(s->f.data.ptr);
- }
- ast_free(s);
- }
- /*! \brief This function must be called once we are sure the other side has
- * given us a call number. All signaling is held here until that point. */
- static void send_signaling(struct chan_iax2_pvt *pvt)
- {
- struct signaling_queue_entry *s = NULL;
- while ((s = AST_LIST_REMOVE_HEAD(&pvt->signaling_queue, next))) {
- iax2_send(pvt, &s->f, 0, -1, 0, 0, 0);
- free_signaling_queue_entry(s);
- }
- pvt->hold_signaling = 0;
- }
- /*! \brief All frames other than that of type AST_FRAME_IAX must be held until
- * we have received a destination call number. */
- static int queue_signalling(struct chan_iax2_pvt *pvt, struct ast_frame *f)
- {
- struct signaling_queue_entry *qe;
- if (f->frametype == AST_FRAME_IAX || !pvt->hold_signaling) {
- return 1; /* do not queue this frame */
- } else if (!(qe = ast_calloc(1, sizeof(struct signaling_queue_entry)))) {
- return -1; /* out of memory */
- }
- /* copy ast_frame into our queue entry */
- qe->f = *f;
- if (qe->f.datalen) {
- /* if there is data in this frame copy it over as well */
- if (!(qe->f.data.ptr = ast_malloc(qe->f.datalen))) {
- free_signaling_queue_entry(qe);
- return -1;
- }
- memcpy(qe->f.data.ptr, f->data.ptr, qe->f.datalen);
- }
- AST_LIST_INSERT_TAIL(&pvt->signaling_queue, qe, next);
- return 0;
- }
- static void pvt_destructor(void *obj)
- {
- struct chan_iax2_pvt *pvt = obj;
- struct iax_frame *cur = NULL;
- struct signaling_queue_entry *s = NULL;
- ast_mutex_lock(&iaxsl[pvt->callno]);
- iax2_destroy_helper(pvt);
- sched_delay_remove(&pvt->addr, pvt->callno_entry);
- pvt->callno_entry = NULL;
- /* Already gone */
- ast_set_flag64(pvt, IAX_ALREADYGONE);
- AST_LIST_TRAVERSE(&frame_queue[pvt->callno], cur, list) {
- /* Cancel any pending transmissions */
- cur->retries = -1;
- }
- ast_mutex_unlock(&iaxsl[pvt->callno]);
- while ((s = AST_LIST_REMOVE_HEAD(&pvt->signaling_queue, next))) {
- free_signaling_queue_entry(s);
- }
- if (pvt->reg) {
- pvt->reg->callno = 0;
- }
- if (!pvt->owner) {
- jb_frame frame;
- if (pvt->vars) {
- ast_variables_destroy(pvt->vars);
- pvt->vars = NULL;
- }
- while (jb_getall(pvt->jb, &frame) == JB_OK) {
- iax2_frame_free(frame.data);
- }
- jb_destroy(pvt->jb);
- ast_string_field_free_memory(pvt);
- }
- if (pvt->callid) {
- ast_callid_unref(pvt->callid);
- }
- }
- static struct chan_iax2_pvt *new_iax(struct sockaddr_in *sin, const char *host)
- {
- struct chan_iax2_pvt *tmp;
- jb_conf jbconf;
- if (!(tmp = ao2_alloc(sizeof(*tmp), pvt_destructor))) {
- return NULL;
- }
- if (ast_string_field_init(tmp, 32)) {
- ao2_ref(tmp, -1);
- tmp = NULL;
- return NULL;
- }
-
- tmp->prefs = prefs;
- tmp->pingid = -1;
- tmp->lagid = -1;
- tmp->autoid = -1;
- tmp->authid = -1;
- tmp->initid = -1;
- tmp->keyrotateid = -1;
- ast_string_field_set(tmp,exten, "s");
- ast_string_field_set(tmp,host, host);
- tmp->jb = jb_new();
- tmp->jbid = -1;
- jbconf.max_jitterbuf = maxjitterbuffer;
- jbconf.resync_threshold = resyncthreshold;
- jbconf.max_contig_interp = maxjitterinterps;
- jbconf.target_extra = jittertargetextra;
- jb_setconf(tmp->jb,&jbconf);
- AST_LIST_HEAD_INIT_NOLOCK(&tmp->dpentries);
- tmp->hold_signaling = 1;
- AST_LIST_HEAD_INIT_NOLOCK(&tmp->signaling_queue);
- return tmp;
- }
- static struct iax_frame *iaxfrdup2(struct iax_frame *fr)
- {
- struct iax_frame *new = iax_frame_new(DIRECTION_INGRESS, fr->af.datalen, fr->cacheable);
- if (new) {
- size_t afdatalen = new->afdatalen;
- memcpy(new, fr, sizeof(*new));
- iax_frame_wrap(new, &fr->af);
- new->afdatalen = afdatalen;
- new->data = NULL;
- new->datalen = 0;
- new->direction = DIRECTION_INGRESS;
- new->retrans = -1;
- }
- return new;
- }
- /* keep these defined in this order. They are used in find_callno to
- * determine whether or not a new call number should be allowed. */
- enum {
- /* do not allow a new call number, only search ones in use for match */
- NEW_PREVENT = 0,
- /* search for match first, then allow a new one to be allocated */
- NEW_ALLOW = 1,
- /* do not search for match, force a new call number */
- NEW_FORCE = 2,
- /* do not search for match, force a new call number. Signifies call number
- * has been calltoken validated */
- NEW_ALLOW_CALLTOKEN_VALIDATED = 3,
- };
- static int match(struct sockaddr_in *sin, unsigned short callno, unsigned short dcallno, const struct chan_iax2_pvt *cur, int check_dcallno)
- {
- if ((cur->addr.sin_addr.s_addr == sin->sin_addr.s_addr) &&
- (cur->addr.sin_port == sin->sin_port)) {
- /* This is the main host */
- if ( (cur->peercallno == 0 || cur->peercallno == callno) &&
- (check_dcallno ? dcallno == cur->callno : 1) ) {
- /* That's us. Be sure we keep track of the peer call number */
- return 1;
- }
- }
- if ((cur->transfer.sin_addr.s_addr == sin->sin_addr.s_addr) &&
- (cur->transfer.sin_port == sin->sin_port) && (cur->transferring)) {
- /* We're transferring */
- if ((dcallno == cur->callno) || (cur->transferring == TRANSFER_MEDIAPASS && cur->transfercallno == callno))
- return 1;
- }
- return 0;
- }
- static int make_trunk(unsigned short callno, int locked)
- {
- int x;
- int res= 0;
- struct callno_entry *callno_entry;
- if (iaxs[callno]->oseqno) {
- ast_log(LOG_WARNING, "Can't make trunk once a call has started!\n");
- return -1;
- }
- if (callno >= TRUNK_CALL_START) {
- ast_log(LOG_WARNING, "Call %d is already a trunk\n", callno);
- return -1;
- }
- if (!(callno_entry = get_unused_callno(1, iaxs[callno]->callno_entry->validated))) {
- ast_log(LOG_WARNING, "Unable to trunk call: Insufficient space\n");
- return -1;
- }
- x = callno_entry->callno;
- ast_mutex_lock(&iaxsl[x]);
- /*!
- * \note We delete these before switching the slot, because if
- * they fire in the meantime, they will generate a warning.
- */
- AST_SCHED_DEL(sched, iaxs[callno]->pingid);
- AST_SCHED_DEL(sched, iaxs[callno]->lagid);
- iaxs[callno]->lagid = iaxs[callno]->pingid = -1;
- iaxs[x] = iaxs[callno];
- iaxs[x]->callno = x;
- /* since we copied over the pvt from a different callno, make sure the old entry is replaced
- * before assigning the new one */
- if (iaxs[x]->callno_entry) {
- iax2_sched_add(sched, MIN_REUSE_TIME * 1000, replace_callno, iaxs[x]->callno_entry);
- }
- iaxs[x]->callno_entry = callno_entry;
- iaxs[callno] = NULL;
- /* Update the two timers that should have been started */
- iaxs[x]->pingid = iax2_sched_add(sched,
- ping_time * 1000, send_ping, (void *)(long)x);
- iaxs[x]->lagid = iax2_sched_add(sched,
- lagrq_time * 1000, send_lagrq, (void *)(long)x);
- if (locked)
- ast_mutex_unlock(&iaxsl[callno]);
- res = x;
- if (!locked)
- ast_mutex_unlock(&iaxsl[x]);
- /* We moved this call from a non-trunked to a trunked call */
- ast_debug(1, "Made call %d into trunk call %d\n", callno, x);
- return res;
- }
- static void store_by_transfercallno(struct chan_iax2_pvt *pvt)
- {
- if (!pvt->transfercallno) {
- ast_log(LOG_ERROR, "This should not be called without a transfer call number.\n");
- return;
- }
- ao2_link(iax_transfercallno_pvts, pvt);
- }
- static void remove_by_transfercallno(struct chan_iax2_pvt *pvt)
- {
- if (!pvt->transfercallno) {
- ast_log(LOG_ERROR, "This should not be called without a transfer call number.\n");
- return;
- }
- ao2_unlink(iax_transfercallno_pvts, pvt);
- }
- static void store_by_peercallno(struct chan_iax2_pvt *pvt)
- {
- if (!pvt->peercallno) {
- ast_log(LOG_ERROR, "This should not be called without a peer call number.\n");
- return;
- }
- ao2_link(iax_peercallno_pvts, pvt);
- }
- static void remove_by_peercallno(struct chan_iax2_pvt *pvt)
- {
- if (!pvt->peercallno) {
- ast_log(LOG_ERROR, "This should not be called without a peer call number.\n");
- return;
- }
- ao2_unlink(iax_peercallno_pvts, pvt);
- }
- static int addr_range_delme_cb(void *obj, void *arg, int flags)
- {
- struct addr_range *lim = obj;
- lim->delme = 1;
- return 0;
- }
- static int addr_range_hash_cb(const void *obj, const int flags)
- {
- const struct addr_range *lim = obj;
- struct sockaddr_in sin;
- ast_sockaddr_to_sin(&lim->ha.addr, &sin);
- return abs((int) sin.sin_addr.s_addr);
- }
- static int addr_range_cmp_cb(void *obj, void *arg, int flags)
- {
- struct addr_range *lim1 = obj, *lim2 = arg;
- return (!(ast_sockaddr_cmp_addr(&lim1->ha.addr, &lim2->ha.addr)) &&
- !(ast_sockaddr_cmp_addr(&lim1->ha.netmask, &lim2->ha.netmask))) ?
- CMP_MATCH | CMP_STOP : 0;
- }
- static int peercnt_hash_cb(const void *obj, const int flags)
- {
- const struct peercnt *peercnt = obj;
- return abs((int) peercnt->addr);
- }
- static int peercnt_cmp_cb(void *obj, void *arg, int flags)
- {
- struct peercnt *peercnt1 = obj, *peercnt2 = arg;
- return (peercnt1->addr == peercnt2->addr) ? CMP_MATCH | CMP_STOP : 0;
- }
- static int addr_range_match_address_cb(void *obj, void *arg, int flags)
- {
- struct addr_range *addr_range = obj;
- struct sockaddr_in *sin = arg;
- struct sockaddr_in ha_netmask_sin;
- struct sockaddr_in ha_addr_sin;
- ast_sockaddr_to_sin(&addr_range->ha.netmask, &ha_netmask_sin);
- ast_sockaddr_to_sin(&addr_range->ha.addr, &ha_addr_sin);
- if ((sin->sin_addr.s_addr & ha_netmask_sin.sin_addr.s_addr) == ha_addr_sin.sin_addr.s_addr) {
- return CMP_MATCH | CMP_STOP;
- }
- return 0;
- }
- /*!
- * \internal
- *
- * \brief compares sin to calltoken_ignores table to determine if validation is required.
- */
- static int calltoken_required(struct sockaddr_in *sin, const char *name, int subclass)
- {
- struct addr_range *addr_range;
- struct iax2_peer *peer = NULL;
- struct iax2_user *user = NULL;
- /* if no username is given, check for guest accounts */
- const char *find = S_OR(name, "guest");
- int res = 1; /* required by default */
- int optional = 0;
- enum calltoken_peer_enum calltoken_required = CALLTOKEN_DEFAULT;
- /* There are only two cases in which calltoken validation is not required.
- * Case 1. sin falls within the list of address ranges specified in the calltoken optional table and
- * the peer definition has not set the requirecalltoken option.
- * Case 2. Username is a valid peer/user, and that peer has requirecalltoken set either auto or no.
- */
- /* ----- Case 1 ----- */
- if ((addr_range = ao2_callback(calltoken_ignores, 0, addr_range_match_address_cb, sin))) {
- ao2_ref(addr_range, -1);
- optional = 1;
- }
- /* ----- Case 2 ----- */
- if ((subclass == IAX_COMMAND_NEW) && (user = find_user(find))) {
- calltoken_required = user->calltoken_required;
- } else if ((subclass == IAX_COMMAND_NEW) && (user = realtime_user(find, sin))) {
- calltoken_required = user->calltoken_required;
- } else if ((subclass != IAX_COMMAND_NEW) && (peer = find_peer(find, 0))) {
- calltoken_required = peer->calltoken_required;
- } else if ((subclass != IAX_COMMAND_NEW) && (peer = realtime_peer(find, sin))) {
- calltoken_required = peer->calltoken_required;
- }
- if (peer) {
- peer_unref(peer);
- }
- if (user) {
- user_unref(user);
- }
- ast_debug(1, "Determining if address %s with username %s requires calltoken validation. Optional = %d calltoken_required = %u\n", ast_inet_ntoa(sin->sin_addr), name, optional, calltoken_required);
- if (((calltoken_required == CALLTOKEN_NO) || (calltoken_required == CALLTOKEN_AUTO)) ||
- (optional && (calltoken_required == CALLTOKEN_DEFAULT))) {
- res = 0;
- }
- return res;
- }
- /*!
- * \internal
- *
- * \brief set peercnt callno limit.
- *
- * \details
- * First looks in custom definitions. If not found, global limit
- * is used. Entries marked as reg already have
- * a custom limit set by a registration and are not modified.
- */
- static void set_peercnt_limit(struct peercnt *peercnt)
- {
- uint16_t limit = global_maxcallno;
- struct addr_range *addr_range;
- struct sockaddr_in sin = {
- .sin_addr.s_addr = peercnt->addr,
- };
- if (peercnt->reg && peercnt->limit) {
- return; /* this peercnt has a custom limit set by a registration */
- }
- if ((addr_range = ao2_callback(callno_limits, 0, addr_range_match_address_cb, &sin))) {
- limit = addr_range->limit;
- ast_debug(1, "custom addr_range %d found for %s\n", limit, ast_inet_ntoa(sin.sin_addr));
- ao2_ref(addr_range, -1);
- }
- peercnt->limit = limit;
- }
- /*!
- * \internal
- * \brief sets limits for all peercnts in table. done on reload to reflect changes in conf.
- */
- static int set_peercnt_limit_all_cb(void *obj, void *arg, int flags)
- {
- struct peercnt *peercnt = obj;
- set_peercnt_limit(peercnt);
- ast_debug(1, "Reset limits for peercnts table\n");
- return 0;
- }
- /*!
- * \internal
- * \brief returns match if delme is set.
- */
- static int prune_addr_range_cb(void *obj, void *arg, int flags)
- {
- struct addr_range *addr_range = obj;
- return addr_range->delme ? CMP_MATCH : 0;
- }
- /*!
- * \internal
- * \brief modifies peercnt entry in peercnts table. Used to set custom limit or mark a registered ip
- */
- static void peercnt_modify(unsigned char reg, uint16_t limit, struct ast_sockaddr *sockaddr)
- {
- /* this function turns off and on custom callno limits set by peer registration */
- struct peercnt *peercnt;
- struct peercnt tmp = {
- .addr = 0,
- };
- struct sockaddr_in sin;
- ast_sockaddr_to_sin(sockaddr, &sin);
- tmp.addr = sin.sin_addr.s_addr;
- if ((peercnt = ao2_find(peercnts, &tmp, OBJ_POINTER))) {
- peercnt->reg = reg;
- if (limit) {
- peercnt->limit = limit;
- } else {
- set_peercnt_limit(peercnt);
- }
- ast_debug(1, "peercnt entry %s modified limit:%d registered:%d", ast_inet_ntoa(sin.sin_addr), peercnt->limit, peercnt->reg);
- ao2_ref(peercnt, -1); /* decrement ref from find */
- }
- }
- /*!
- * \internal
- * \brief adds an ip to the peercnts table, increments connection count if it already exists
- *
- * \details First searches for the address in the peercnts table. If found
- * the current count is incremented. If not found a new peercnt is allocated
- * and linked into the peercnts table with a call number count of 1.
- */
- static int peercnt_add(struct sockaddr_in *sin)
- {
- struct peercnt *peercnt;
- unsigned long addr = sin->sin_addr.s_addr;
- int res = 0;
- struct peercnt tmp = {
- .addr = addr,
- };
- /* Reasoning for peercnts container lock: Two identical ip addresses
- * could be added by different threads at the "same time". Without the container
- * lock, both threads could alloc space for the same object and attempt
- * to link to table. With the lock, one would create the object and link
- * to table while the other would find the already created peercnt object
- * rather than creating a new one. */
- ao2_lock(peercnts);
- if ((peercnt = ao2_find(peercnts, &tmp, OBJ_POINTER))) {
- ao2_lock(peercnt);
- } else if ((peercnt = ao2_alloc(sizeof(*peercnt), NULL))) {
- ao2_lock(peercnt);
- /* create and set defaults */
- peercnt->addr = addr;
- set_peercnt_limit(peercnt);
- /* guarantees it does not go away after unlocking table
- * ao2_find automatically adds this */
- ao2_link(peercnts, peercnt);
- } else {
- ao2_unlock(peercnts);
- return -1;
- }
- /* check to see if the address has hit its callno limit. If not increment cur. */
- if (peercnt->limit > peercnt->cur) {
- peercnt->cur++;
- ast_debug(1, "ip callno count incremented to %d for %s\n", peercnt->cur, ast_inet_ntoa(sin->sin_addr));
- } else { /* max num call numbers for this peer has been reached! */
- ast_log(LOG_ERROR, "maxcallnumber limit of %d for %s has been reached!\n", peercnt->limit, ast_inet_ntoa(sin->sin_addr));
- res = -1;
- }
- /* clean up locks and ref count */
- ao2_unlock(peercnt);
- ao2_unlock(peercnts);
- ao2_ref(peercnt, -1); /* decrement ref from find/alloc, only the container ref remains. */
- return res;
- }
- /*!
- * \internal
- * \brief decrements a peercnts table entry
- */
- static void peercnt_remove(struct peercnt *peercnt)
- {
- struct sockaddr_in sin = {
- .sin_addr.s_addr = peercnt->addr,
- };
- /*
- * Container locked here since peercnt may be unlinked from
- * list. If left unlocked, peercnt_add could try and grab this
- * entry from the table and modify it at the "same time" this
- * thread attemps to unlink it.
- */
- ao2_lock(peercnts);
- peercnt->cur--;
- ast_debug(1, "ip callno count decremented to %d for %s\n", peercnt->cur, ast_inet_ntoa(sin.sin_addr));
- /* if this was the last connection from the peer remove it from table */
- if (peercnt->cur == 0) {
- ao2_unlink(peercnts, peercnt);/* decrements ref from table, last ref is left to scheduler */
- }
- ao2_unlock(peercnts);
- }
- /*!
- * \internal
- * \brief called by scheduler to decrement object
- */
- static int peercnt_remove_cb(const void *obj)
- {
- struct peercnt *peercnt = (struct peercnt *) obj;
- peercnt_remove(peercnt);
- ao2_ref(peercnt, -1); /* decrement ref from scheduler */
- return 0;
- }
- /*!
- * \internal
- * \brief decrements peercnts connection count, finds by addr
- */
- static int peercnt_remove_by_addr(struct sockaddr_in *sin)
- {
- struct peercnt *peercnt;
- struct peercnt tmp = {
- .addr = sin->sin_addr.s_addr,
- };
- if ((peercnt = ao2_find(peercnts, &tmp, OBJ_POINTER))) {
- peercnt_remove(peercnt);
- ao2_ref(peercnt, -1); /* decrement ref from find */
- }
- return 0;
- }
- /*!
- * \internal
- * \brief Create callno_limit entry based on configuration
- */
- static void build_callno_limits(struct ast_variable *v)
- {
- struct addr_range *addr_range = NULL;
- struct addr_range tmp;
- struct ast_ha *ha;
- int limit;
- int error;
- int found;
- for (; v; v = v->next) {
- limit = -1;
- error = 0;
- found = 0;
- ha = ast_append_ha("permit", v->name, NULL, &error);
- /* check for valid config information */
- if (error) {
- ast_log(LOG_ERROR, "Call number limit for %s could not be added, Invalid address range\n.", v->name);
- continue;
- } else if ((sscanf(v->value, "%d", &limit) != 1) || (limit < 0)) {
- ast_log(LOG_ERROR, "Call number limit for %s could not be added. Invalid limit %s\n.", v->name, v->value);
- ast_free_ha(ha);
- continue;
- }
- ast_copy_ha(ha, &tmp.ha);
- /* find or create the addr_range */
- if ((addr_range = ao2_find(callno_limits, &tmp, OBJ_POINTER))) {
- ao2_lock(addr_range);
- found = 1;
- } else if (!(addr_range = ao2_alloc(sizeof(*addr_range), NULL))) {
- ast_free_ha(ha);
- return; /* out of memory */
- }
- /* copy over config data into addr_range object */
- ast_copy_ha(ha, &addr_range->ha); /* this is safe because only one ha is possible for each limit */
- ast_free_ha(ha); /* cleanup the tmp ha */
- addr_range->limit = limit;
- addr_range->delme = 0;
- /* cleanup */
- if (found) {
- ao2_unlock(addr_range);
- } else {
- ao2_link(callno_limits, addr_range);
- }
- ao2_ref(addr_range, -1); /* decrement ref from ao2_find and ao2_alloc, only container ref remains */
- }
- }
- /*!
- * \internal
- * \brief Create calltoken_ignores entry based on configuration
- */
- static int add_calltoken_ignore(const char *addr)
- {
- struct addr_range tmp;
- struct addr_range *addr_range = NULL;
- struct ast_ha *ha = NULL;
- int error = 0;
- if (ast_strlen_zero(addr)) {
- ast_log(LOG_WARNING, "invalid calltokenoptional %s\n", addr);
- return -1;
- }
- ha = ast_append_ha("permit", addr, NULL, &error);
- /* check for valid config information */
- if (error) {
- ast_log(LOG_WARNING, "Error %d creating calltokenoptional entry %s\n", error, addr);
- return -1;
- }
- ast_copy_ha(ha, &tmp.ha);
- /* find or create the addr_range */
- if ((addr_range = ao2_find(calltoken_ignores, &tmp, OBJ_POINTER))) {
- ao2_lock(addr_range);
- addr_range->delme = 0;
- ao2_unlock(addr_range);
- } else if ((addr_range = ao2_alloc(sizeof(*addr_range), NULL))) {
- /* copy over config data into addr_range object */
- ast_copy_ha(ha, &addr_range->ha); /* this is safe because only one ha is possible */
- ao2_link(calltoken_ignores, addr_range);
- } else {
- ast_free_ha(ha);
- return -1;
- }
- ast_free_ha(ha);
- ao2_ref(addr_range, -1); /* decrement ref from ao2_find and ao2_alloc, only container ref remains */
- return 0;
- }
- static char *handle_cli_iax2_show_callno_limits(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct ao2_iterator i;
- struct peercnt *peercnt;
- struct sockaddr_in sin;
- int found = 0;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show callnumber usage";
- e->usage =
- "Usage: iax2 show callnumber usage [IP address]\n"
- " Shows current IP addresses which are consuming iax2 call numbers\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- case CLI_HANDLER:
- if (a->argc < 4 || a->argc > 5)
- return CLI_SHOWUSAGE;
- if (a->argc == 4) {
- ast_cli(a->fd, "%-15s %-12s %-12s\n", "Address", "Callno Usage", "Callno Limit");
- }
- i = ao2_iterator_init(peercnts, 0);
- while ((peercnt = ao2_iterator_next(&i))) {
- sin.sin_addr.s_addr = peercnt->addr;
- if (a->argc == 5) {
- if (!strcasecmp(a->argv[4], ast_inet_ntoa(sin.sin_addr))) {
- ast_cli(a->fd, "%-15s %-12s %-12s\n", "Address", "Callno Usage", "Callno Limit");
- ast_cli(a->fd, "%-15s %-12d %-12d\n", ast_inet_ntoa(sin.sin_addr), peercnt->cur, peercnt->limit);
- ao2_ref(peercnt, -1);
- found = 1;
- break;
- }
- } else {
- ast_cli(a->fd, "%-15s %-12d %-12d\n", ast_inet_ntoa(sin.sin_addr), peercnt->cur, peercnt->limit);
- }
- ao2_ref(peercnt, -1);
- }
- ao2_iterator_destroy(&i);
- if (a->argc == 4) {
- ast_cli(a->fd, "\nNon-CallToken Validation Callno Limit: %d\n"
- "Non-CallToken Validated Callno Used: %d\n",
- global_maxcallno_nonval,
- total_nonval_callno_used);
- ast_cli(a->fd, "Total Available Callno: %d\n"
- "Regular Callno Available: %d\n"
- "Trunk Callno Available: %d\n",
- ao2_container_count(callno_pool) + ao2_container_count(callno_pool_trunk),
- ao2_container_count(callno_pool),
- ao2_container_count(callno_pool_trunk));
- } else if (a->argc == 5 && !found) {
- ast_cli(a->fd, "No call number table entries for %s found\n", a->argv[4] );
- }
- return CLI_SUCCESS;
- default:
- return NULL;
- }
- }
- static struct callno_entry *get_unused_callno(int trunk, int validated)
- {
- struct callno_entry *callno_entry = NULL;
- if ((!ao2_container_count(callno_pool) && !trunk) || (!ao2_container_count(callno_pool_trunk) && trunk)) {
- ast_log(LOG_WARNING, "Out of CallNumbers\n");
- /* Minor optimization for the extreme case. */
- return NULL;
- }
- /* the callno_pool container is locked here primarily to ensure thread
- * safety of the total_nonval_callno_used check and increment */
- ao2_lock(callno_pool);
- /* only a certain number of nonvalidated call numbers should be allocated.
- * If there ever is an attack, this separates the calltoken validating
- * users from the non calltoken validating users. */
- if (!validated && (total_nonval_callno_used >= global_maxcallno_nonval)) {
- ast_log(LOG_WARNING, "NON-CallToken callnumber limit is reached. Current:%d Max:%d\n", total_nonval_callno_used, global_maxcallno_nonval);
- ao2_unlock(callno_pool);
- return NULL;
- }
- /* unlink the object from the container, taking over ownership
- * of the reference the container had to the object */
- callno_entry = ao2_find((trunk ? callno_pool_trunk : callno_pool), NULL, OBJ_POINTER | OBJ_UNLINK | OBJ_CONTINUE);
- if (callno_entry) {
- callno_entry->validated = validated;
- if (!validated) {
- total_nonval_callno_used++;
- }
- }
- ao2_unlock(callno_pool);
- return callno_entry;
- }
- static int replace_callno(const void *obj)
- {
- struct callno_entry *callno_entry = (struct callno_entry *) obj;
- /* the callno_pool container is locked here primarily to ensure thread
- * safety of the total_nonval_callno_used check and decrement */
- ao2_lock(callno_pool);
- if (!callno_entry->validated && (total_nonval_callno_used != 0)) {
- total_nonval_callno_used--;
- } else if (!callno_entry->validated && (total_nonval_callno_used == 0)) {
- ast_log(LOG_ERROR, "Attempted to decrement total non calltoken validated callnumbers below zero... Callno is:%d \n", callno_entry->callno);
- }
- if (callno_entry->callno < TRUNK_CALL_START) {
- ao2_link(callno_pool, callno_entry);
- } else {
- ao2_link(callno_pool_trunk, callno_entry);
- }
- ao2_ref(callno_entry, -1); /* only container ref remains */
- ao2_unlock(callno_pool);
- return 0;
- }
- static int callno_hash(const void *obj, const int flags)
- {
- return abs(ast_random());
- }
- static int create_callno_pools(void)
- {
- uint16_t i;
- if (!(callno_pool = ao2_container_alloc(CALLNO_POOL_BUCKETS, callno_hash, NULL))) {
- return -1;
- }
- if (!(callno_pool_trunk = ao2_container_alloc(CALLNO_POOL_BUCKETS, callno_hash, NULL))) {
- return -1;
- }
- /* start at 2, 0 and 1 are reserved */
- for (i = 2; i < IAX_MAX_CALLS; i++) {
- struct callno_entry *callno_entry;
- if (!(callno_entry = ao2_alloc(sizeof(*callno_entry), NULL))) {
- return -1;
- }
- callno_entry->callno = i;
- if (i < TRUNK_CALL_START) {
- ao2_link(callno_pool, callno_entry);
- } else {
- ao2_link(callno_pool_trunk, callno_entry);
- }
- ao2_ref(callno_entry, -1);
- }
- return 0;
- }
- /*!
- * \internal
- * \brief Schedules delayed removal of iax2_pvt call number data
- *
- * \note After MIN_REUSE_TIME has passed for a destroyed iax2_pvt, the callno is
- * available again, and the address from the previous connection must be decremented
- * from the peercnts table. This function schedules these operations to take place.
- */
- static void sched_delay_remove(struct sockaddr_in *sin, struct callno_entry *callno_entry)
- {
- int i;
- struct peercnt *peercnt;
- struct peercnt tmp = {
- .addr = sin->sin_addr.s_addr,
- };
- if ((peercnt = ao2_find(peercnts, &tmp, OBJ_POINTER))) {
- /* refcount is incremented with ao2_find. keep that ref for the scheduler */
- ast_debug(1, "schedule decrement of callno used for %s in %d seconds\n", ast_inet_ntoa(sin->sin_addr), MIN_REUSE_TIME);
- i = iax2_sched_add(sched, MIN_REUSE_TIME * 1000, peercnt_remove_cb, peercnt);
- if (i == -1) {
- ao2_ref(peercnt, -1);
- }
- }
- iax2_sched_add(sched, MIN_REUSE_TIME * 1000, replace_callno, callno_entry);
- }
- /*!
- * \internal
- * \brief returns whether or not a frame is capable of starting a new IAX2 dialog.
- *
- * \note For this implementation, inbound pokes should _NOT_ be capable of allocating
- * a new callno.
- */
- static inline int attribute_pure iax2_allow_new(int frametype, int subclass, int inbound)
- {
- if (frametype != AST_FRAME_IAX) {
- return 0;
- }
- switch (subclass) {
- case IAX_COMMAND_NEW:
- case IAX_COMMAND_REGREQ:
- case IAX_COMMAND_FWDOWNL:
- case IAX_COMMAND_REGREL:
- return 1;
- case IAX_COMMAND_POKE:
- if (!inbound) {
- return 1;
- }
- break;
- }
- return 0;
- }
- /*
- * \note Calling this function while holding another pvt lock can cause a deadlock.
- */
- static int __find_callno(unsigned short callno, unsigned short dcallno, struct sockaddr_in *sin, int new, int sockfd, int return_locked, int check_dcallno)
- {
- int res = 0;
- int x;
- /* this call is calltoken validated as long as it is either NEW_FORCE
- * or NEW_ALLOW_CALLTOKEN_VALIDATED */
- int validated = (new > NEW_ALLOW) ? 1 : 0;
- char host[80];
- if (new <= NEW_ALLOW) {
- if (callno) {
- struct chan_iax2_pvt *pvt;
- struct chan_iax2_pvt tmp_pvt = {
- .callno = dcallno,
- .peercallno = callno,
- .transfercallno = callno,
- /* hack!! */
- .frames_received = check_dcallno,
- };
- memcpy(&tmp_pvt.addr, sin, sizeof(tmp_pvt.addr));
- /* this works for finding normal call numbers not involving transfering */
- if ((pvt = ao2_find(iax_peercallno_pvts, &tmp_pvt, OBJ_POINTER))) {
- if (return_locked) {
- ast_mutex_lock(&iaxsl[pvt->callno]);
- }
- res = pvt->callno;
- ao2_ref(pvt, -1);
- pvt = NULL;
- return res;
- }
- /* this searches for transfer call numbers that might not get caught otherwise */
- memset(&tmp_pvt.addr, 0, sizeof(tmp_pvt.addr));
- memcpy(&tmp_pvt.transfer, sin, sizeof(tmp_pvt.transfer));
- if ((pvt = ao2_find(iax_transfercallno_pvts, &tmp_pvt, OBJ_POINTER))) {
- if (return_locked) {
- ast_mutex_lock(&iaxsl[pvt->callno]);
- }
- res = pvt->callno;
- ao2_ref(pvt, -1);
- pvt = NULL;
- return res;
- }
- }
- /* This will occur on the first response to a message that we initiated,
- * such as a PING. */
- if (dcallno) {
- ast_mutex_lock(&iaxsl[dcallno]);
- }
- if (callno && dcallno && iaxs[dcallno] && !iaxs[dcallno]->peercallno && match(sin, callno, dcallno, iaxs[dcallno], check_dcallno)) {
- iaxs[dcallno]->peercallno = callno;
- res = dcallno;
- store_by_peercallno(iaxs[dcallno]);
- if (!res || !return_locked) {
- ast_mutex_unlock(&iaxsl[dcallno]);
- }
- return res;
- }
- if (dcallno) {
- ast_mutex_unlock(&iaxsl[dcallno]);
- }
- }
- if (!res && (new >= NEW_ALLOW)) {
- struct callno_entry *callno_entry;
- /* It may seem odd that we look through the peer list for a name for
- * this *incoming* call. Well, it is weird. However, users don't
- * have an IP address/port number that we can match against. So,
- * this is just checking for a peer that has that IP/port and
- * assuming that we have a user of the same name. This isn't always
- * correct, but it will be changed if needed after authentication. */
- if (!iax2_getpeername(*sin, host, sizeof(host)))
- snprintf(host, sizeof(host), "%s:%d", ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
- if (peercnt_add(sin)) {
- /* This address has hit its callnumber limit. When the limit
- * is reached, the connection is not added to the peercnts table.*/
- return 0;
- }
- if (!(callno_entry = get_unused_callno(0, validated))) {
- /* since we ran out of space, remove the peercnt
- * entry we added earlier */
- peercnt_remove_by_addr(sin);
- ast_log(LOG_WARNING, "No more space\n");
- return 0;
- }
- x = callno_entry->callno;
- ast_mutex_lock(&iaxsl[x]);
- iaxs[x] = new_iax(sin, host);
- if (iaxs[x]) {
- if (iaxdebug)
- ast_debug(1, "Creating new call structure %d\n", x);
- iaxs[x]->callno_entry = callno_entry;
- iaxs[x]->sockfd = sockfd;
- iaxs[x]->addr.sin_port = sin->sin_port;
- iaxs[x]->addr.sin_family = sin->sin_family;
- iaxs[x]->addr.sin_addr.s_addr = sin->sin_addr.s_addr;
- iaxs[x]->peercallno = callno;
- iaxs[x]->callno = x;
- iaxs[x]->pingtime = DEFAULT_RETRY_TIME;
- iaxs[x]->expiry = min_reg_expire;
- iaxs[x]->pingid = iax2_sched_add(sched, ping_time * 1000, send_ping, (void *)(long)x);
- iaxs[x]->lagid = iax2_sched_add(sched, lagrq_time * 1000, send_lagrq, (void *)(long)x);
- iaxs[x]->amaflags = amaflags;
- ast_copy_flags64(iaxs[x], &globalflags, IAX_NOTRANSFER | IAX_TRANSFERMEDIA | IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE | IAX_FORCE_ENCRYPT);
- ast_string_field_set(iaxs[x], accountcode, accountcode);
- ast_string_field_set(iaxs[x], mohinterpret, mohinterpret);
- ast_string_field_set(iaxs[x], mohsuggest, mohsuggest);
- ast_string_field_set(iaxs[x], parkinglot, default_parkinglot);
- if (iaxs[x]->peercallno) {
- store_by_peercallno(iaxs[x]);
- }
- } else {
- ast_log(LOG_WARNING, "Out of resources\n");
- ast_mutex_unlock(&iaxsl[x]);
- replace_callno(callno_entry);
- return 0;
- }
- if (!return_locked)
- ast_mutex_unlock(&iaxsl[x]);
- res = x;
- }
- return res;
- }
- static int find_callno(unsigned short callno, unsigned short dcallno, struct sockaddr_in *sin, int new, int sockfd, int full_frame) {
- return __find_callno(callno, dcallno, sin, new, sockfd, 0, full_frame);
- }
- static int find_callno_locked(unsigned short callno, unsigned short dcallno, struct sockaddr_in *sin, int new, int sockfd, int full_frame) {
- return __find_callno(callno, dcallno, sin, new, sockfd, 1, full_frame);
- }
- /*!
- * \brief Queue a frame to a call's owning asterisk channel
- *
- * \pre This function assumes that iaxsl[callno] is locked when called.
- *
- * \note IMPORTANT NOTE!!! Any time this function is used, even if iaxs[callno]
- * was valid before calling it, it may no longer be valid after calling it.
- * This function may unlock and lock the mutex associated with this callno,
- * meaning that another thread may grab it and destroy the call.
- */
- static int iax2_queue_frame(int callno, struct ast_frame *f)
- {
- iax2_lock_owner(callno);
- if (iaxs[callno] && iaxs[callno]->owner) {
- ast_queue_frame(iaxs[callno]->owner, f);
- ast_channel_unlock(iaxs[callno]->owner);
- }
- return 0;
- }
- /*!
- * \brief Queue a hangup frame on the ast_channel owner
- *
- * This function queues a hangup frame on the owner of the IAX2 pvt struct that
- * is active for the given call number.
- *
- * \pre Assumes lock for callno is already held.
- *
- * \note IMPORTANT NOTE!!! Any time this function is used, even if iaxs[callno]
- * was valid before calling it, it may no longer be valid after calling it.
- * This function may unlock and lock the mutex associated with this callno,
- * meaning that another thread may grab it and destroy the call.
- */
- static int iax2_queue_hangup(int callno)
- {
- iax2_lock_owner(callno);
- if (iaxs[callno] && iaxs[callno]->owner) {
- ast_queue_hangup(iaxs[callno]->owner);
- ast_channel_unlock(iaxs[callno]->owner);
- }
- return 0;
- }
- /*!
- * \brief Queue a control frame on the ast_channel owner
- *
- * This function queues a control frame on the owner of the IAX2 pvt struct that
- * is active for the given call number.
- *
- * \pre Assumes lock for callno is already held.
- *
- * \note IMPORTANT NOTE!!! Any time this function is used, even if iaxs[callno]
- * was valid before calling it, it may no longer be valid after calling it.
- * This function may unlock and lock the mutex associated with this callno,
- * meaning that another thread may grab it and destroy the call.
- */
- static int iax2_queue_control_data(int callno,
- enum ast_control_frame_type control, const void *data, size_t datalen)
- {
- iax2_lock_owner(callno);
- if (iaxs[callno] && iaxs[callno]->owner) {
- ast_queue_control_data(iaxs[callno]->owner, control, data, datalen);
- ast_channel_unlock(iaxs[callno]->owner);
- }
- return 0;
- }
- static void destroy_firmware(struct iax_firmware *cur)
- {
- /* Close firmware */
- if (cur->fwh) {
- munmap((void*)cur->fwh, ntohl(cur->fwh->datalen) + sizeof(*(cur->fwh)));
- }
- close(cur->fd);
- ast_free(cur);
- }
- static int try_firmware(char *s)
- {
- struct stat stbuf;
- struct iax_firmware *cur = NULL;
- int ifd, fd, res, len, chunk;
- struct ast_iax2_firmware_header *fwh, fwh2;
- struct MD5Context md5;
- unsigned char sum[16], buf[1024];
- char *s2, *last;
- s2 = ast_alloca(strlen(s) + 100);
- last = strrchr(s, '/');
- if (last)
- last++;
- else
- last = s;
- snprintf(s2, strlen(s) + 100, "/var/tmp/%s-%ld", last, ast_random());
- if (stat(s, &stbuf) < 0) {
- ast_log(LOG_WARNING, "Failed to stat '%s': %s\n", s, strerror(errno));
- return -1;
- }
- /* Make sure it's not a directory */
- if (S_ISDIR(stbuf.st_mode))
- return -1;
- ifd = open(s, O_RDONLY);
- if (ifd < 0) {
- ast_log(LOG_WARNING, "Cannot open '%s': %s\n", s, strerror(errno));
- return -1;
- }
- fd = open(s2, O_RDWR | O_CREAT | O_EXCL, AST_FILE_MODE);
- if (fd < 0) {
- ast_log(LOG_WARNING, "Cannot open '%s' for writing: %s\n", s2, strerror(errno));
- close(ifd);
- return -1;
- }
- /* Unlink our newly created file */
- unlink(s2);
-
- /* Now copy the firmware into it */
- len = stbuf.st_size;
- while(len) {
- chunk = len;
- if (chunk > sizeof(buf))
- chunk = sizeof(buf);
- res = read(ifd, buf, chunk);
- if (res != chunk) {
- ast_log(LOG_WARNING, "Only read %d of %d bytes of data :(: %s\n", res, chunk, strerror(errno));
- close(ifd);
- close(fd);
- return -1;
- }
- res = write(fd, buf, chunk);
- if (res != chunk) {
- ast_log(LOG_WARNING, "Only write %d of %d bytes of data :(: %s\n", res, chunk, strerror(errno));
- close(ifd);
- close(fd);
- return -1;
- }
- len -= chunk;
- }
- close(ifd);
- /* Return to the beginning */
- lseek(fd, 0, SEEK_SET);
- if ((res = read(fd, &fwh2, sizeof(fwh2))) != sizeof(fwh2)) {
- ast_log(LOG_WARNING, "Unable to read firmware header in '%s'\n", s);
- close(fd);
- return -1;
- }
- if (ntohl(fwh2.magic) != IAX_FIRMWARE_MAGIC) {
- ast_log(LOG_WARNING, "'%s' is not a valid firmware file\n", s);
- close(fd);
- return -1;
- }
- if (ntohl(fwh2.datalen) != (stbuf.st_size - sizeof(fwh2))) {
- ast_log(LOG_WARNING, "Invalid data length in firmware '%s'\n", s);
- close(fd);
- return -1;
- }
- if (fwh2.devname[sizeof(fwh2.devname) - 1] || ast_strlen_zero((char *)fwh2.devname)) {
- ast_log(LOG_WARNING, "No or invalid device type specified for '%s'\n", s);
- close(fd);
- return -1;
- }
- fwh = (struct ast_iax2_firmware_header*)mmap(NULL, stbuf.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
- if (fwh == MAP_FAILED) {
- ast_log(LOG_WARNING, "mmap failed: %s\n", strerror(errno));
- close(fd);
- return -1;
- }
- MD5Init(&md5);
- MD5Update(&md5, fwh->data, ntohl(fwh->datalen));
- MD5Final(sum, &md5);
- if (memcmp(sum, fwh->chksum, sizeof(sum))) {
- ast_log(LOG_WARNING, "Firmware file '%s' fails checksum\n", s);
- munmap((void*)fwh, stbuf.st_size);
- close(fd);
- return -1;
- }
- AST_LIST_TRAVERSE(&firmwares, cur, list) {
- if (!strcmp((char *)cur->fwh->devname, (char *)fwh->devname)) {
- /* Found a candidate */
- if (cur->dead || (ntohs(cur->fwh->version) < ntohs(fwh->version)))
- /* The version we have on loaded is older, load this one instead */
- break;
- /* This version is no newer than what we have. Don't worry about it.
- We'll consider it a proper load anyhow though */
- munmap((void*)fwh, stbuf.st_size);
- close(fd);
- return 0;
- }
- }
-
- if (!cur && ((cur = ast_calloc(1, sizeof(*cur))))) {
- cur->fd = -1;
- AST_LIST_INSERT_TAIL(&firmwares, cur, list);
- }
-
- if (cur) {
- if (cur->fwh)
- munmap((void*)cur->fwh, cur->mmaplen);
- if (cur->fd > -1)
- close(cur->fd);
- cur->fwh = fwh;
- cur->fd = fd;
- cur->mmaplen = stbuf.st_size;
- cur->dead = 0;
- }
-
- return 0;
- }
- static int iax_check_version(char *dev)
- {
- int res = 0;
- struct iax_firmware *cur = NULL;
- if (ast_strlen_zero(dev))
- return 0;
- AST_LIST_LOCK(&firmwares);
- AST_LIST_TRAVERSE(&firmwares, cur, list) {
- if (!strcmp(dev, (char *)cur->fwh->devname)) {
- res = ntohs(cur->fwh->version);
- break;
- }
- }
- AST_LIST_UNLOCK(&firmwares);
- return res;
- }
- static int iax_firmware_append(struct iax_ie_data *ied, const unsigned char *dev, unsigned int desc)
- {
- int res = -1;
- unsigned int bs = desc & 0xff;
- unsigned int start = (desc >> 8) & 0xffffff;
- unsigned int bytes;
- struct iax_firmware *cur;
- if (ast_strlen_zero((char *)dev) || !bs)
- return -1;
- start *= bs;
-
- AST_LIST_LOCK(&firmwares);
- AST_LIST_TRAVERSE(&firmwares, cur, list) {
- if (strcmp((char *)dev, (char *)cur->fwh->devname))
- continue;
- iax_ie_append_int(ied, IAX_IE_FWBLOCKDESC, desc);
- if (start < ntohl(cur->fwh->datalen)) {
- bytes = ntohl(cur->fwh->datalen) - start;
- if (bytes > bs)
- bytes = bs;
- iax_ie_append_raw(ied, IAX_IE_FWBLOCKDATA, cur->fwh->data + start, bytes);
- } else {
- bytes = 0;
- iax_ie_append(ied, IAX_IE_FWBLOCKDATA);
- }
- if (bytes == bs)
- res = 0;
- else
- res = 1;
- break;
- }
- AST_LIST_UNLOCK(&firmwares);
- return res;
- }
- static void reload_firmware(int unload)
- {
- struct iax_firmware *cur = NULL;
- DIR *fwd;
- struct dirent *de;
- char dir[256], fn[256];
- AST_LIST_LOCK(&firmwares);
- /* Mark all as dead */
- AST_LIST_TRAVERSE(&firmwares, cur, list)
- cur->dead = 1;
- /* Now that we have marked them dead... load new ones */
- if (!unload) {
- snprintf(dir, sizeof(dir), "%s/firmware/iax", ast_config_AST_DATA_DIR);
- fwd = opendir(dir);
- if (fwd) {
- while((de = readdir(fwd))) {
- if (de->d_name[0] != '.') {
- snprintf(fn, sizeof(fn), "%s/%s", dir, de->d_name);
- if (!try_firmware(fn)) {
- ast_verb(2, "Loaded firmware '%s'\n", de->d_name);
- }
- }
- }
- closedir(fwd);
- } else
- ast_log(LOG_WARNING, "Error opening firmware directory '%s': %s\n", dir, strerror(errno));
- }
- /* Clean up leftovers */
- AST_LIST_TRAVERSE_SAFE_BEGIN(&firmwares, cur, list) {
- if (!cur->dead)
- continue;
- AST_LIST_REMOVE_CURRENT(list);
- destroy_firmware(cur);
- }
- AST_LIST_TRAVERSE_SAFE_END;
- AST_LIST_UNLOCK(&firmwares);
- }
- /*!
- * \note This function assumes that iaxsl[callno] is locked when called.
- *
- * \note IMPORTANT NOTE!!! Any time this function is used, even if iaxs[callno]
- * was valid before calling it, it may no longer be valid after calling it.
- * This function calls iax2_queue_frame(), which may unlock and lock the mutex
- * associated with this callno, meaning that another thread may grab it and destroy the call.
- */
- static int __do_deliver(void *data)
- {
- /* Just deliver the packet by using queueing. This is called by
- the IAX thread with the iaxsl lock held. */
- struct iax_frame *fr = data;
- fr->retrans = -1;
- ast_clear_flag(&fr->af, AST_FRFLAG_HAS_TIMING_INFO);
- if (iaxs[fr->callno] && !ast_test_flag64(iaxs[fr->callno], IAX_ALREADYGONE))
- iax2_queue_frame(fr->callno, &fr->af);
- /* Free our iax frame */
- iax2_frame_free(fr);
- /* And don't run again */
- return 0;
- }
- static int handle_error(void)
- {
- /* XXX Ideally we should figure out why an error occurred and then abort those
- rather than continuing to try. Unfortunately, the published interface does
- not seem to work XXX */
- #if 0
- struct sockaddr_in *sin;
- int res;
- struct msghdr m;
- struct sock_extended_err e;
- m.msg_name = NULL;
- m.msg_namelen = 0;
- m.msg_iov = NULL;
- m.msg_control = &e;
- m.msg_controllen = sizeof(e);
- m.msg_flags = 0;
- res = recvmsg(netsocket, &m, MSG_ERRQUEUE);
- if (res < 0)
- ast_log(LOG_WARNING, "Error detected, but unable to read error: %s\n", strerror(errno));
- else {
- if (m.msg_controllen) {
- sin = (struct sockaddr_in *)SO_EE_OFFENDER(&e);
- if (sin)
- ast_log(LOG_WARNING, "Receive error from %s\n", ast_inet_ntoa(sin->sin_addr));
- else
- ast_log(LOG_WARNING, "No address detected??\n");
- } else {
- ast_log(LOG_WARNING, "Local error: %s\n", strerror(e.ee_errno));
- }
- }
- #endif
- return 0;
- }
- static int transmit_trunk(struct iax_frame *f, struct sockaddr_in *sin, int sockfd)
- {
- int res;
- res = sendto(sockfd, f->data, f->datalen, 0,(struct sockaddr *)sin,
- sizeof(*sin));
- if (res < 0) {
- ast_debug(1, "Received error: %s\n", strerror(errno));
- handle_error();
- } else
- res = 0;
- return res;
- }
- static int send_packet(struct iax_frame *f)
- {
- int res;
- int callno = f->callno;
- /* Don't send if there was an error, but return error instead */
- if (!callno || !iaxs[callno] || iaxs[callno]->error)
- return -1;
- /* Called with iaxsl held */
- if (iaxdebug)
- ast_debug(3, "Sending %u on %d/%d to %s:%d\n", f->ts, callno, iaxs[callno]->peercallno, ast_inet_ntoa(iaxs[callno]->addr.sin_addr), ntohs(iaxs[callno]->addr.sin_port));
- if (f->transfer) {
- iax_outputframe(f, NULL, 0, &iaxs[callno]->transfer, f->datalen - sizeof(struct ast_iax2_full_hdr));
- res = sendto(iaxs[callno]->sockfd, f->data, f->datalen, 0,(struct sockaddr *)&iaxs[callno]->transfer, sizeof(iaxs[callno]->transfer));
- } else {
- iax_outputframe(f, NULL, 0, &iaxs[callno]->addr, f->datalen - sizeof(struct ast_iax2_full_hdr));
- res = sendto(iaxs[callno]->sockfd, f->data, f->datalen, 0,(struct sockaddr *)&iaxs[callno]->addr, sizeof(iaxs[callno]->addr));
- }
- if (res < 0) {
- if (iaxdebug)
- ast_debug(1, "Received error: %s\n", strerror(errno));
- handle_error();
- } else
- res = 0;
- return res;
- }
- /*!
- * \note Since this function calls iax2_queue_hangup(), the pvt struct
- * for the given call number may disappear during its execution.
- */
- static int iax2_predestroy(int callno)
- {
- struct ast_channel *c = NULL;
- struct chan_iax2_pvt *pvt = iaxs[callno];
- if (!pvt)
- return -1;
- if (!ast_test_flag64(pvt, IAX_ALREADYGONE)) {
- iax2_destroy_helper(pvt);
- ast_set_flag64(pvt, IAX_ALREADYGONE);
- }
- if ((c = pvt->owner)) {
- ast_channel_tech_pvt_set(c, NULL);
- iax2_queue_hangup(callno);
- pvt->owner = NULL;
- ast_module_unref(ast_module_info->self);
- }
- return 0;
- }
- static void iax2_destroy(int callno)
- {
- struct chan_iax2_pvt *pvt = NULL;
- struct ast_channel *owner = NULL;
- retry:
- if ((pvt = iaxs[callno])) {
- #if 0
- /* iax2_destroy_helper gets called from this function later on. When
- * called twice, we get the (previously) familiar FRACK! errors in
- * devmode, from the scheduler. An alternative to this approach is to
- * reset the scheduler entries to -1 when they're deleted in
- * iax2_destroy_helper(). That approach was previously decided to be
- * "wrong" because "the memory is going to be deallocated anyway. Why
- * should we be resetting those values?" */
- iax2_destroy_helper(pvt);
- #endif
- }
- owner = pvt ? pvt->owner : NULL;
- if (owner) {
- if (ast_channel_trylock(owner)) {
- ast_debug(3, "Avoiding IAX destroy deadlock\n");
- DEADLOCK_AVOIDANCE(&iaxsl[callno]);
- goto retry;
- }
- }
- if (!owner) {
- iaxs[callno] = NULL;
- }
- if (pvt) {
- if (!owner) {
- pvt->owner = NULL;
- } else {
- /* If there's an owner, prod it to give up */
- /* It is ok to use ast_queue_hangup() here instead of iax2_queue_hangup()
- * because we already hold the owner channel lock. */
- ast_queue_hangup(owner);
- }
- if (pvt->peercallno) {
- remove_by_peercallno(pvt);
- }
- if (pvt->transfercallno) {
- remove_by_transfercallno(pvt);
- }
- if (!owner) {
- ao2_ref(pvt, -1);
- pvt = NULL;
- }
- }
- if (owner) {
- ast_channel_unlock(owner);
- }
- }
- static int update_packet(struct iax_frame *f)
- {
- /* Called with iaxsl lock held, and iaxs[callno] non-NULL */
- struct ast_iax2_full_hdr *fh = f->data;
- struct ast_frame af;
- /* if frame is encrypted. decrypt before updating it. */
- if (f->encmethods) {
- decode_frame(&f->mydcx, fh, &af, &f->datalen);
- }
- /* Mark this as a retransmission */
- fh->dcallno = ntohs(IAX_FLAG_RETRANS | f->dcallno);
- /* Update iseqno */
- f->iseqno = iaxs[f->callno]->iseqno;
- fh->iseqno = f->iseqno;
- /* Now re-encrypt the frame */
- if (f->encmethods) {
- /* since this is a retransmit frame, create a new random padding
- * before re-encrypting. */
- build_rand_pad(f->semirand, sizeof(f->semirand));
- encrypt_frame(&f->ecx, fh, f->semirand, &f->datalen);
- }
- return 0;
- }
- static int attempt_transmit(const void *data);
- static void __attempt_transmit(const void *data)
- {
- /* Attempt to transmit the frame to the remote peer...
- Called without iaxsl held. */
- struct iax_frame *f = (struct iax_frame *)data;
- int freeme = 0;
- int callno = f->callno;
- /* Make sure this call is still active */
- if (callno)
- ast_mutex_lock(&iaxsl[callno]);
- if (callno && iaxs[callno]) {
- if (f->retries < 0) {
- /* Already ACK'd */
- freeme = 1;
- } else if (f->retries >= max_retries) {
- /* Too many attempts. Record an error. */
- if (f->transfer) {
- /* Transfer timeout */
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_TXREJ, 0, NULL, 0, -1);
- } else if (f->final) {
- iax2_destroy(callno);
- } else {
- if (iaxs[callno]->owner) {
- ast_log(LOG_WARNING, "Max retries exceeded to host %s on %s (type = %u, subclass = %d, ts=%u, seqno=%d)\n",
- ast_inet_ntoa(iaxs[f->callno]->addr.sin_addr),
- ast_channel_name(iaxs[f->callno]->owner),
- f->af.frametype,
- f->af.subclass.integer,
- f->ts,
- f->oseqno);
- }
- iaxs[callno]->error = ETIMEDOUT;
- if (iaxs[callno]->owner) {
- struct ast_frame fr = { AST_FRAME_CONTROL, { AST_CONTROL_HANGUP }, .data.uint32 = AST_CAUSE_DESTINATION_OUT_OF_ORDER };
- /* Hangup the fd */
- iax2_queue_frame(callno, &fr); /* XXX */
- /* Remember, owner could disappear */
- if (iaxs[callno] && iaxs[callno]->owner)
- ast_channel_hangupcause_set(iaxs[callno]->owner, AST_CAUSE_DESTINATION_OUT_OF_ORDER);
- } else {
- if (iaxs[callno]->reg) {
- memset(&iaxs[callno]->reg->us, 0, sizeof(iaxs[callno]->reg->us));
- iaxs[callno]->reg->regstate = REG_STATE_TIMEOUT;
- iaxs[callno]->reg->refresh = IAX_DEFAULT_REG_EXPIRE;
- }
- iax2_destroy(callno);
- }
- }
- freeme = 1;
- } else {
- /* Update it if it needs it */
- update_packet(f);
- /* Attempt transmission */
- send_packet(f);
- f->retries++;
- /* Try again later after 10 times as long */
- f->retrytime *= 10;
- if (f->retrytime > MAX_RETRY_TIME)
- f->retrytime = MAX_RETRY_TIME;
- /* Transfer messages max out at one second */
- if (f->transfer && (f->retrytime > 1000))
- f->retrytime = 1000;
- f->retrans = iax2_sched_add(sched, f->retrytime, attempt_transmit, f);
- }
- } else {
- /* Make sure it gets freed */
- f->retries = -1;
- freeme = 1;
- }
- if (freeme) {
- /* Don't attempt delivery, just remove it from the queue */
- AST_LIST_REMOVE(&frame_queue[callno], f, list);
- ast_mutex_unlock(&iaxsl[callno]);
- f->retrans = -1; /* this is safe because this is the scheduled function */
- /* Free the IAX frame */
- iax2_frame_free(f);
- } else if (callno) {
- ast_mutex_unlock(&iaxsl[callno]);
- }
- }
- static int attempt_transmit(const void *data)
- {
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__attempt_transmit, data))
- #endif
- __attempt_transmit(data);
- return 0;
- }
- static char *handle_cli_iax2_prune_realtime(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct iax2_peer *peer = NULL;
- struct iax2_user *user = NULL;
- static const char * const choices[] = { "all", NULL };
- char *cmplt;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 prune realtime";
- e->usage =
- "Usage: iax2 prune realtime [<peername>|all]\n"
- " Prunes object(s) from the cache\n";
- return NULL;
- case CLI_GENERATE:
- if (a->pos == 3) {
- cmplt = ast_cli_complete(a->word, choices, a->n);
- if (!cmplt)
- cmplt = complete_iax2_peers(a->line, a->word, a->pos, a->n - sizeof(choices), IAX_RTCACHEFRIENDS);
- return cmplt;
- }
- return NULL;
- }
- if (a->argc != 4)
- return CLI_SHOWUSAGE;
- if (!strcmp(a->argv[3], "all")) {
- prune_users();
- prune_peers();
- ast_cli(a->fd, "Cache flushed successfully.\n");
- return CLI_SUCCESS;
- }
- peer = find_peer(a->argv[3], 0);
- user = find_user(a->argv[3]);
- if (peer || user) {
- if (peer) {
- if (ast_test_flag64(peer, IAX_RTCACHEFRIENDS)) {
- ast_set_flag64(peer, IAX_RTAUTOCLEAR);
- expire_registry(peer_ref(peer));
- ast_cli(a->fd, "Peer %s was removed from the cache.\n", a->argv[3]);
- } else {
- ast_cli(a->fd, "Peer %s is not eligible for this operation.\n", a->argv[3]);
- }
- peer_unref(peer);
- }
- if (user) {
- if (ast_test_flag64(user, IAX_RTCACHEFRIENDS)) {
- ast_set_flag64(user, IAX_RTAUTOCLEAR);
- ast_cli(a->fd, "User %s was removed from the cache.\n", a->argv[3]);
- } else {
- ast_cli(a->fd, "User %s is not eligible for this operation.\n", a->argv[3]);
- }
- ao2_unlink(users,user);
- user_unref(user);
- }
- } else {
- ast_cli(a->fd, "%s was not found in the cache.\n", a->argv[3]);
- }
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_test_losspct(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 test losspct";
- e->usage =
- "Usage: iax2 test losspct <percentage>\n"
- " For testing, throws away <percentage> percent of incoming packets\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 4)
- return CLI_SHOWUSAGE;
- test_losspct = atoi(a->argv[3]);
- return CLI_SUCCESS;
- }
- #ifdef IAXTESTS
- static char *handle_cli_iax2_test_late(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 test late";
- e->usage =
- "Usage: iax2 test late <ms>\n"
- " For testing, count the next frame as <ms> ms late\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 4)
- return CLI_SHOWUSAGE;
- test_late = atoi(a->argv[3]);
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_test_resync(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 test resync";
- e->usage =
- "Usage: iax2 test resync <ms>\n"
- " For testing, adjust all future frames by <ms> ms\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 4)
- return CLI_SHOWUSAGE;
- test_resync = atoi(a->argv[3]);
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_test_jitter(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 test jitter";
- e->usage =
- "Usage: iax2 test jitter <ms> <pct>\n"
- " For testing, simulate maximum jitter of +/- <ms> on <pct>\n"
- " percentage of packets. If <pct> is not specified, adds\n"
- " jitter to all packets.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc < 4 || a->argc > 5)
- return CLI_SHOWUSAGE;
- test_jit = atoi(a->argv[3]);
- if (a->argc == 5)
- test_jitpct = atoi(a->argv[4]);
- return CLI_SUCCESS;
- }
- #endif /* IAXTESTS */
- /*! \brief peer_status: Report Peer status in character string */
- /* returns 1 if peer is online, -1 if unmonitored */
- static int peer_status(struct iax2_peer *peer, char *status, int statuslen)
- {
- int res = 0;
- if (peer->maxms) {
- if (peer->lastms < 0) {
- ast_copy_string(status, "UNREACHABLE", statuslen);
- } else if (peer->lastms > peer->maxms) {
- snprintf(status, statuslen, "LAGGED (%d ms)", peer->lastms);
- res = 1;
- } else if (peer->lastms) {
- snprintf(status, statuslen, "OK (%d ms)", peer->lastms);
- res = 1;
- } else {
- ast_copy_string(status, "UNKNOWN", statuslen);
- }
- } else {
- ast_copy_string(status, "Unmonitored", statuslen);
- res = -1;
- }
- return res;
- }
- /*! \brief Show one peer in detail */
- static char *handle_cli_iax2_show_peer(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- char status[30];
- char cbuf[256];
- struct iax2_peer *peer;
- char codec_buf[512];
- struct ast_str *encmethods = ast_str_alloca(256);
- int x = 0, load_realtime = 0;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show peer";
- e->usage =
- "Usage: iax2 show peer <name>\n"
- " Display details on specific IAX peer\n";
- return NULL;
- case CLI_GENERATE:
- if (a->pos == 3)
- return complete_iax2_peers(a->line, a->word, a->pos, a->n, 0);
- return NULL;
- }
- if (a->argc < 4)
- return CLI_SHOWUSAGE;
- load_realtime = (a->argc == 5 && !strcmp(a->argv[4], "load")) ? 1 : 0;
- peer = find_peer(a->argv[3], load_realtime);
- if (peer) {
- struct sockaddr_in peer_addr;
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- encmethods_to_str(peer->encmethods, &encmethods);
- ast_cli(a->fd, "\n\n");
- ast_cli(a->fd, " * Name : %s\n", peer->name);
- ast_cli(a->fd, " Description : %s\n", peer->description);
- ast_cli(a->fd, " Secret : %s\n", ast_strlen_zero(peer->secret) ? "<Not set>" : "<Set>");
- ast_cli(a->fd, " Context : %s\n", peer->context);
- ast_cli(a->fd, " Parking lot : %s\n", peer->parkinglot);
- ast_cli(a->fd, " Mailbox : %s\n", peer->mailbox);
- ast_cli(a->fd, " Dynamic : %s\n", ast_test_flag64(peer, IAX_DYNAMIC) ? "Yes" : "No");
- ast_cli(a->fd, " Callnum limit: %d\n", peer->maxcallno);
- ast_cli(a->fd, " Calltoken req: %s\n", (peer->calltoken_required == CALLTOKEN_YES) ? "Yes" : ((peer->calltoken_required == CALLTOKEN_AUTO) ? "Auto" : "No"));
- ast_cli(a->fd, " Trunk : %s\n", ast_test_flag64(peer, IAX_TRUNK) ? "Yes" : "No");
- ast_cli(a->fd, " Encryption : %s\n", peer->encmethods ? ast_str_buffer(encmethods) : "No");
- ast_cli(a->fd, " Callerid : %s\n", ast_callerid_merge(cbuf, sizeof(cbuf), peer->cid_name, peer->cid_num, "<unspecified>"));
- ast_cli(a->fd, " Expire : %d\n", peer->expire);
- ast_cli(a->fd, " ACL : %s\n", (ast_acl_list_is_empty(peer->acl) ? "No" : "Yes"));
- ast_cli(a->fd, " Addr->IP : %s Port %d\n", peer_addr.sin_addr.s_addr ? ast_inet_ntoa(peer_addr.sin_addr) : "(Unspecified)", ntohs(peer_addr.sin_port));
- ast_cli(a->fd, " Defaddr->IP : %s Port %d\n", ast_inet_ntoa(peer->defaddr.sin_addr), ntohs(peer->defaddr.sin_port));
- ast_cli(a->fd, " Username : %s\n", peer->username);
- ast_cli(a->fd, " Codecs : ");
- iax2_getformatname_multiple(codec_buf, sizeof(codec_buf) -1, peer->capability);
- ast_cli(a->fd, "%s\n", codec_buf);
- ast_cli(a->fd, " Codec Order : (");
- for(x = 0; x < AST_CODEC_PREF_SIZE; x++) {
- struct ast_format tmpfmt;
- if(!(ast_codec_pref_index(&peer->prefs, x, &tmpfmt)))
- break;
- ast_cli(a->fd, "%s", ast_getformatname(&tmpfmt));
- if(x < 31 && ast_codec_pref_index(&peer->prefs, x+1, &tmpfmt))
- ast_cli(a->fd, "|");
- }
- if (!x)
- ast_cli(a->fd, "none");
- ast_cli(a->fd, ")\n");
- ast_cli(a->fd, " Status : ");
- peer_status(peer, status, sizeof(status));
- ast_cli(a->fd, "%s\n",status);
- ast_cli(a->fd, " Qualify : every %dms when OK, every %dms when UNREACHABLE (sample smoothing %s)\n", peer->pokefreqok, peer->pokefreqnotok, peer->smoothing ? "On" : "Off");
- ast_cli(a->fd, "\n");
- peer_unref(peer);
- } else {
- ast_cli(a->fd, "Peer %s not found.\n", a->argv[3]);
- ast_cli(a->fd, "\n");
- }
- return CLI_SUCCESS;
- }
- static char *complete_iax2_peers(const char *line, const char *word, int pos, int state, uint64_t flags)
- {
- int which = 0;
- struct iax2_peer *peer;
- char *res = NULL;
- int wordlen = strlen(word);
- struct ao2_iterator i;
- i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- if (!strncasecmp(peer->name, word, wordlen) && ++which > state
- && (!flags || ast_test_flag64(peer, flags))) {
- res = ast_strdup(peer->name);
- peer_unref(peer);
- break;
- }
- peer_unref(peer);
- }
- ao2_iterator_destroy(&i);
- return res;
- }
- static char *handle_cli_iax2_show_stats(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct iax_frame *cur;
- int cnt = 0, dead = 0, final = 0, i = 0;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show stats";
- e->usage =
- "Usage: iax2 show stats\n"
- " Display statistics on IAX channel driver.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 3)
- return CLI_SHOWUSAGE;
- for (i = 0; i < ARRAY_LEN(frame_queue); i++) {
- ast_mutex_lock(&iaxsl[i]);
- AST_LIST_TRAVERSE(&frame_queue[i], cur, list) {
- if (cur->retries < 0)
- dead++;
- if (cur->final)
- final++;
- cnt++;
- }
- ast_mutex_unlock(&iaxsl[i]);
- }
- ast_cli(a->fd, " IAX Statistics\n");
- ast_cli(a->fd, "---------------------\n");
- ast_cli(a->fd, "Outstanding frames: %d (%d ingress, %d egress)\n", iax_get_frames(), iax_get_iframes(), iax_get_oframes());
- ast_cli(a->fd, "%d timed and %d untimed transmits; MTU %d/%d/%d\n", trunk_timed, trunk_untimed,
- trunk_maxmtu, trunk_nmaxmtu, global_max_trunk_mtu);
- ast_cli(a->fd, "Packets in transmit queue: %d dead, %d final, %d total\n\n", dead, final, cnt);
- trunk_timed = trunk_untimed = 0;
- if (trunk_maxmtu > trunk_nmaxmtu)
- trunk_nmaxmtu = trunk_maxmtu;
- return CLI_SUCCESS;
- }
- /*! \brief Set trunk MTU from CLI */
- static char *handle_cli_iax2_set_mtu(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- int mtuv;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 set mtu";
- e->usage =
- "Usage: iax2 set mtu <value>\n"
- " Set the system-wide IAX IP mtu to <value> bytes net or\n"
- " zero to disable. Disabling means that the operating system\n"
- " must handle fragmentation of UDP packets when the IAX2 trunk\n"
- " packet exceeds the UDP payload size. This is substantially\n"
- " below the IP mtu. Try 1240 on ethernets. Must be 172 or\n"
- " greater for G.711 samples.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 4)
- return CLI_SHOWUSAGE;
- if (strncasecmp(a->argv[3], "default", strlen(a->argv[3])) == 0)
- mtuv = MAX_TRUNK_MTU;
- else
- mtuv = atoi(a->argv[3]);
- if (mtuv == 0) {
- ast_cli(a->fd, "Trunk MTU control disabled (mtu was %d)\n", global_max_trunk_mtu);
- global_max_trunk_mtu = 0;
- return CLI_SUCCESS;
- }
- if (mtuv < 172 || mtuv > 4000) {
- ast_cli(a->fd, "Trunk MTU must be between 172 and 4000\n");
- return CLI_SHOWUSAGE;
- }
- ast_cli(a->fd, "Trunk MTU changed from %d to %d\n", global_max_trunk_mtu, mtuv);
- global_max_trunk_mtu = mtuv;
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_show_cache(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct iax2_dpcache *dp = NULL;
- char tmp[1024], *pc = NULL;
- int s, x, y;
- struct timeval now = ast_tvnow();
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show cache";
- e->usage =
- "Usage: iax2 show cache\n"
- " Display currently cached IAX Dialplan results.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- AST_LIST_LOCK(&dpcache);
- ast_cli(a->fd, "%-20.20s %-12.12s %-9.9s %-8.8s %s\n", "Peer/Context", "Exten", "Exp.", "Wait.", "Flags");
- AST_LIST_TRAVERSE(&dpcache, dp, cache_list) {
- s = dp->expiry.tv_sec - now.tv_sec;
- tmp[0] = '\0';
- if (dp->flags & CACHE_FLAG_EXISTS)
- strncat(tmp, "EXISTS|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_NONEXISTENT)
- strncat(tmp, "NONEXISTENT|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_CANEXIST)
- strncat(tmp, "CANEXIST|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_PENDING)
- strncat(tmp, "PENDING|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_TIMEOUT)
- strncat(tmp, "TIMEOUT|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_TRANSMITTED)
- strncat(tmp, "TRANSMITTED|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_MATCHMORE)
- strncat(tmp, "MATCHMORE|", sizeof(tmp) - strlen(tmp) - 1);
- if (dp->flags & CACHE_FLAG_UNKNOWN)
- strncat(tmp, "UNKNOWN|", sizeof(tmp) - strlen(tmp) - 1);
- /* Trim trailing pipe */
- if (!ast_strlen_zero(tmp)) {
- tmp[strlen(tmp) - 1] = '\0';
- } else {
- ast_copy_string(tmp, "(none)", sizeof(tmp));
- }
- y = 0;
- pc = strchr(dp->peercontext, '@');
- if (!pc) {
- pc = dp->peercontext;
- } else {
- pc++;
- }
- for (x = 0; x < ARRAY_LEN(dp->waiters); x++) {
- if (dp->waiters[x] > -1)
- y++;
- }
- if (s > 0) {
- ast_cli(a->fd, "%-20.20s %-12.12s %-9d %-8d %s\n", pc, dp->exten, s, y, tmp);
- } else {
- ast_cli(a->fd, "%-20.20s %-12.12s %-9.9s %-8d %s\n", pc, dp->exten, "(expired)", y, tmp);
- }
- }
- AST_LIST_UNLOCK(&dpcache);
- return CLI_SUCCESS;
- }
- static unsigned int calc_rxstamp(struct chan_iax2_pvt *p, unsigned int offset);
- static void unwrap_timestamp(struct iax_frame *fr)
- {
- /* Video mini frames only encode the lower 15 bits of the session
- * timestamp, but other frame types (e.g. audio) encode 16 bits. */
- const int ts_shift = (fr->af.frametype == AST_FRAME_VIDEO) ? 15 : 16;
- const int lower_mask = (1 << ts_shift) - 1;
- const int upper_mask = ~lower_mask;
- const int last_upper = iaxs[fr->callno]->last & upper_mask;
- if ( (fr->ts & upper_mask) == last_upper ) {
- const int x = fr->ts - iaxs[fr->callno]->last;
- const int threshold = (ts_shift == 15) ? 25000 : 50000;
- if (x < -threshold) {
- /* Sudden big jump backwards in timestamp:
- What likely happened here is that miniframe timestamp has circled but we haven't
- gotten the update from the main packet. We'll just pretend that we did, and
- update the timestamp appropriately. */
- fr->ts = (last_upper + (1 << ts_shift)) | (fr->ts & lower_mask);
- if (iaxdebug)
- ast_debug(1, "schedule_delivery: pushed forward timestamp\n");
- } else if (x > threshold) {
- /* Sudden apparent big jump forwards in timestamp:
- What's likely happened is this is an old miniframe belonging to the previous
- top 15 or 16-bit timestamp that has turned up out of order.
- Adjust the timestamp appropriately. */
- fr->ts = (last_upper - (1 << ts_shift)) | (fr->ts & lower_mask);
- if (iaxdebug)
- ast_debug(1, "schedule_delivery: pushed back timestamp\n");
- }
- }
- }
- static int get_from_jb(const void *p);
- static void update_jbsched(struct chan_iax2_pvt *pvt)
- {
- int when;
- when = ast_tvdiff_ms(ast_tvnow(), pvt->rxcore);
- when = jb_next(pvt->jb) - when;
- if (when <= 0) {
- /* XXX should really just empty until when > 0.. */
- when = 1;
- }
- pvt->jbid = iax2_sched_replace(pvt->jbid, sched, when, get_from_jb,
- CALLNO_TO_PTR(pvt->callno));
- }
- static void __get_from_jb(const void *p)
- {
- int callno = PTR_TO_CALLNO(p);
- struct chan_iax2_pvt *pvt = NULL;
- struct iax_frame *fr;
- jb_frame frame;
- int ret;
- long ms;
- long next;
- struct timeval now = ast_tvnow();
-
- /* Make sure we have a valid private structure before going on */
- ast_mutex_lock(&iaxsl[callno]);
- pvt = iaxs[callno];
- if (!pvt) {
- /* No go! */
- ast_mutex_unlock(&iaxsl[callno]);
- return;
- }
- pvt->jbid = -1;
-
- /* round up a millisecond since ast_sched_runq does; */
- /* prevents us from spinning while waiting for our now */
- /* to catch up with runq's now */
- now.tv_usec += 1000;
-
- ms = ast_tvdiff_ms(now, pvt->rxcore);
-
- if(ms >= (next = jb_next(pvt->jb))) {
- struct ast_format voicefmt;
- ast_format_from_old_bitfield(&voicefmt, pvt->voiceformat);
- ret = jb_get(pvt->jb, &frame, ms, ast_codec_interp_len(&voicefmt));
- switch(ret) {
- case JB_OK:
- fr = frame.data;
- __do_deliver(fr);
- /* __do_deliver() can cause the call to disappear */
- pvt = iaxs[callno];
- break;
- case JB_INTERP:
- {
- struct ast_frame af = { 0, };
-
- /* create an interpolation frame */
- af.frametype = AST_FRAME_VOICE;
- ast_format_copy(&af.subclass.format, &voicefmt);
- af.samples = frame.ms * (ast_format_rate(&voicefmt) / 1000);
- af.src = "IAX2 JB interpolation";
- af.delivery = ast_tvadd(pvt->rxcore, ast_samp2tv(next, 1000));
- af.offset = AST_FRIENDLY_OFFSET;
-
- /* queue the frame: For consistency, we would call __do_deliver here, but __do_deliver wants an iax_frame,
- * which we'd need to malloc, and then it would free it. That seems like a drag */
- if (!ast_test_flag64(iaxs[callno], IAX_ALREADYGONE)) {
- iax2_queue_frame(callno, &af);
- /* iax2_queue_frame() could cause the call to disappear */
- pvt = iaxs[callno];
- }
- }
- break;
- case JB_DROP:
- iax2_frame_free(frame.data);
- break;
- case JB_NOFRAME:
- case JB_EMPTY:
- /* do nothing */
- break;
- default:
- /* shouldn't happen */
- break;
- }
- }
- if (pvt)
- update_jbsched(pvt);
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int get_from_jb(const void *data)
- {
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__get_from_jb, data))
- #endif
- __get_from_jb(data);
- return 0;
- }
- /*!
- * \note This function assumes fr->callno is locked
- *
- * \note IMPORTANT NOTE!!! Any time this function is used, even if iaxs[callno]
- * was valid before calling it, it may no longer be valid after calling it.
- */
- static int schedule_delivery(struct iax_frame *fr, int updatehistory, int fromtrunk, unsigned int *tsout)
- {
- int type, len;
- int ret;
- int needfree = 0;
- struct ast_channel *owner = NULL;
- struct ast_channel *bridge = NULL;
- /*
- * Clear fr->af.data if there is no data in the buffer. Things
- * like AST_CONTROL_HOLD without a suggested music class must
- * have a NULL pointer.
- */
- if (!fr->af.datalen) {
- memset(&fr->af.data, 0, sizeof(fr->af.data));
- }
- /* Attempt to recover wrapped timestamps */
- unwrap_timestamp(fr);
- /* delivery time is sender's sent timestamp converted back into absolute time according to our clock */
- if ( !fromtrunk && !ast_tvzero(iaxs[fr->callno]->rxcore))
- fr->af.delivery = ast_tvadd(iaxs[fr->callno]->rxcore, ast_samp2tv(fr->ts, 1000));
- else {
- #if 0
- ast_debug(1, "schedule_delivery: set delivery to 0 as we don't have an rxcore yet, or frame is from trunk.\n");
- #endif
- fr->af.delivery = ast_tv(0,0);
- }
- type = JB_TYPE_CONTROL;
- len = 0;
- if(fr->af.frametype == AST_FRAME_VOICE) {
- type = JB_TYPE_VOICE;
- len = ast_codec_get_samples(&fr->af) / (ast_format_rate(&fr->af.subclass.format) / 1000);
- } else if(fr->af.frametype == AST_FRAME_CNG) {
- type = JB_TYPE_SILENCE;
- }
- if ( (!ast_test_flag64(iaxs[fr->callno], IAX_USEJITTERBUF)) ) {
- if (tsout)
- *tsout = fr->ts;
- __do_deliver(fr);
- return -1;
- }
- iax2_lock_owner(fr->callno);
- if (!iaxs[fr->callno]) {
- /* The call dissappeared so discard this frame that we could not send. */
- iax2_frame_free(fr);
- return -1;
- }
- if ((owner = iaxs[fr->callno]->owner))
- bridge = ast_bridged_channel(owner);
- /* if the user hasn't requested we force the use of the jitterbuffer, and we're bridged to
- * a channel that can accept jitter, then flush and suspend the jb, and send this frame straight through */
- if ( (!ast_test_flag64(iaxs[fr->callno], IAX_FORCEJITTERBUF)) && owner && bridge && (ast_channel_tech(bridge)->properties & AST_CHAN_TP_WANTSJITTER) ) {
- jb_frame frame;
- ast_channel_unlock(owner);
- /* deliver any frames in the jb */
- while (jb_getall(iaxs[fr->callno]->jb, &frame) == JB_OK) {
- __do_deliver(frame.data);
- /* __do_deliver() can make the call disappear */
- if (!iaxs[fr->callno])
- return -1;
- }
- jb_reset(iaxs[fr->callno]->jb);
- AST_SCHED_DEL(sched, iaxs[fr->callno]->jbid);
- /* deliver this frame now */
- if (tsout)
- *tsout = fr->ts;
- __do_deliver(fr);
- return -1;
- }
- if (owner) {
- ast_channel_unlock(owner);
- }
- /* insert into jitterbuffer */
- /* TODO: Perhaps we could act immediately if it's not droppable and late */
- ret = jb_put(iaxs[fr->callno]->jb, fr, type, len, fr->ts,
- calc_rxstamp(iaxs[fr->callno],fr->ts));
- if (ret == JB_DROP) {
- needfree++;
- } else if (ret == JB_SCHED) {
- update_jbsched(iaxs[fr->callno]);
- }
- if (tsout)
- *tsout = fr->ts;
- if (needfree) {
- /* Free our iax frame */
- iax2_frame_free(fr);
- return -1;
- }
- return 0;
- }
- static int transmit_frame(void *data)
- {
- struct iax_frame *fr = data;
- ast_mutex_lock(&iaxsl[fr->callno]);
- fr->sentyet = 1;
- if (iaxs[fr->callno]) {
- send_packet(fr);
- }
- if (fr->retries < 0) {
- ast_mutex_unlock(&iaxsl[fr->callno]);
- /* No retransmit requested */
- iax_frame_free(fr);
- } else {
- /* We need reliable delivery. Schedule a retransmission */
- AST_LIST_INSERT_TAIL(&frame_queue[fr->callno], fr, list);
- fr->retries++;
- fr->retrans = iax2_sched_add(sched, fr->retrytime, attempt_transmit, fr);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- }
- return 0;
- }
- static int iax2_transmit(struct iax_frame *fr)
- {
- fr->sentyet = 0;
- return ast_taskprocessor_push(transmit_processor, transmit_frame, fr);
- }
- static int iax2_digit_begin(struct ast_channel *c, char digit)
- {
- return send_command_locked(PTR_TO_CALLNO(ast_channel_tech_pvt(c)), AST_FRAME_DTMF_BEGIN, digit, 0, NULL, 0, -1);
- }
- static int iax2_digit_end(struct ast_channel *c, char digit, unsigned int duration)
- {
- return send_command_locked(PTR_TO_CALLNO(ast_channel_tech_pvt(c)), AST_FRAME_DTMF_END, digit, 0, NULL, 0, -1);
- }
- static int iax2_sendtext(struct ast_channel *c, const char *text)
- {
-
- return send_command_locked(PTR_TO_CALLNO(ast_channel_tech_pvt(c)), AST_FRAME_TEXT,
- 0, 0, (unsigned char *)text, strlen(text) + 1, -1);
- }
- static int iax2_sendimage(struct ast_channel *c, struct ast_frame *img)
- {
- return send_command_locked(PTR_TO_CALLNO(ast_channel_tech_pvt(c)), AST_FRAME_IMAGE, img->subclass.integer, 0, img->data.ptr, img->datalen, -1);
- }
- static int iax2_sendhtml(struct ast_channel *c, int subclass, const char *data, int datalen)
- {
- return send_command_locked(PTR_TO_CALLNO(ast_channel_tech_pvt(c)), AST_FRAME_HTML, subclass, 0, (unsigned char *)data, datalen, -1);
- }
- static int iax2_fixup(struct ast_channel *oldchannel, struct ast_channel *newchan)
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(newchan));
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno])
- iaxs[callno]->owner = newchan;
- else
- ast_log(LOG_WARNING, "Uh, this isn't a good sign...\n");
- ast_mutex_unlock(&iaxsl[callno]);
- return 0;
- }
- /*!
- * \note This function calls reg_source_db -> iax2_poke_peer -> find_callno,
- * so do not call this with a pvt lock held.
- */
- static struct iax2_peer *realtime_peer(const char *peername, struct sockaddr_in *sin)
- {
- struct ast_variable *var = NULL;
- struct ast_variable *tmp;
- struct iax2_peer *peer=NULL;
- time_t regseconds = 0, nowtime;
- int dynamic=0;
- if (peername) {
- var = ast_load_realtime("iaxpeers", "name", peername, "host", "dynamic", SENTINEL);
- if (!var && sin)
- var = ast_load_realtime("iaxpeers", "name", peername, "host", ast_inet_ntoa(sin->sin_addr), SENTINEL);
- } else if (sin) {
- char porta[25];
- sprintf(porta, "%d", ntohs(sin->sin_port));
- var = ast_load_realtime("iaxpeers", "ipaddr", ast_inet_ntoa(sin->sin_addr), "port", porta, SENTINEL);
- if (var) {
- /* We'll need the peer name in order to build the structure! */
- for (tmp = var; tmp; tmp = tmp->next) {
- if (!strcasecmp(tmp->name, "name"))
- peername = tmp->value;
- }
- }
- }
- if (!var && peername) { /* Last ditch effort */
- var = ast_load_realtime("iaxpeers", "name", peername, SENTINEL);
- /*!\note
- * If this one loaded something, then we need to ensure that the host
- * field matched. The only reason why we can't have this as a criteria
- * is because we only have the IP address and the host field might be
- * set as a name (and the reverse PTR might not match).
- */
- if (var && sin) {
- for (tmp = var; tmp; tmp = tmp->next) {
- if (!strcasecmp(tmp->name, "host")) {
- struct ast_hostent ahp;
- struct hostent *hp;
- if (!(hp = ast_gethostbyname(tmp->value, &ahp)) || memcmp(hp->h_addr, &sin->sin_addr, hp->h_length)) {
- /* No match */
- ast_variables_destroy(var);
- var = NULL;
- }
- break;
- }
- }
- }
- }
- if (!var)
- return NULL;
- peer = build_peer(peername, var, NULL, ast_test_flag64((&globalflags), IAX_RTCACHEFRIENDS) ? 0 : 1);
- if (!peer) {
- ast_variables_destroy(var);
- return NULL;
- }
- for (tmp = var; tmp; tmp = tmp->next) {
- /* Make sure it's not a user only... */
- if (!strcasecmp(tmp->name, "type")) {
- if (strcasecmp(tmp->value, "friend") &&
- strcasecmp(tmp->value, "peer")) {
- /* Whoops, we weren't supposed to exist! */
- peer = peer_unref(peer);
- break;
- }
- } else if (!strcasecmp(tmp->name, "regseconds")) {
- ast_get_time_t(tmp->value, ®seconds, 0, NULL);
- } else if (!strcasecmp(tmp->name, "ipaddr")) {
- if (!ast_sockaddr_parse(&peer->addr, tmp->value, PARSE_PORT_IGNORE)) {
- ast_log(LOG_WARNING, "Failed to parse sockaddr '%s' for ipaddr of realtime peer '%s'\n", tmp->value, tmp->name);
- }
- } else if (!strcasecmp(tmp->name, "port")) {
- ast_sockaddr_set_port(&peer->addr, atoi(tmp->value));
- } else if (!strcasecmp(tmp->name, "host")) {
- if (!strcasecmp(tmp->value, "dynamic"))
- dynamic = 1;
- }
- }
- ast_variables_destroy(var);
- if (!peer)
- return NULL;
- if (ast_test_flag64((&globalflags), IAX_RTCACHEFRIENDS)) {
- ast_copy_flags64(peer, &globalflags, IAX_RTAUTOCLEAR|IAX_RTCACHEFRIENDS);
- if (ast_test_flag64(peer, IAX_RTAUTOCLEAR)) {
- if (peer->expire > -1) {
- if (!AST_SCHED_DEL(sched, peer->expire)) {
- peer->expire = -1;
- peer_unref(peer);
- }
- }
- peer->expire = iax2_sched_add(sched, (global_rtautoclear) * 1000, expire_registry, peer_ref(peer));
- if (peer->expire == -1)
- peer_unref(peer);
- }
- ao2_link(peers, peer);
- if (ast_test_flag64(peer, IAX_DYNAMIC))
- reg_source_db(peer);
- } else {
- ast_set_flag64(peer, IAX_TEMPONLY);
- }
- if (!ast_test_flag64(&globalflags, IAX_RTIGNOREREGEXPIRE) && dynamic) {
- time(&nowtime);
- if ((nowtime - regseconds) > IAX_DEFAULT_REG_EXPIRE) {
- memset(&peer->addr, 0, sizeof(peer->addr));
- realtime_update_peer(peer->name, &peer->addr, 0);
- ast_debug(1, "realtime_peer: Bah, '%s' is expired (%d/%d/%d)!\n",
- peername, (int)(nowtime - regseconds), (int)regseconds, (int)nowtime);
- }
- else {
- ast_debug(1, "realtime_peer: Registration for '%s' still active (%d/%d/%d)!\n",
- peername, (int)(nowtime - regseconds), (int)regseconds, (int)nowtime);
- }
- }
- return peer;
- }
- static struct iax2_user *realtime_user(const char *username, struct sockaddr_in *sin)
- {
- struct ast_variable *var;
- struct ast_variable *tmp;
- struct iax2_user *user=NULL;
- var = ast_load_realtime("iaxusers", "name", username, "host", "dynamic", SENTINEL);
- if (!var)
- var = ast_load_realtime("iaxusers", "name", username, "host", ast_inet_ntoa(sin->sin_addr), SENTINEL);
- if (!var && sin) {
- char porta[6];
- snprintf(porta, sizeof(porta), "%d", ntohs(sin->sin_port));
- var = ast_load_realtime("iaxusers", "name", username, "ipaddr", ast_inet_ntoa(sin->sin_addr), "port", porta, SENTINEL);
- if (!var)
- var = ast_load_realtime("iaxusers", "ipaddr", ast_inet_ntoa(sin->sin_addr), "port", porta, SENTINEL);
- }
- if (!var) { /* Last ditch effort */
- var = ast_load_realtime("iaxusers", "name", username, SENTINEL);
- /*!\note
- * If this one loaded something, then we need to ensure that the host
- * field matched. The only reason why we can't have this as a criteria
- * is because we only have the IP address and the host field might be
- * set as a name (and the reverse PTR might not match).
- */
- if (var) {
- for (tmp = var; tmp; tmp = tmp->next) {
- if (!strcasecmp(tmp->name, "host")) {
- struct ast_hostent ahp;
- struct hostent *hp;
- if (!(hp = ast_gethostbyname(tmp->value, &ahp)) || memcmp(hp->h_addr, &sin->sin_addr, hp->h_length)) {
- /* No match */
- ast_variables_destroy(var);
- var = NULL;
- }
- break;
- }
- }
- }
- }
- if (!var)
- return NULL;
- tmp = var;
- while(tmp) {
- /* Make sure it's not a peer only... */
- if (!strcasecmp(tmp->name, "type")) {
- if (strcasecmp(tmp->value, "friend") &&
- strcasecmp(tmp->value, "user")) {
- return NULL;
- }
- }
- tmp = tmp->next;
- }
- user = build_user(username, var, NULL, !ast_test_flag64((&globalflags), IAX_RTCACHEFRIENDS));
- ast_variables_destroy(var);
- if (!user)
- return NULL;
- if (ast_test_flag64((&globalflags), IAX_RTCACHEFRIENDS)) {
- ast_set_flag64(user, IAX_RTCACHEFRIENDS);
- ao2_link(users, user);
- } else {
- ast_set_flag64(user, IAX_TEMPONLY);
- }
- return user;
- }
- static void realtime_update_peer(const char *peername, struct ast_sockaddr *sockaddr, time_t regtime)
- {
- char port[10];
- char regseconds[20];
- const char *sysname = ast_config_AST_SYSTEM_NAME;
- char *syslabel = NULL;
- if (ast_strlen_zero(sysname)) /* No system name, disable this */
- sysname = NULL;
- else if (ast_test_flag64(&globalflags, IAX_RTSAVE_SYSNAME))
- syslabel = "regserver";
- snprintf(regseconds, sizeof(regseconds), "%d", (int)regtime);
- snprintf(port, sizeof(port), "%d", ast_sockaddr_port(sockaddr));
- ast_update_realtime("iaxpeers", "name", peername,
- "ipaddr", ast_sockaddr_stringify_addr(sockaddr), "port", port,
- "regseconds", regseconds, syslabel, sysname, SENTINEL); /* note syslable can be NULL */
- }
- struct create_addr_info {
- iax2_format capability;
- uint64_t flags;
- int maxtime;
- int encmethods;
- int found;
- int sockfd;
- int adsi;
- char username[80];
- char secret[80];
- char outkey[80];
- char timezone[80];
- char prefs[32];
- char cid_num[80];
- char cid_name[80];
- char context[AST_MAX_CONTEXT];
- char peercontext[AST_MAX_CONTEXT];
- char mohinterpret[MAX_MUSICCLASS];
- char mohsuggest[MAX_MUSICCLASS];
- };
- static int create_addr(const char *peername, struct ast_channel *c, struct sockaddr_in *sin, struct create_addr_info *cai)
- {
- struct iax2_peer *peer;
- int res = -1;
- struct ast_codec_pref ourprefs;
- struct sockaddr_in peer_addr;
- ast_clear_flag64(cai, IAX_SENDANI | IAX_TRUNK);
- cai->sockfd = defaultsockfd;
- cai->maxtime = 0;
- sin->sin_family = AF_INET;
- if (!(peer = find_peer(peername, 1))) {
- struct ast_sockaddr sin_tmp;
- cai->found = 0;
- sin_tmp.ss.ss_family = AF_INET;
- if (ast_get_ip_or_srv(&sin_tmp, peername, srvlookup ? "_iax._udp" : NULL)) {
- ast_log(LOG_WARNING, "No such host: %s\n", peername);
- return -1;
- }
- ast_sockaddr_to_sin(&sin_tmp, sin);
- if (sin->sin_port == 0) {
- sin->sin_port = htons(IAX_DEFAULT_PORTNO);
- }
- /* use global iax prefs for unknown peer/user */
- /* But move the calling channel's native codec to the top of the preference list */
- memcpy(&ourprefs, &prefs, sizeof(ourprefs));
- if (c) {
- struct ast_format tmpfmt;
- ast_format_cap_iter_start(ast_channel_nativeformats(c));
- while (!(ast_format_cap_iter_next(ast_channel_nativeformats(c), &tmpfmt))) {
- ast_codec_pref_prepend(&ourprefs, &tmpfmt, 1);
- }
- ast_format_cap_iter_end(ast_channel_nativeformats(c));
- }
- ast_codec_pref_convert(&ourprefs, cai->prefs, sizeof(cai->prefs), 1);
- return 0;
- }
- cai->found = 1;
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- /* if the peer has no address (current or default), return failure */
- if (!(peer_addr.sin_addr.s_addr || peer->defaddr.sin_addr.s_addr)) {
- goto return_unref;
- }
- /* if the peer is being monitored and is currently unreachable, return failure */
- if (peer->maxms && ((peer->lastms > peer->maxms) || (peer->lastms < 0)))
- goto return_unref;
- ast_copy_flags64(cai, peer, IAX_SENDANI | IAX_TRUNK | IAX_NOTRANSFER | IAX_TRANSFERMEDIA | IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE | IAX_FORCE_ENCRYPT);
- cai->maxtime = peer->maxms;
- cai->capability = peer->capability;
- cai->encmethods = peer->encmethods;
- cai->sockfd = peer->sockfd;
- cai->adsi = peer->adsi;
- memcpy(&ourprefs, &peer->prefs, sizeof(ourprefs));
- /* Move the calling channel's native codec to the top of the preference list */
- if (c) {
- struct ast_format tmpfmt;
- ast_format_cap_iter_start(ast_channel_nativeformats(c));
- while (!(ast_format_cap_iter_next(ast_channel_nativeformats(c), &tmpfmt))) {
- ast_debug(1, "prepending %s to prefs\n", ast_getformatname(&tmpfmt));
- ast_codec_pref_prepend(&ourprefs, &tmpfmt, 1);
- }
- ast_format_cap_iter_end(ast_channel_nativeformats(c));
- }
- ast_codec_pref_convert(&ourprefs, cai->prefs, sizeof(cai->prefs), 1);
- ast_copy_string(cai->context, peer->context, sizeof(cai->context));
- ast_copy_string(cai->peercontext, peer->peercontext, sizeof(cai->peercontext));
- ast_copy_string(cai->username, peer->username, sizeof(cai->username));
- ast_copy_string(cai->timezone, peer->zonetag, sizeof(cai->timezone));
- ast_copy_string(cai->outkey, peer->outkey, sizeof(cai->outkey));
- ast_copy_string(cai->cid_num, peer->cid_num, sizeof(cai->cid_num));
- ast_copy_string(cai->cid_name, peer->cid_name, sizeof(cai->cid_name));
- ast_copy_string(cai->mohinterpret, peer->mohinterpret, sizeof(cai->mohinterpret));
- ast_copy_string(cai->mohsuggest, peer->mohsuggest, sizeof(cai->mohsuggest));
- if (ast_strlen_zero(peer->dbsecret)) {
- ast_copy_string(cai->secret, peer->secret, sizeof(cai->secret));
- } else {
- char *family;
- char *key = NULL;
- family = ast_strdupa(peer->dbsecret);
- key = strchr(family, '/');
- if (key)
- *key++ = '\0';
- if (!key || ast_db_get(family, key, cai->secret, sizeof(cai->secret))) {
- ast_log(LOG_WARNING, "Unable to retrieve database password for family/key '%s'!\n", peer->dbsecret);
- goto return_unref;
- }
- }
- if (peer_addr.sin_addr.s_addr) {
- sin->sin_addr = peer_addr.sin_addr;
- sin->sin_port = peer_addr.sin_port;
- } else {
- sin->sin_addr = peer->defaddr.sin_addr;
- sin->sin_port = peer->defaddr.sin_port;
- }
- res = 0;
- return_unref:
- peer_unref(peer);
- return res;
- }
- static void __auto_congest(const void *nothing)
- {
- int callno = PTR_TO_CALLNO(nothing);
- struct ast_frame f = { AST_FRAME_CONTROL, { AST_CONTROL_CONGESTION } };
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- iaxs[callno]->initid = -1;
- iax2_queue_frame(callno, &f);
- ast_log(LOG_NOTICE, "Auto-congesting call due to slow response\n");
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int auto_congest(const void *data)
- {
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__auto_congest, data))
- #endif
- __auto_congest(data);
- return 0;
- }
- static unsigned int iax2_datetime(const char *tz)
- {
- struct timeval t = ast_tvnow();
- struct ast_tm tm;
- unsigned int tmp;
- ast_localtime(&t, &tm, ast_strlen_zero(tz) ? NULL : tz);
- tmp = (tm.tm_sec >> 1) & 0x1f; /* 5 bits of seconds */
- tmp |= (tm.tm_min & 0x3f) << 5; /* 6 bits of minutes */
- tmp |= (tm.tm_hour & 0x1f) << 11; /* 5 bits of hours */
- tmp |= (tm.tm_mday & 0x1f) << 16; /* 5 bits of day of month */
- tmp |= ((tm.tm_mon + 1) & 0xf) << 21; /* 4 bits of month */
- tmp |= ((tm.tm_year - 100) & 0x7f) << 25; /* 7 bits of year */
- return tmp;
- }
- struct parsed_dial_string {
- char *username;
- char *password;
- char *key;
- char *peer;
- char *port;
- char *exten;
- char *context;
- char *options;
- };
- static int send_apathetic_reply(unsigned short callno, unsigned short dcallno,
- struct sockaddr_in *sin, int command, int ts, unsigned char seqno,
- int sockfd, struct iax_ie_data *ied)
- {
- struct {
- struct ast_iax2_full_hdr f;
- struct iax_ie_data ied;
- } data;
- size_t size = sizeof(struct ast_iax2_full_hdr);
- if (ied) {
- size += ied->pos;
- memcpy(&data.ied, ied->buf, ied->pos);
- }
- data.f.scallno = htons(0x8000 | callno);
- data.f.dcallno = htons(dcallno & ~IAX_FLAG_RETRANS);
- data.f.ts = htonl(ts);
- data.f.iseqno = seqno;
- data.f.oseqno = 0;
- data.f.type = AST_FRAME_IAX;
- data.f.csub = compress_subclass(command);
- iax_outputframe(NULL, &data.f, 0, sin, size - sizeof(struct ast_iax2_full_hdr));
- return sendto(sockfd, &data, size, 0, (struct sockaddr *)sin, sizeof(*sin));
- }
- static void add_empty_calltoken_ie(struct chan_iax2_pvt *pvt, struct iax_ie_data *ied)
- {
- /* first make sure their are two empty bytes left in ied->buf */
- if (pvt && ied && (2 < ((int) sizeof(ied->buf) - ied->pos))) {
- ied->buf[ied->pos++] = IAX_IE_CALLTOKEN; /* type */
- ied->buf[ied->pos++] = 0; /* data size, ZERO in this case */
- pvt->calltoken_ie_len = 2;
- }
- }
- static void resend_with_token(int callno, struct iax_frame *f, const char *newtoken)
- {
- struct chan_iax2_pvt *pvt = iaxs[callno];
- int frametype = f->af.frametype;
- int subclass = f->af.subclass.integer;
- struct {
- struct ast_iax2_full_hdr fh;
- struct iax_ie_data ied;
- } data = {
- .ied.buf = { 0 },
- .ied.pos = 0,
- };
- /* total len - header len gives us the frame's IE len */
- int ie_data_pos = f->datalen - sizeof(struct ast_iax2_full_hdr);
- if (!pvt) {
- return; /* this should not be possible if called from socket_process() */
- }
- /*
- * Check to make sure last frame sent is valid for call token resend
- * 1. Frame should _NOT_ be encrypted since it starts the IAX dialog
- * 2. Frame should _NOT_ already have a destination callno
- * 3. Frame must be a valid iax_frame subclass capable of starting dialog
- * 4. Pvt must have a calltoken_ie_len which represents the number of
- * bytes at the end of the frame used for the previous calltoken ie.
- * 5. Pvt's calltoken_ie_len must be _LESS_ than the total IE length
- * 6. Total length of f->data must be _LESS_ than size of our data struct
- * because f->data must be able to fit within data.
- */
- if (f->encmethods || f->dcallno || !iax2_allow_new(frametype, subclass, 0)
- || !pvt->calltoken_ie_len || (pvt->calltoken_ie_len > ie_data_pos) ||
- (f->datalen > sizeof(data))) {
- return; /* ignore resend, token was not valid for the dialog */
- }
- /* token is valid
- * 1. Copy frame data over
- * 2. Redo calltoken IE, it will always be the last ie in the frame.
- * NOTE: Having the ie always be last is not protocol specified,
- * it is only an implementation choice. Since we only expect the ie to
- * be last for frames we have sent, this can no way be affected by
- * another end point.
- * 3. Remove frame from queue
- * 4. Free old frame
- * 5. Clear previous seqnos
- * 6. Resend with CALLTOKEN ie.
- */
- /* ---1.--- */
- memcpy(&data, f->data, f->datalen);
- data.ied.pos = ie_data_pos;
- /* ---2.--- */
- /* move to the beginning of the calltoken ie so we can write over it */
- data.ied.pos -= pvt->calltoken_ie_len;
- iax_ie_append_str(&data.ied, IAX_IE_CALLTOKEN, newtoken);
- /* make sure to update token length incase it ever has to be stripped off again */
- pvt->calltoken_ie_len = data.ied.pos - ie_data_pos; /* new pos minus old pos tells how big token ie is */
- /* ---3.--- */
- AST_LIST_REMOVE(&frame_queue[callno], f, list);
- /* ---4.--- */
- iax2_frame_free(f);
- /* ---5.--- */
- pvt->oseqno = 0;
- pvt->rseqno = 0;
- pvt->iseqno = 0;
- pvt->aseqno = 0;
- if (pvt->peercallno) {
- remove_by_peercallno(pvt);
- pvt->peercallno = 0;
- }
- /* ---6.--- */
- send_command(pvt, AST_FRAME_IAX, subclass, 0, data.ied.buf, data.ied.pos, -1);
- }
- static void requirecalltoken_mark_auto(const char *name, int subclass)
- {
- struct iax2_user *user = NULL;
- struct iax2_peer *peer = NULL;
- if (ast_strlen_zero(name)) {
- return; /* no username given */
- }
- if ((subclass == IAX_COMMAND_NEW) && (user = find_user(name)) && (user->calltoken_required == CALLTOKEN_AUTO)) {
- user->calltoken_required = CALLTOKEN_YES;
- } else if ((subclass != IAX_COMMAND_NEW) && (peer = find_peer(name, 1)) && (peer->calltoken_required == CALLTOKEN_AUTO)) {
- peer->calltoken_required = CALLTOKEN_YES;
- }
- if (peer) {
- peer_unref(peer);
- }
- if (user) {
- user_unref(user);
- }
- }
- /*!
- * \internal
- *
- * \brief handles calltoken logic for a received iax_frame.
- *
- * \note frametype must be AST_FRAME_IAX.
- *
- * \note
- * Three different cases are possible here.
- * Case 1. An empty calltoken is provided. This means the client supports
- * calltokens but has not yet received one from us. In this case
- * a full calltoken IE is created and sent in a calltoken fullframe.
- * Case 2. A full calltoken is received and must be checked for validity.
- * Case 3. No calltoken is received indicating that the client does not
- * support calltokens. In this case it is up to the configuration
- * to decide how this should be handled (reject or permit without calltoken)
- */
- static int handle_call_token(struct ast_iax2_full_hdr *fh, struct iax_ies *ies,
- struct sockaddr_in *sin, int fd)
- {
- #define CALLTOKEN_HASH_FORMAT "%s%d%u%d" /* address + port + ts + randomcalldata */
- #define CALLTOKEN_IE_FORMAT "%u?%s" /* time + ? + (40 char hash) */
- struct ast_str *buf = ast_str_alloca(256);
- time_t t = time(NULL);
- char hash[41]; /* 40 char sha1 hash */
- int subclass = uncompress_subclass(fh->csub);
- /* ----- Case 1 ----- */
- if (ies->calltoken && !ies->calltokendata) { /* empty calltoken is provided, client supports calltokens */
- struct iax_ie_data ied = {
- .buf = { 0 },
- .pos = 0,
- };
- /* create the hash with their address data and our timestamp */
- ast_str_set(&buf, 0, CALLTOKEN_HASH_FORMAT, ast_inet_ntoa(sin->sin_addr), sin->sin_port, (unsigned int) t, randomcalltokendata);
- ast_sha1_hash(hash, ast_str_buffer(buf));
- ast_str_set(&buf, 0, CALLTOKEN_IE_FORMAT, (unsigned int) t, hash);
- iax_ie_append_str(&ied, IAX_IE_CALLTOKEN, ast_str_buffer(buf));
- send_apathetic_reply(1, ntohs(fh->scallno), sin, IAX_COMMAND_CALLTOKEN, ntohl(fh->ts), fh->iseqno + 1, fd, &ied);
- return 1;
- /* ----- Case 2 ----- */
- } else if (ies->calltoken && ies->calltokendata) { /* calltoken received, check to see if it is valid */
- char *rec_hash = NULL; /* the received hash, make sure it matches with ours. */
- char *rec_ts = NULL; /* received timestamp */
- unsigned int rec_time; /* received time_t */
- /* split the timestamp from the hash data */
- rec_hash = strchr((char *) ies->calltokendata, '?');
- if (rec_hash) {
- *rec_hash++ = '\0';
- rec_ts = (char *) ies->calltokendata;
- }
- /* check that we have valid data before we do any comparisons */
- if (!rec_hash || !rec_ts) {
- goto reject;
- } else if (sscanf(rec_ts, "%u", &rec_time) != 1) {
- goto reject;
- }
- /* create a hash with their address and the _TOKEN'S_ timestamp */
- ast_str_set(&buf, 0, CALLTOKEN_HASH_FORMAT, ast_inet_ntoa(sin->sin_addr), sin->sin_port, (unsigned int) rec_time, randomcalltokendata);
- ast_sha1_hash(hash, ast_str_buffer(buf));
- /* compare hashes and then check timestamp delay */
- if (strcmp(hash, rec_hash)) {
- ast_log(LOG_WARNING, "Address %s failed CallToken hash inspection\n", ast_inet_ntoa(sin->sin_addr));
- goto reject; /* received hash does not match ours, reject */
- } else if ((t < rec_time) || ((t - rec_time) >= MAX_CALLTOKEN_DELAY)) {
- ast_log(LOG_WARNING, "Too much delay in IAX2 calltoken timestamp from address %s\n", ast_inet_ntoa(sin->sin_addr));
- goto reject; /* too much delay, reject */
- }
- /* at this point the call token is valid, returning 0
- * will allow socket_process to continue as usual */
- requirecalltoken_mark_auto(ies->username, subclass);
- return 0;
- /* ----- Case 3 ----- */
- } else { /* calltokens are not supported for this client, how do we respond? */
- if (calltoken_required(sin, ies->username, subclass)) {
- ast_log(LOG_ERROR, "Call rejected, CallToken Support required. If unexpected, resolve by placing address %s in the calltokenoptional list or setting user %s requirecalltoken=no\n", ast_inet_ntoa(sin->sin_addr), S_OR(ies->username, "guest"));
- goto reject;
- }
- return 0; /* calltoken is not required for this addr, so permit it. */
- }
- reject:
- /* received frame has failed calltoken inspection, send apathetic reject messages */
- if (subclass == IAX_COMMAND_REGREQ || subclass == IAX_COMMAND_REGREL) {
- send_apathetic_reply(1, ntohs(fh->scallno), sin, IAX_COMMAND_REGREJ, ntohl(fh->ts), fh->iseqno + 1, fd, NULL);
- } else {
- send_apathetic_reply(1, ntohs(fh->scallno), sin, IAX_COMMAND_REJECT, ntohl(fh->ts), fh->iseqno + 1, fd, NULL);
- }
- return 1;
- }
- /*!
- * \brief Parses an IAX dial string into its component parts.
- * \param data the string to be parsed
- * \param pds pointer to a \c struct \c parsed_dial_string to be filled in
- * \return nothing
- *
- * This function parses the string and fills the structure
- * with pointers to its component parts. The input string
- * will be modified.
- *
- * \note This function supports both plaintext passwords and RSA
- * key names; if the password string is formatted as '[keyname]',
- * then the keyname will be placed into the key field, and the
- * password field will be set to NULL.
- *
- * \note The dial string format is:
- * [username[:password]@]peer[:port][/exten[@@context]][/options]
- */
- static void parse_dial_string(char *data, struct parsed_dial_string *pds)
- {
- if (ast_strlen_zero(data))
- return;
- pds->peer = strsep(&data, "/");
- pds->exten = strsep(&data, "/");
- pds->options = data;
- if (pds->exten) {
- data = pds->exten;
- pds->exten = strsep(&data, "@");
- pds->context = data;
- }
- if (strchr(pds->peer, '@')) {
- data = pds->peer;
- pds->username = strsep(&data, "@");
- pds->peer = data;
- }
- if (pds->username) {
- data = pds->username;
- pds->username = strsep(&data, ":");
- pds->password = data;
- }
- data = pds->peer;
- pds->peer = strsep(&data, ":");
- pds->port = data;
- /* check for a key name wrapped in [] in the secret position, if found,
- move it to the key field instead
- */
- if (pds->password && (pds->password[0] == '[')) {
- pds->key = ast_strip_quoted(pds->password, "[", "]");
- pds->password = NULL;
- }
- }
- static int iax2_call(struct ast_channel *c, const char *dest, int timeout)
- {
- struct sockaddr_in sin;
- char *l=NULL, *n=NULL, *tmpstr;
- struct iax_ie_data ied;
- char *defaultrdest = "s";
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- struct parsed_dial_string pds;
- struct create_addr_info cai;
- struct ast_var_t *var;
- struct ast_datastore *variablestore = ast_channel_datastore_find(c, &iax2_variable_datastore_info, NULL);
- const char* osp_token_ptr;
- unsigned int osp_token_length;
- unsigned char osp_block_index;
- unsigned int osp_block_length;
- unsigned char osp_buffer[256];
- iax2_format iax2_tmpfmt;
- if ((ast_channel_state(c) != AST_STATE_DOWN) && (ast_channel_state(c) != AST_STATE_RESERVED)) {
- ast_log(LOG_WARNING, "Channel is already in use (%s)?\n", ast_channel_name(c));
- return -1;
- }
- memset(&cai, 0, sizeof(cai));
- cai.encmethods = iax2_encryption;
- memset(&pds, 0, sizeof(pds));
- tmpstr = ast_strdupa(dest);
- parse_dial_string(tmpstr, &pds);
- if (ast_strlen_zero(pds.peer)) {
- ast_log(LOG_WARNING, "No peer provided in the IAX2 dial string '%s'\n", dest);
- return -1;
- }
- if (!pds.exten) {
- pds.exten = defaultrdest;
- }
- if (create_addr(pds.peer, c, &sin, &cai)) {
- ast_log(LOG_WARNING, "No address associated with '%s'\n", pds.peer);
- return -1;
- }
- if (ast_test_flag64(iaxs[callno], IAX_FORCE_ENCRYPT) && !cai.encmethods) {
- ast_log(LOG_WARNING, "Encryption forced for call, but not enabled\n");
- ast_channel_hangupcause_set(c, AST_CAUSE_BEARERCAPABILITY_NOTAVAIL);
- return -1;
- }
- if (ast_strlen_zero(cai.secret) && ast_test_flag64(iaxs[callno], IAX_FORCE_ENCRYPT)) {
- ast_log(LOG_WARNING, "Call terminated. No secret given and force encrypt enabled\n");
- return -1;
- }
- if (!pds.username && !ast_strlen_zero(cai.username))
- pds.username = cai.username;
- if (!pds.password && !ast_strlen_zero(cai.secret))
- pds.password = cai.secret;
- if (!pds.key && !ast_strlen_zero(cai.outkey))
- pds.key = cai.outkey;
- if (!pds.context && !ast_strlen_zero(cai.peercontext))
- pds.context = cai.peercontext;
- /* Keep track of the context for outgoing calls too */
- ast_channel_context_set(c, cai.context);
- if (pds.port)
- sin.sin_port = htons(atoi(pds.port));
- l = ast_channel_connected(c)->id.number.valid ? ast_channel_connected(c)->id.number.str : NULL;
- n = ast_channel_connected(c)->id.name.valid ? ast_channel_connected(c)->id.name.str : NULL;
- /* Now build request */
- memset(&ied, 0, sizeof(ied));
- /* On new call, first IE MUST be IAX version of caller */
- iax_ie_append_short(&ied, IAX_IE_VERSION, IAX_PROTO_VERSION);
- iax_ie_append_str(&ied, IAX_IE_CALLED_NUMBER, pds.exten);
- if (pds.options && strchr(pds.options, 'a')) {
- /* Request auto answer */
- iax_ie_append(&ied, IAX_IE_AUTOANSWER);
- }
- /* WARNING: this breaks down at 190 bits! */
- iax_ie_append_str(&ied, IAX_IE_CODEC_PREFS, cai.prefs);
- if (l) {
- iax_ie_append_str(&ied, IAX_IE_CALLING_NUMBER, l);
- iax_ie_append_byte(&ied, IAX_IE_CALLINGPRES,
- ast_party_id_presentation(&ast_channel_connected(c)->id));
- } else if (n) {
- iax_ie_append_byte(&ied, IAX_IE_CALLINGPRES,
- ast_party_id_presentation(&ast_channel_connected(c)->id));
- } else {
- iax_ie_append_byte(&ied, IAX_IE_CALLINGPRES, AST_PRES_NUMBER_NOT_AVAILABLE);
- }
- iax_ie_append_byte(&ied, IAX_IE_CALLINGTON, ast_channel_connected(c)->id.number.plan);
- iax_ie_append_short(&ied, IAX_IE_CALLINGTNS, ast_channel_dialed(c)->transit_network_select);
- if (n)
- iax_ie_append_str(&ied, IAX_IE_CALLING_NAME, n);
- if (ast_test_flag64(iaxs[callno], IAX_SENDANI)
- && ast_channel_connected(c)->ani.number.valid
- && ast_channel_connected(c)->ani.number.str) {
- iax_ie_append_str(&ied, IAX_IE_CALLING_ANI, ast_channel_connected(c)->ani.number.str);
- }
- if (!ast_strlen_zero(ast_channel_language(c)))
- iax_ie_append_str(&ied, IAX_IE_LANGUAGE, ast_channel_language(c));
- if (!ast_strlen_zero(ast_channel_dialed(c)->number.str)) {
- iax_ie_append_str(&ied, IAX_IE_DNID, ast_channel_dialed(c)->number.str);
- }
- if (ast_channel_redirecting(c)->from.number.valid
- && !ast_strlen_zero(ast_channel_redirecting(c)->from.number.str)) {
- iax_ie_append_str(&ied, IAX_IE_RDNIS, ast_channel_redirecting(c)->from.number.str);
- }
- if (pds.context)
- iax_ie_append_str(&ied, IAX_IE_CALLED_CONTEXT, pds.context);
- if (pds.username)
- iax_ie_append_str(&ied, IAX_IE_USERNAME, pds.username);
- if (cai.encmethods)
- iax_ie_append_short(&ied, IAX_IE_ENCRYPTION, cai.encmethods);
- ast_mutex_lock(&iaxsl[callno]);
- if (!ast_strlen_zero(ast_channel_context(c)))
- ast_string_field_set(iaxs[callno], context, ast_channel_context(c));
- if (pds.username)
- ast_string_field_set(iaxs[callno], username, pds.username);
- iaxs[callno]->encmethods = cai.encmethods;
- iaxs[callno]->adsi = cai.adsi;
-
- ast_string_field_set(iaxs[callno], mohinterpret, cai.mohinterpret);
- ast_string_field_set(iaxs[callno], mohsuggest, cai.mohsuggest);
- if (pds.key)
- ast_string_field_set(iaxs[callno], outkey, pds.key);
- if (pds.password)
- ast_string_field_set(iaxs[callno], secret, pds.password);
- iax2_tmpfmt = ast_format_cap_to_old_bitfield(ast_channel_nativeformats(c));
- iax_ie_append_int(&ied, IAX_IE_FORMAT, (int) iax2_tmpfmt);
- iax_ie_append_versioned_uint64(&ied, IAX_IE_FORMAT2, 0, iax2_tmpfmt);
- iax_ie_append_int(&ied, IAX_IE_CAPABILITY, (int) iaxs[callno]->capability);
- iax_ie_append_versioned_uint64(&ied, IAX_IE_CAPABILITY2, 0, iaxs[callno]->capability);
- iax_ie_append_short(&ied, IAX_IE_ADSICPE, ast_channel_adsicpe(c));
- iax_ie_append_int(&ied, IAX_IE_DATETIME, iax2_datetime(cai.timezone));
- if (iaxs[callno]->maxtime) {
- /* Initialize pingtime and auto-congest time */
- iaxs[callno]->pingtime = iaxs[callno]->maxtime / 2;
- iaxs[callno]->initid = iax2_sched_add(sched, iaxs[callno]->maxtime * 2, auto_congest, CALLNO_TO_PTR(callno));
- } else if (autokill) {
- iaxs[callno]->pingtime = autokill / 2;
- iaxs[callno]->initid = iax2_sched_add(sched, autokill * 2, auto_congest, CALLNO_TO_PTR(callno));
- }
- /* Check if there is an OSP token */
- osp_token_ptr = pbx_builtin_getvar_helper(c, "IAX2OSPTOKEN");
- if (!ast_strlen_zero(osp_token_ptr)) {
- if ((osp_token_length = strlen(osp_token_ptr)) <= IAX_MAX_OSPTOKEN_SIZE) {
- osp_block_index = 0;
- while (osp_token_length > 0) {
- osp_block_length = IAX_MAX_OSPBLOCK_SIZE < osp_token_length ? IAX_MAX_OSPBLOCK_SIZE : osp_token_length;
- osp_buffer[0] = osp_block_index;
- memcpy(osp_buffer + 1, osp_token_ptr, osp_block_length);
- iax_ie_append_raw(&ied, IAX_IE_OSPTOKEN, osp_buffer, osp_block_length + 1);
- osp_block_index++;
- osp_token_ptr += osp_block_length;
- osp_token_length -= osp_block_length;
- }
- } else
- ast_log(LOG_WARNING, "OSP token is too long\n");
- } else if (iaxdebug)
- ast_debug(1, "OSP token is undefined\n");
- /* send the command using the appropriate socket for this peer */
- iaxs[callno]->sockfd = cai.sockfd;
- /* Add remote vars */
- if (variablestore) {
- AST_LIST_HEAD(, ast_var_t) *variablelist = variablestore->data;
- ast_debug(1, "Found an IAX variable store on this channel\n");
- AST_LIST_LOCK(variablelist);
- AST_LIST_TRAVERSE(variablelist, var, entries) {
- char tmp[256];
- int i;
- ast_debug(1, "Found IAXVAR '%s' with value '%s' (to transmit)\n", ast_var_name(var), ast_var_value(var));
- /* Automatically divide the value up into sized chunks */
- for (i = 0; i < strlen(ast_var_value(var)); i += 255 - (strlen(ast_var_name(var)) + 1)) {
- snprintf(tmp, sizeof(tmp), "%s=%s", ast_var_name(var), ast_var_value(var) + i);
- iax_ie_append_str(&ied, IAX_IE_VARIABLE, tmp);
- }
- }
- AST_LIST_UNLOCK(variablelist);
- }
- /* Transmit the string in a "NEW" request */
- add_empty_calltoken_ie(iaxs[callno], &ied); /* this _MUST_ be the last ie added */
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_NEW, 0, ied.buf, ied.pos, -1);
- ast_mutex_unlock(&iaxsl[callno]);
- ast_setstate(c, AST_STATE_RINGING);
- return 0;
- }
- static int iax2_hangup(struct ast_channel *c)
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- struct iax_ie_data ied;
- int alreadygone;
- memset(&ied, 0, sizeof(ied));
- ast_mutex_lock(&iaxsl[callno]);
- if (callno && iaxs[callno]) {
- ast_debug(1, "We're hanging up %s now...\n", ast_channel_name(c));
- alreadygone = ast_test_flag64(iaxs[callno], IAX_ALREADYGONE);
- /* Send the hangup unless we have had a transmission error or are already gone */
- iax_ie_append_byte(&ied, IAX_IE_CAUSECODE, (unsigned char)ast_channel_hangupcause(c));
- if (!iaxs[callno]->error && !alreadygone) {
- if (send_command_final(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_HANGUP, 0, ied.buf, ied.pos, -1)) {
- ast_log(LOG_WARNING, "No final packet could be sent for callno %d\n", callno);
- }
- if (!iaxs[callno]) {
- ast_mutex_unlock(&iaxsl[callno]);
- return 0;
- }
- }
- /* Explicitly predestroy it */
- iax2_predestroy(callno);
- /* If we were already gone to begin with, destroy us now */
- if (iaxs[callno] && alreadygone) {
- ast_debug(1, "Really destroying %s now...\n", ast_channel_name(c));
- iax2_destroy(callno);
- } else if (iaxs[callno]) {
- if (ast_sched_add(sched, 10000, scheduled_destroy, CALLNO_TO_PTR(callno)) < 0) {
- ast_log(LOG_ERROR, "Unable to schedule iax2 callno %d destruction?!! Destroying immediately.\n", callno);
- iax2_destroy(callno);
- }
- }
- } else if (ast_channel_tech_pvt(c)) {
- /* If this call no longer exists, but the channel still
- * references it we need to set the channel's tech_pvt to null
- * to avoid ast_channel_free() trying to free it.
- */
- ast_channel_tech_pvt_set(c, NULL);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- ast_verb(3, "Hungup '%s'\n", ast_channel_name(c));
- return 0;
- }
- /*!
- * \note expects the pvt to be locked
- */
- static int wait_for_peercallno(struct chan_iax2_pvt *pvt)
- {
- unsigned short callno = pvt->callno;
- if (!pvt->peercallno) {
- /* We don't know the remote side's call number, yet. :( */
- int count = 10;
- while (count-- && pvt && !pvt->peercallno) {
- DEADLOCK_AVOIDANCE(&iaxsl[callno]);
- pvt = iaxs[callno];
- }
- if (!pvt || !pvt->peercallno) {
- return -1;
- }
- }
- return 0;
- }
- static int iax2_setoption(struct ast_channel *c, int option, void *data, int datalen)
- {
- struct ast_option_header *h;
- int res;
- switch (option) {
- case AST_OPTION_TXGAIN:
- case AST_OPTION_RXGAIN:
- /* these two cannot be sent, because they require a result */
- errno = ENOSYS;
- return -1;
- case AST_OPTION_OPRMODE:
- errno = EINVAL;
- return -1;
- case AST_OPTION_SECURE_SIGNALING:
- case AST_OPTION_SECURE_MEDIA:
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- ast_mutex_lock(&iaxsl[callno]);
- if ((*(int *) data)) {
- ast_set_flag64(iaxs[callno], IAX_FORCE_ENCRYPT);
- } else {
- ast_clear_flag64(iaxs[callno], IAX_FORCE_ENCRYPT);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- return 0;
- }
- /* These options are sent to the other side across the network where
- * they will be passed to whatever channel is bridged there. Don't
- * do anything silly like pass an option that transmits pointers to
- * memory on this machine to a remote machine to use */
- case AST_OPTION_TONE_VERIFY:
- case AST_OPTION_TDD:
- case AST_OPTION_RELAXDTMF:
- case AST_OPTION_AUDIO_MODE:
- case AST_OPTION_DIGIT_DETECT:
- case AST_OPTION_FAX_DETECT:
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- struct chan_iax2_pvt *pvt;
- ast_mutex_lock(&iaxsl[callno]);
- pvt = iaxs[callno];
- if (wait_for_peercallno(pvt)) {
- ast_mutex_unlock(&iaxsl[callno]);
- return -1;
- }
- ast_mutex_unlock(&iaxsl[callno]);
- if (!(h = ast_malloc(datalen + sizeof(*h)))) {
- return -1;
- }
- h->flag = AST_OPTION_FLAG_REQUEST;
- h->option = htons(option);
- memcpy(h->data, data, datalen);
- res = send_command_locked(PTR_TO_CALLNO(ast_channel_tech_pvt(c)), AST_FRAME_CONTROL,
- AST_CONTROL_OPTION, 0, (unsigned char *) h,
- datalen + sizeof(*h), -1);
- ast_free(h);
- return res;
- }
- default:
- return -1;
- }
- /* Just in case someone does a break instead of a return */
- return -1;
- }
- static int iax2_queryoption(struct ast_channel *c, int option, void *data, int *datalen)
- {
- switch (option) {
- case AST_OPTION_SECURE_SIGNALING:
- case AST_OPTION_SECURE_MEDIA:
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- ast_mutex_lock(&iaxsl[callno]);
- *((int *) data) = ast_test_flag64(iaxs[callno], IAX_FORCE_ENCRYPT) ? 1 : 0;
- ast_mutex_unlock(&iaxsl[callno]);
- return 0;
- }
- default:
- return -1;
- }
- }
- static struct ast_frame *iax2_read(struct ast_channel *c)
- {
- ast_debug(1, "I should never be called!\n");
- return &ast_null_frame;
- }
- static int iax2_key_rotate(const void *vpvt)
- {
- int res = 0;
- struct chan_iax2_pvt *pvt = (void *) vpvt;
- struct MD5Context md5;
- char key[17] = "";
- struct iax_ie_data ied = {
- .pos = 0,
- };
- ast_mutex_lock(&iaxsl[pvt->callno]);
- pvt->keyrotateid = ast_sched_add(sched, 120000 + (ast_random() % 180001), iax2_key_rotate, vpvt);
- snprintf(key, sizeof(key), "%lX", (unsigned long)ast_random());
- MD5Init(&md5);
- MD5Update(&md5, (unsigned char *) key, strlen(key));
- MD5Final((unsigned char *) key, &md5);
- IAX_DEBUGDIGEST("Sending", key);
- iax_ie_append_raw(&ied, IAX_IE_CHALLENGE, key, 16);
- res = send_command(pvt, AST_FRAME_IAX, IAX_COMMAND_RTKEY, 0, ied.buf, ied.pos, -1);
- build_ecx_key((unsigned char *) key, pvt);
- ast_mutex_unlock(&iaxsl[pvt->callno]);
- return res;
- }
- static int iax2_start_transfer(unsigned short callno0, unsigned short callno1, int mediaonly)
- {
- int res;
- struct iax_ie_data ied0;
- struct iax_ie_data ied1;
- unsigned int transferid = (unsigned int)ast_random();
- if (IAX_CALLENCRYPTED(iaxs[callno0]) || IAX_CALLENCRYPTED(iaxs[callno1])) {
- ast_debug(1, "transfers are not supported for encrypted calls at this time\n");
- ast_set_flag64(iaxs[callno0], IAX_NOTRANSFER);
- ast_set_flag64(iaxs[callno1], IAX_NOTRANSFER);
- return 0;
- }
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_addr(&ied0, IAX_IE_APPARENT_ADDR, &iaxs[callno1]->addr);
- iax_ie_append_short(&ied0, IAX_IE_CALLNO, iaxs[callno1]->peercallno);
- iax_ie_append_int(&ied0, IAX_IE_TRANSFERID, transferid);
- memset(&ied1, 0, sizeof(ied1));
- iax_ie_append_addr(&ied1, IAX_IE_APPARENT_ADDR, &iaxs[callno0]->addr);
- iax_ie_append_short(&ied1, IAX_IE_CALLNO, iaxs[callno0]->peercallno);
- iax_ie_append_int(&ied1, IAX_IE_TRANSFERID, transferid);
- res = send_command(iaxs[callno0], AST_FRAME_IAX, IAX_COMMAND_TXREQ, 0, ied0.buf, ied0.pos, -1);
- if (res)
- return -1;
- res = send_command(iaxs[callno1], AST_FRAME_IAX, IAX_COMMAND_TXREQ, 0, ied1.buf, ied1.pos, -1);
- if (res)
- return -1;
- iaxs[callno0]->transferring = mediaonly ? TRANSFER_MBEGIN : TRANSFER_BEGIN;
- iaxs[callno1]->transferring = mediaonly ? TRANSFER_MBEGIN : TRANSFER_BEGIN;
- return 0;
- }
- static void lock_both(unsigned short callno0, unsigned short callno1)
- {
- ast_mutex_lock(&iaxsl[callno0]);
- while (ast_mutex_trylock(&iaxsl[callno1])) {
- DEADLOCK_AVOIDANCE(&iaxsl[callno0]);
- }
- }
- static void unlock_both(unsigned short callno0, unsigned short callno1)
- {
- ast_mutex_unlock(&iaxsl[callno1]);
- ast_mutex_unlock(&iaxsl[callno0]);
- }
- static enum ast_bridge_result iax2_bridge(struct ast_channel *c0, struct ast_channel *c1, int flags, struct ast_frame **fo, struct ast_channel **rc, int timeoutms)
- {
- struct ast_channel *cs[3];
- struct ast_channel *who, *other;
- int to = -1;
- int res = -1;
- int transferstarted=0;
- struct ast_frame *f;
- unsigned short callno0 = PTR_TO_CALLNO(ast_channel_tech_pvt(c0));
- unsigned short callno1 = PTR_TO_CALLNO(ast_channel_tech_pvt(c1));
- struct timeval waittimer = {0, 0};
- /* We currently do not support native bridging if a timeoutms value has been provided */
- if (timeoutms > 0) {
- return AST_BRIDGE_FAILED;
- }
- timeoutms = -1;
- lock_both(callno0, callno1);
- if (!iaxs[callno0] || !iaxs[callno1]) {
- unlock_both(callno0, callno1);
- return AST_BRIDGE_FAILED;
- }
- /* Put them in native bridge mode */
- if (!(flags & (AST_BRIDGE_DTMF_CHANNEL_0 | AST_BRIDGE_DTMF_CHANNEL_1))) {
- iaxs[callno0]->bridgecallno = callno1;
- iaxs[callno1]->bridgecallno = callno0;
- }
- unlock_both(callno0, callno1);
- /* If not, try to bridge until we can execute a transfer, if we can */
- cs[0] = c0;
- cs[1] = c1;
- for (/* ever */;;) {
- /* Check in case we got masqueraded into */
- if ((ast_channel_tech(c0) != &iax2_tech) || (ast_channel_tech(c1) != &iax2_tech)) {
- ast_verb(3, "Can't masquerade, we're different...\n");
- /* Remove from native mode */
- if (ast_channel_tech(c0) == &iax2_tech) {
- ast_mutex_lock(&iaxsl[callno0]);
- iaxs[callno0]->bridgecallno = 0;
- ast_mutex_unlock(&iaxsl[callno0]);
- }
- if (ast_channel_tech(c1) == &iax2_tech) {
- ast_mutex_lock(&iaxsl[callno1]);
- iaxs[callno1]->bridgecallno = 0;
- ast_mutex_unlock(&iaxsl[callno1]);
- }
- return AST_BRIDGE_FAILED_NOWARN;
- }
- if (!(ast_format_cap_identical(ast_channel_nativeformats(c0), ast_channel_nativeformats(c1)))) {
- char buf0[256];
- char buf1[256];
- ast_getformatname_multiple(buf0, sizeof(buf0), ast_channel_nativeformats(c0));
- ast_getformatname_multiple(buf1, sizeof(buf1), ast_channel_nativeformats(c1));
- ast_verb(3, "Operating with different codecs [%s] [%s] , can't native bridge...\n", buf0, buf1);
- /* Remove from native mode */
- lock_both(callno0, callno1);
- if (iaxs[callno0])
- iaxs[callno0]->bridgecallno = 0;
- if (iaxs[callno1])
- iaxs[callno1]->bridgecallno = 0;
- unlock_both(callno0, callno1);
- return AST_BRIDGE_FAILED_NOWARN;
- }
- /* check if transferred and if we really want native bridging */
- if (!transferstarted && !ast_test_flag64(iaxs[callno0], IAX_NOTRANSFER) && !ast_test_flag64(iaxs[callno1], IAX_NOTRANSFER)) {
- /* Try the transfer */
- if (iax2_start_transfer(callno0, callno1, (flags & (AST_BRIDGE_DTMF_CHANNEL_0 | AST_BRIDGE_DTMF_CHANNEL_1)) ||
- ast_test_flag64(iaxs[callno0], IAX_TRANSFERMEDIA) | ast_test_flag64(iaxs[callno1], IAX_TRANSFERMEDIA)))
- ast_log(LOG_WARNING, "Unable to start the transfer\n");
- transferstarted = 1;
- }
- if ((iaxs[callno0]->transferring == TRANSFER_RELEASED) && (iaxs[callno1]->transferring == TRANSFER_RELEASED)) {
- /* Call has been transferred. We're no longer involved */
- struct timeval now = ast_tvnow();
- if (ast_tvzero(waittimer)) {
- waittimer = now;
- } else if (now.tv_sec - waittimer.tv_sec > IAX_LINGER_TIMEOUT) {
- ast_channel_softhangup_internal_flag_add(c0, AST_SOFTHANGUP_DEV);
- ast_channel_softhangup_internal_flag_add(c1, AST_SOFTHANGUP_DEV);
- *fo = NULL;
- *rc = c0;
- res = AST_BRIDGE_COMPLETE;
- break;
- }
- }
- to = 1000;
- who = ast_waitfor_n(cs, 2, &to);
- /* XXX This will need to be updated to calculate
- * timeout correctly once timeoutms is allowed to be
- * > 0. Right now, this can go badly if the waitfor
- * times out in less than a millisecond
- */
- if (timeoutms > -1) {
- timeoutms -= (1000 - to);
- if (timeoutms < 0)
- timeoutms = 0;
- }
- if (!who) {
- if (!timeoutms) {
- res = AST_BRIDGE_RETRY;
- break;
- }
- if (ast_check_hangup(c0) || ast_check_hangup(c1)) {
- res = AST_BRIDGE_FAILED;
- break;
- }
- continue;
- }
- f = ast_read(who);
- if (!f) {
- *fo = NULL;
- *rc = who;
- res = AST_BRIDGE_COMPLETE;
- break;
- }
- other = (who == c0) ? c1 : c0; /* the 'other' channel */
- if (f->frametype == AST_FRAME_CONTROL && !(flags & AST_BRIDGE_IGNORE_SIGS)) {
- switch (f->subclass.integer) {
- case AST_CONTROL_VIDUPDATE:
- case AST_CONTROL_SRCUPDATE:
- case AST_CONTROL_SRCCHANGE:
- case AST_CONTROL_T38_PARAMETERS:
- ast_write(other, f);
- break;
- case AST_CONTROL_PVT_CAUSE_CODE:
- ast_channel_hangupcause_hash_set(other, f->data.ptr, f->datalen);
- break;
- default:
- *fo = f;
- *rc = who;
- res = AST_BRIDGE_COMPLETE;
- break;
- }
- if (res == AST_BRIDGE_COMPLETE) {
- break;
- }
- } else if (f->frametype == AST_FRAME_VOICE
- || f->frametype == AST_FRAME_TEXT
- || f->frametype == AST_FRAME_VIDEO
- || f->frametype == AST_FRAME_IMAGE) {
- ast_write(other, f);
- } else if (f->frametype == AST_FRAME_DTMF) {
- /* monitored dtmf take out of the bridge.
- * check if we monitor the specific source.
- */
- int monitored_source = (who == c0) ? AST_BRIDGE_DTMF_CHANNEL_0 : AST_BRIDGE_DTMF_CHANNEL_1;
- if (flags & monitored_source) {
- *rc = who;
- *fo = f;
- res = AST_BRIDGE_COMPLETE;
- /* Remove from native mode */
- break;
- }
- ast_write(other, f);
- }
- ast_frfree(f);
- /* Swap who gets priority */
- cs[2] = cs[0];
- cs[0] = cs[1];
- cs[1] = cs[2];
- }
- lock_both(callno0, callno1);
- if(iaxs[callno0])
- iaxs[callno0]->bridgecallno = 0;
- if(iaxs[callno1])
- iaxs[callno1]->bridgecallno = 0;
- unlock_both(callno0, callno1);
- return res;
- }
- static int iax2_answer(struct ast_channel *c)
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- ast_debug(1, "Answering IAX2 call\n");
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno])
- iax2_ami_channelupdate(iaxs[callno]);
- ast_mutex_unlock(&iaxsl[callno]);
- return send_command_locked(callno, AST_FRAME_CONTROL, AST_CONTROL_ANSWER, 0, NULL, 0, -1);
- }
- static int iax2_indicate(struct ast_channel *c, int condition, const void *data, size_t datalen)
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- struct chan_iax2_pvt *pvt;
- int res = 0;
- if (iaxdebug)
- ast_debug(1, "Indicating condition %d\n", condition);
- ast_mutex_lock(&iaxsl[callno]);
- pvt = iaxs[callno];
- if (wait_for_peercallno(pvt)) {
- res = -1;
- goto done;
- }
- switch (condition) {
- case AST_CONTROL_HOLD:
- if (strcasecmp(pvt->mohinterpret, "passthrough")) {
- ast_moh_start(c, data, pvt->mohinterpret);
- goto done;
- }
- break;
- case AST_CONTROL_UNHOLD:
- if (strcasecmp(pvt->mohinterpret, "passthrough")) {
- ast_moh_stop(c);
- goto done;
- }
- break;
- case AST_CONTROL_CONNECTED_LINE:
- case AST_CONTROL_REDIRECTING:
- if (!ast_test_flag64(pvt, IAX_SENDCONNECTEDLINE)) {
- /* We are not configured to allow sending these updates. */
- ast_debug(2, "Callno %d: Config blocked sending control frame %d.\n",
- callno, condition);
- goto done;
- }
- break;
- case AST_CONTROL_PVT_CAUSE_CODE:
- res = -1;
- goto done;
- }
- res = send_command(pvt, AST_FRAME_CONTROL, condition, 0, data, datalen, -1);
- done:
- ast_mutex_unlock(&iaxsl[callno]);
- return res;
- }
- static int iax2_transfer(struct ast_channel *c, const char *dest)
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- struct iax_ie_data ied = { "", };
- char tmp[256], *context;
- enum ast_control_transfer message = AST_TRANSFER_SUCCESS;
- ast_copy_string(tmp, dest, sizeof(tmp));
- context = strchr(tmp, '@');
- if (context) {
- *context = '\0';
- context++;
- }
- iax_ie_append_str(&ied, IAX_IE_CALLED_NUMBER, tmp);
- if (context)
- iax_ie_append_str(&ied, IAX_IE_CALLED_CONTEXT, context);
- ast_debug(1, "Transferring '%s' to '%s'\n", ast_channel_name(c), dest);
- ast_queue_control_data(c, AST_CONTROL_TRANSFER, &message, sizeof(message));
- return send_command_locked(callno, AST_FRAME_IAX, IAX_COMMAND_TRANSFER, 0, ied.buf, ied.pos, -1);
- }
- static int iax2_getpeertrunk(struct sockaddr_in sin)
- {
- struct iax2_peer *peer;
- int res = 0;
- struct ao2_iterator i;
- i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- struct sockaddr_in peer_addr;
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- if ((peer_addr.sin_addr.s_addr == sin.sin_addr.s_addr) &&
- (peer_addr.sin_port == sin.sin_port)) {
- res = ast_test_flag64(peer, IAX_TRUNK);
- peer_unref(peer);
- break;
- }
- peer_unref(peer);
- }
- ao2_iterator_destroy(&i);
- return res;
- }
- /*! \brief Create new call, interface with the PBX core */
- static struct ast_channel *ast_iax2_new(int callno, int state, iax2_format capability, const char *linkedid, unsigned int cachable)
- {
- struct ast_channel *tmp;
- struct chan_iax2_pvt *i;
- struct ast_variable *v = NULL;
- struct ast_format tmpfmt;
- struct ast_callid *callid;
- if (!(i = iaxs[callno])) {
- ast_log(LOG_WARNING, "No IAX2 pvt found for callno '%d' !\n", callno);
- return NULL;
- }
- /* Don't hold call lock */
- ast_mutex_unlock(&iaxsl[callno]);
- tmp = ast_channel_alloc(1, state, i->cid_num, i->cid_name, i->accountcode, i->exten, i->context, linkedid, i->amaflags, "IAX2/%s-%d", i->host, i->callno);
- ast_mutex_lock(&iaxsl[callno]);
- if (i != iaxs[callno]) {
- if (tmp) {
- /* unlock and relock iaxsl[callno] to preserve locking order */
- ast_mutex_unlock(&iaxsl[callno]);
- tmp = ast_channel_release(tmp);
- ast_mutex_lock(&iaxsl[callno]);
- }
- return NULL;
- }
- iax2_ami_channelupdate(i);
- if (!tmp) {
- return NULL;
- }
- if ((callid = iaxs[callno]->callid)) {
- ast_channel_callid_set(tmp, callid);
- }
- ast_channel_tech_set(tmp, &iax2_tech);
- /* We can support any format by default, until we get restricted */
- ast_format_cap_from_old_bitfield(ast_channel_nativeformats(tmp), capability);
- ast_best_codec(ast_channel_nativeformats(tmp), &tmpfmt);
- ast_format_copy(ast_channel_readformat(tmp), &tmpfmt);
- ast_format_copy(ast_channel_rawreadformat(tmp), &tmpfmt);
- ast_format_copy(ast_channel_writeformat(tmp), &tmpfmt);
- ast_format_copy(ast_channel_rawwriteformat(tmp), &tmpfmt);
- ast_channel_tech_pvt_set(tmp, CALLNO_TO_PTR(i->callno));
- if (!ast_strlen_zero(i->parkinglot))
- ast_channel_parkinglot_set(tmp, i->parkinglot);
- /* Don't use ast_set_callerid() here because it will
- * generate a NewCallerID event before the NewChannel event */
- if (!ast_strlen_zero(i->ani)) {
- ast_channel_caller(tmp)->ani.number.valid = 1;
- ast_channel_caller(tmp)->ani.number.str = ast_strdup(i->ani);
- } else if (!ast_strlen_zero(i->cid_num)) {
- ast_channel_caller(tmp)->ani.number.valid = 1;
- ast_channel_caller(tmp)->ani.number.str = ast_strdup(i->cid_num);
- }
- ast_channel_dialed(tmp)->number.str = ast_strdup(i->dnid);
- if (!ast_strlen_zero(i->rdnis)) {
- ast_channel_redirecting(tmp)->from.number.valid = 1;
- ast_channel_redirecting(tmp)->from.number.str = ast_strdup(i->rdnis);
- }
- ast_channel_caller(tmp)->id.name.presentation = i->calling_pres;
- ast_channel_caller(tmp)->id.number.presentation = i->calling_pres;
- ast_channel_caller(tmp)->id.number.plan = i->calling_ton;
- ast_channel_dialed(tmp)->transit_network_select = i->calling_tns;
- if (!ast_strlen_zero(i->language))
- ast_channel_language_set(tmp, i->language);
- if (!ast_strlen_zero(i->accountcode))
- ast_channel_accountcode_set(tmp, i->accountcode);
- if (i->amaflags)
- ast_channel_amaflags_set(tmp, i->amaflags);
- ast_channel_context_set(tmp, i->context);
- ast_channel_exten_set(tmp, i->exten);
- if (i->adsi)
- ast_channel_adsicpe_set(tmp, i->peeradsicpe);
- else
- ast_channel_adsicpe_set(tmp, AST_ADSI_UNAVAILABLE);
- i->owner = tmp;
- i->capability = capability;
- if (!cachable) {
- ast_set_flag(ast_channel_flags(tmp), AST_FLAG_DISABLE_DEVSTATE_CACHE);
- }
- /* Set inherited variables */
- if (i->vars) {
- for (v = i->vars ; v ; v = v->next)
- pbx_builtin_setvar_helper(tmp, v->name, v->value);
- }
- if (i->iaxvars) {
- struct ast_datastore *variablestore;
- struct ast_variable *var, *prev = NULL;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- ast_debug(1, "Loading up the channel with IAXVARs\n");
- varlist = ast_calloc(1, sizeof(*varlist));
- variablestore = ast_datastore_alloc(&iax2_variable_datastore_info, NULL);
- if (variablestore && varlist) {
- variablestore->data = varlist;
- variablestore->inheritance = DATASTORE_INHERIT_FOREVER;
- AST_LIST_HEAD_INIT(varlist);
- for (var = i->iaxvars; var; var = var->next) {
- struct ast_var_t *newvar = ast_var_assign(var->name, var->value);
- if (prev)
- ast_free(prev);
- prev = var;
- if (!newvar) {
- /* Don't abort list traversal, as this would leave i->iaxvars in an inconsistent state. */
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- } else {
- AST_LIST_INSERT_TAIL(varlist, newvar, entries);
- }
- }
- if (prev)
- ast_free(prev);
- i->iaxvars = NULL;
- ast_channel_datastore_add(i->owner, variablestore);
- } else {
- if (variablestore) {
- ast_datastore_free(variablestore);
- }
- if (varlist) {
- ast_free(varlist);
- }
- }
- }
- if (state != AST_STATE_DOWN) {
- if (ast_pbx_start(tmp)) {
- ast_log(LOG_WARNING, "Unable to start PBX on %s\n", ast_channel_name(tmp));
- ast_hangup(tmp);
- i->owner = NULL;
- return NULL;
- }
- }
- ast_module_ref(ast_module_info->self);
- return tmp;
- }
- static unsigned int calc_txpeerstamp(struct iax2_trunk_peer *tpeer, int sampms, struct timeval *now)
- {
- unsigned long int mssincetx; /* unsigned to handle overflows */
- long int ms, pred;
- tpeer->trunkact = *now;
- mssincetx = ast_tvdiff_ms(*now, tpeer->lasttxtime);
- if (mssincetx > 5000 || ast_tvzero(tpeer->txtrunktime)) {
- /* If it's been at least 5 seconds since the last time we transmitted on this trunk, reset our timers */
- tpeer->txtrunktime = *now;
- tpeer->lastsent = 999999;
- }
- /* Update last transmit time now */
- tpeer->lasttxtime = *now;
-
- /* Calculate ms offset */
- ms = ast_tvdiff_ms(*now, tpeer->txtrunktime);
- /* Predict from last value */
- pred = tpeer->lastsent + sampms;
- if (labs(ms - pred) < MAX_TIMESTAMP_SKEW)
- ms = pred;
-
- /* We never send the same timestamp twice, so fudge a little if we must */
- if (ms == tpeer->lastsent)
- ms = tpeer->lastsent + 1;
- tpeer->lastsent = ms;
- return ms;
- }
- static unsigned int fix_peerts(struct timeval *rxtrunktime, int callno, unsigned int ts)
- {
- long ms; /* NOT unsigned */
- if (ast_tvzero(iaxs[callno]->rxcore)) {
- /* Initialize rxcore time if appropriate */
- iaxs[callno]->rxcore = ast_tvnow();
- /* Round to nearest 20ms so traces look pretty */
- iaxs[callno]->rxcore.tv_usec -= iaxs[callno]->rxcore.tv_usec % 20000;
- }
- /* Calculate difference between trunk and channel */
- ms = ast_tvdiff_ms(*rxtrunktime, iaxs[callno]->rxcore);
- /* Return as the sum of trunk time and the difference between trunk and real time */
- return ms + ts;
- }
- static unsigned int calc_timestamp(struct chan_iax2_pvt *p, unsigned int ts, struct ast_frame *f)
- {
- int ms;
- int voice = 0;
- int genuine = 0;
- int adjust;
- int rate = ast_format_rate(&f->subclass.format) / 1000;
- struct timeval *delivery = NULL;
- /* What sort of frame do we have?: voice is self-explanatory
- "genuine" means an IAX frame - things like LAGRQ/RP, PING/PONG, ACK
- non-genuine frames are CONTROL frames [ringing etc], DTMF
- The "genuine" distinction is needed because genuine frames must get a clock-based timestamp,
- the others need a timestamp slaved to the voice frames so that they go in sequence
- */
- if (f->frametype == AST_FRAME_VOICE) {
- voice = 1;
- delivery = &f->delivery;
- } else if (f->frametype == AST_FRAME_IAX) {
- genuine = 1;
- } else if (f->frametype == AST_FRAME_CNG) {
- p->notsilenttx = 0;
- }
- if (ast_tvzero(p->offset)) {
- p->offset = ast_tvnow();
- /* Round to nearest 20ms for nice looking traces */
- p->offset.tv_usec -= p->offset.tv_usec % 20000;
- }
- /* If the timestamp is specified, just send it as is */
- if (ts)
- return ts;
- /* If we have a time that the frame arrived, always use it to make our timestamp */
- if (delivery && !ast_tvzero(*delivery)) {
- ms = ast_tvdiff_ms(*delivery, p->offset);
- if (ms < 0) {
- ms = 0;
- }
- if (iaxdebug)
- ast_debug(3, "calc_timestamp: call %d/%d: Timestamp slaved to delivery time\n", p->callno, iaxs[p->callno]->peercallno);
- } else {
- ms = ast_tvdiff_ms(ast_tvnow(), p->offset);
- if (ms < 0)
- ms = 0;
- if (voice) {
- /* On a voice frame, use predicted values if appropriate */
- adjust = (ms - p->nextpred);
- if (p->notsilenttx && abs(adjust) <= MAX_TIMESTAMP_SKEW) {
- /* Adjust our txcore, keeping voice and non-voice synchronized */
- /* AN EXPLANATION:
- When we send voice, we usually send "calculated" timestamps worked out
- on the basis of the number of samples sent. When we send other frames,
- we usually send timestamps worked out from the real clock.
- The problem is that they can tend to drift out of step because the
- source channel's clock and our clock may not be exactly at the same rate.
- We fix this by continuously "tweaking" p->offset. p->offset is "time zero"
- for this call. Moving it adjusts timestamps for non-voice frames.
- We make the adjustment in the style of a moving average. Each time we
- adjust p->offset by 10% of the difference between our clock-derived
- timestamp and the predicted timestamp. That's why you see "10000"
- below even though IAX2 timestamps are in milliseconds.
- The use of a moving average avoids offset moving too radically.
- Generally, "adjust" roams back and forth around 0, with offset hardly
- changing at all. But if a consistent different starts to develop it
- will be eliminated over the course of 10 frames (200-300msecs)
- */
- if (adjust < 0)
- p->offset = ast_tvsub(p->offset, ast_samp2tv(abs(adjust), 10000));
- else if (adjust > 0)
- p->offset = ast_tvadd(p->offset, ast_samp2tv(adjust, 10000));
- if (!p->nextpred) {
- p->nextpred = ms; /*f->samples / rate;*/
- if (p->nextpred <= p->lastsent)
- p->nextpred = p->lastsent + 3;
- }
- ms = p->nextpred;
- } else {
- /* in this case, just use the actual
- * time, since we're either way off
- * (shouldn't happen), or we're ending a
- * silent period -- and seed the next
- * predicted time. Also, round ms to the
- * next multiple of frame size (so our
- * silent periods are multiples of
- * frame size too) */
- if (iaxdebug && abs(adjust) > MAX_TIMESTAMP_SKEW )
- ast_debug(1, "predicted timestamp skew (%d) > max (%d), using real ts instead.\n",
- abs(adjust), MAX_TIMESTAMP_SKEW);
- if (f->samples >= rate) /* check to make sure we don't core dump */
- {
- int diff = ms % (f->samples / rate);
- if (diff)
- ms += f->samples/rate - diff;
- }
- p->nextpred = ms;
- p->notsilenttx = 1;
- }
- } else if ( f->frametype == AST_FRAME_VIDEO ) {
- /*
- * IAX2 draft 03 says that timestamps MUST be in order.
- * It does not say anything about several frames having the same timestamp
- * When transporting video, we can have a frame that spans multiple iax packets
- * (so called slices), so it would make sense to use the same timestamp for all of
- * them
- * We do want to make sure that frames don't go backwards though
- */
- if ( (unsigned int)ms < p->lastsent )
- ms = p->lastsent;
- } else {
- /* On a dataframe, use last value + 3 (to accomodate jitter buffer shrinking) if appropriate unless
- it's a genuine frame */
- adjust = (ms - p->lastsent);
- if (genuine) {
- /* genuine (IAX LAGRQ etc) must keep their clock-based stamps */
- if (ms <= p->lastsent)
- ms = p->lastsent + 3;
- } else if (abs(adjust) <= MAX_TIMESTAMP_SKEW) {
- /* non-genuine frames (!?) (DTMF, CONTROL) should be pulled into the predicted stream stamps */
- ms = p->lastsent + 3;
- }
- }
- }
- p->lastsent = ms;
- if (voice)
- p->nextpred = p->nextpred + f->samples / rate;
- return ms;
- }
- static unsigned int calc_rxstamp(struct chan_iax2_pvt *p, unsigned int offset)
- {
- /* Returns where in "receive time" we are. That is, how many ms
- since we received (or would have received) the frame with timestamp 0 */
- int ms;
- #ifdef IAXTESTS
- int jit;
- #endif /* IAXTESTS */
- /* Setup rxcore if necessary */
- if (ast_tvzero(p->rxcore)) {
- p->rxcore = ast_tvnow();
- if (iaxdebug)
- ast_debug(1, "calc_rxstamp: call=%d: rxcore set to %d.%6.6d - %ums\n",
- p->callno, (int)(p->rxcore.tv_sec), (int)(p->rxcore.tv_usec), offset);
- p->rxcore = ast_tvsub(p->rxcore, ast_samp2tv(offset, 1000));
- #if 1
- if (iaxdebug)
- ast_debug(1, "calc_rxstamp: call=%d: works out as %d.%6.6d\n",
- p->callno, (int)(p->rxcore.tv_sec),(int)( p->rxcore.tv_usec));
- #endif
- }
- ms = ast_tvdiff_ms(ast_tvnow(), p->rxcore);
- #ifdef IAXTESTS
- if (test_jit) {
- if (!test_jitpct || ((100.0 * ast_random() / (RAND_MAX + 1.0)) < test_jitpct)) {
- jit = (int)((float)test_jit * ast_random() / (RAND_MAX + 1.0));
- if ((int)(2.0 * ast_random() / (RAND_MAX + 1.0)))
- jit = -jit;
- ms += jit;
- }
- }
- if (test_late) {
- ms += test_late;
- test_late = 0;
- }
- #endif /* IAXTESTS */
- return ms;
- }
- static struct iax2_trunk_peer *find_tpeer(struct sockaddr_in *sin, int fd)
- {
- struct iax2_trunk_peer *tpeer = NULL;
-
- /* Finds and locks trunk peer */
- AST_LIST_LOCK(&tpeers);
- AST_LIST_TRAVERSE(&tpeers, tpeer, list) {
- if (!inaddrcmp(&tpeer->addr, sin)) {
- ast_mutex_lock(&tpeer->lock);
- break;
- }
- }
- if (!tpeer) {
- if ((tpeer = ast_calloc(1, sizeof(*tpeer)))) {
- ast_mutex_init(&tpeer->lock);
- tpeer->lastsent = 9999;
- memcpy(&tpeer->addr, sin, sizeof(tpeer->addr));
- tpeer->trunkact = ast_tvnow();
- ast_mutex_lock(&tpeer->lock);
- tpeer->sockfd = fd;
- #ifdef SO_NO_CHECK
- setsockopt(tpeer->sockfd, SOL_SOCKET, SO_NO_CHECK, &nochecksums, sizeof(nochecksums));
- #endif
- ast_debug(1, "Created trunk peer for '%s:%d'\n", ast_inet_ntoa(tpeer->addr.sin_addr), ntohs(tpeer->addr.sin_port));
- AST_LIST_INSERT_TAIL(&tpeers, tpeer, list);
- }
- }
- AST_LIST_UNLOCK(&tpeers);
- return tpeer;
- }
- static int iax2_trunk_queue(struct chan_iax2_pvt *pvt, struct iax_frame *fr)
- {
- struct ast_frame *f;
- struct iax2_trunk_peer *tpeer;
- void *tmp, *ptr;
- struct timeval now;
- struct ast_iax2_meta_trunk_entry *met;
- struct ast_iax2_meta_trunk_mini *mtm;
- f = &fr->af;
- tpeer = find_tpeer(&pvt->addr, pvt->sockfd);
- if (tpeer) {
- if (tpeer->trunkdatalen + f->datalen + 4 >= tpeer->trunkdataalloc) {
- /* Need to reallocate space */
- if (tpeer->trunkdataalloc < trunkmaxsize) {
- if (!(tmp = ast_realloc(tpeer->trunkdata, tpeer->trunkdataalloc + DEFAULT_TRUNKDATA + IAX2_TRUNK_PREFACE))) {
- ast_mutex_unlock(&tpeer->lock);
- return -1;
- }
- tpeer->trunkdataalloc += DEFAULT_TRUNKDATA;
- tpeer->trunkdata = tmp;
- ast_debug(1, "Expanded trunk '%s:%d' to %u bytes\n", ast_inet_ntoa(tpeer->addr.sin_addr), ntohs(tpeer->addr.sin_port), tpeer->trunkdataalloc);
- } else {
- ast_log(LOG_WARNING, "Maximum trunk data space exceeded to %s:%d\n", ast_inet_ntoa(tpeer->addr.sin_addr), ntohs(tpeer->addr.sin_port));
- ast_mutex_unlock(&tpeer->lock);
- return -1;
- }
- }
- /* Append to meta frame */
- ptr = tpeer->trunkdata + IAX2_TRUNK_PREFACE + tpeer->trunkdatalen;
- if (ast_test_flag64(&globalflags, IAX_TRUNKTIMESTAMPS)) {
- mtm = (struct ast_iax2_meta_trunk_mini *)ptr;
- mtm->len = htons(f->datalen);
- mtm->mini.callno = htons(pvt->callno);
- mtm->mini.ts = htons(0xffff & fr->ts);
- ptr += sizeof(struct ast_iax2_meta_trunk_mini);
- tpeer->trunkdatalen += sizeof(struct ast_iax2_meta_trunk_mini);
- } else {
- met = (struct ast_iax2_meta_trunk_entry *)ptr;
- /* Store call number and length in meta header */
- met->callno = htons(pvt->callno);
- met->len = htons(f->datalen);
- /* Advance pointers/decrease length past trunk entry header */
- ptr += sizeof(struct ast_iax2_meta_trunk_entry);
- tpeer->trunkdatalen += sizeof(struct ast_iax2_meta_trunk_entry);
- }
- /* Copy actual trunk data */
- memcpy(ptr, f->data.ptr, f->datalen);
- tpeer->trunkdatalen += f->datalen;
- tpeer->calls++;
- /* track the largest mtu we actually have sent */
- if (tpeer->trunkdatalen + f->datalen + 4 > trunk_maxmtu)
- trunk_maxmtu = tpeer->trunkdatalen + f->datalen + 4 ;
- /* if we have enough for a full MTU, ship it now without waiting */
- if (global_max_trunk_mtu > 0 && tpeer->trunkdatalen + f->datalen + 4 >= global_max_trunk_mtu) {
- now = ast_tvnow();
- send_trunk(tpeer, &now);
- trunk_untimed ++;
- }
- ast_mutex_unlock(&tpeer->lock);
- }
- return 0;
- }
- /* IAX2 encryption requires 16 to 32 bytes of random padding to be present
- * before the encryption data. This function randomizes that data. */
- static void build_rand_pad(unsigned char *buf, ssize_t len)
- {
- long tmp;
- for (tmp = ast_random(); len > 0; tmp = ast_random()) {
- memcpy(buf, (unsigned char *) &tmp, (len > sizeof(tmp)) ? sizeof(tmp) : len);
- buf += sizeof(tmp);
- len -= sizeof(tmp);
- }
- }
- static void build_encryption_keys(const unsigned char *digest, struct chan_iax2_pvt *pvt)
- {
- build_ecx_key(digest, pvt);
- ast_aes_set_decrypt_key(digest, &pvt->dcx);
- }
- static void build_ecx_key(const unsigned char *digest, struct chan_iax2_pvt *pvt)
- {
- /* it is required to hold the corresponding decrypt key to our encrypt key
- * in the pvt struct because queued frames occasionally need to be decrypted and
- * re-encrypted when updated for a retransmission */
- build_rand_pad(pvt->semirand, sizeof(pvt->semirand));
- ast_aes_set_encrypt_key(digest, &pvt->ecx);
- ast_aes_set_decrypt_key(digest, &pvt->mydcx);
- }
- static void memcpy_decrypt(unsigned char *dst, const unsigned char *src, int len, ast_aes_decrypt_key *dcx)
- {
- #if 0
- /* Debug with "fake encryption" */
- int x;
- if (len % 16)
- ast_log(LOG_WARNING, "len should be multiple of 16, not %d!\n", len);
- for (x=0;x<len;x++)
- dst[x] = src[x] ^ 0xff;
- #else
- unsigned char lastblock[16] = { 0 };
- int x;
- while(len > 0) {
- ast_aes_decrypt(src, dst, dcx);
- for (x=0;x<16;x++)
- dst[x] ^= lastblock[x];
- memcpy(lastblock, src, sizeof(lastblock));
- dst += 16;
- src += 16;
- len -= 16;
- }
- #endif
- }
- static void memcpy_encrypt(unsigned char *dst, const unsigned char *src, int len, ast_aes_encrypt_key *ecx)
- {
- #if 0
- /* Debug with "fake encryption" */
- int x;
- if (len % 16)
- ast_log(LOG_WARNING, "len should be multiple of 16, not %d!\n", len);
- for (x=0;x<len;x++)
- dst[x] = src[x] ^ 0xff;
- #else
- unsigned char curblock[16] = { 0 };
- int x;
- while(len > 0) {
- for (x=0;x<16;x++)
- curblock[x] ^= src[x];
- ast_aes_encrypt(curblock, dst, ecx);
- memcpy(curblock, dst, sizeof(curblock));
- dst += 16;
- src += 16;
- len -= 16;
- }
- #endif
- }
- static int decode_frame(ast_aes_decrypt_key *dcx, struct ast_iax2_full_hdr *fh, struct ast_frame *f, int *datalen)
- {
- int padding;
- unsigned char *workspace;
- workspace = ast_alloca(*datalen);
- memset(f, 0, sizeof(*f));
- if (ntohs(fh->scallno) & IAX_FLAG_FULL) {
- struct ast_iax2_full_enc_hdr *efh = (struct ast_iax2_full_enc_hdr *)fh;
- if (*datalen < 16 + sizeof(struct ast_iax2_full_hdr))
- return -1;
- /* Decrypt */
- memcpy_decrypt(workspace, efh->encdata, *datalen - sizeof(struct ast_iax2_full_enc_hdr), dcx);
- padding = 16 + (workspace[15] & 0x0f);
- if (iaxdebug)
- ast_debug(1, "Decoding full frame with length %d (padding = %d) (15=%02hhx)\n", *datalen, padding, workspace[15]);
- if (*datalen < padding + sizeof(struct ast_iax2_full_hdr))
- return -1;
- *datalen -= padding;
- memcpy(efh->encdata, workspace + padding, *datalen - sizeof(struct ast_iax2_full_enc_hdr));
- f->frametype = fh->type;
- if (f->frametype == AST_FRAME_VIDEO) {
- ast_format_from_old_bitfield(&f->subclass.format, (uncompress_subclass(fh->csub & ~0x40) | ((fh->csub >> 6) & 0x1)));
- } else if (f->frametype == AST_FRAME_VOICE) {
- ast_format_from_old_bitfield(&f->subclass.format, uncompress_subclass(fh->csub));
- } else {
- f->subclass.integer = uncompress_subclass(fh->csub);
- }
- } else {
- struct ast_iax2_mini_enc_hdr *efh = (struct ast_iax2_mini_enc_hdr *)fh;
- if (iaxdebug)
- ast_debug(1, "Decoding mini with length %d\n", *datalen);
- if (*datalen < 16 + sizeof(struct ast_iax2_mini_hdr))
- return -1;
- /* Decrypt */
- memcpy_decrypt(workspace, efh->encdata, *datalen - sizeof(struct ast_iax2_mini_enc_hdr), dcx);
- padding = 16 + (workspace[15] & 0x0f);
- if (*datalen < padding + sizeof(struct ast_iax2_mini_hdr))
- return -1;
- *datalen -= padding;
- memcpy(efh->encdata, workspace + padding, *datalen - sizeof(struct ast_iax2_mini_enc_hdr));
- }
- return 0;
- }
- static int encrypt_frame(ast_aes_encrypt_key *ecx, struct ast_iax2_full_hdr *fh, unsigned char *poo, int *datalen)
- {
- int padding;
- unsigned char *workspace;
- workspace = ast_alloca(*datalen + 32);
- if (ntohs(fh->scallno) & IAX_FLAG_FULL) {
- struct ast_iax2_full_enc_hdr *efh = (struct ast_iax2_full_enc_hdr *)fh;
- if (iaxdebug)
- ast_debug(1, "Encoding full frame %d/%d with length %d\n", fh->type, fh->csub, *datalen);
- padding = 16 - ((*datalen - sizeof(struct ast_iax2_full_enc_hdr)) % 16);
- padding = 16 + (padding & 0xf);
- memcpy(workspace, poo, padding);
- memcpy(workspace + padding, efh->encdata, *datalen - sizeof(struct ast_iax2_full_enc_hdr));
- workspace[15] &= 0xf0;
- workspace[15] |= (padding & 0xf);
- if (iaxdebug)
- ast_debug(1, "Encoding full frame %d/%d with length %d + %d padding (15=%02hhx)\n", fh->type, fh->csub, *datalen, padding, workspace[15]);
- *datalen += padding;
- memcpy_encrypt(efh->encdata, workspace, *datalen - sizeof(struct ast_iax2_full_enc_hdr), ecx);
- if (*datalen >= 32 + sizeof(struct ast_iax2_full_enc_hdr))
- memcpy(poo, workspace + *datalen - 32, 32);
- } else {
- struct ast_iax2_mini_enc_hdr *efh = (struct ast_iax2_mini_enc_hdr *)fh;
- if (iaxdebug)
- ast_debug(1, "Encoding mini frame with length %d\n", *datalen);
- padding = 16 - ((*datalen - sizeof(struct ast_iax2_mini_enc_hdr)) % 16);
- padding = 16 + (padding & 0xf);
- memcpy(workspace, poo, padding);
- memcpy(workspace + padding, efh->encdata, *datalen - sizeof(struct ast_iax2_mini_enc_hdr));
- workspace[15] &= 0xf0;
- workspace[15] |= (padding & 0x0f);
- *datalen += padding;
- memcpy_encrypt(efh->encdata, workspace, *datalen - sizeof(struct ast_iax2_mini_enc_hdr), ecx);
- if (*datalen >= 32 + sizeof(struct ast_iax2_mini_enc_hdr))
- memcpy(poo, workspace + *datalen - 32, 32);
- }
- return 0;
- }
- static int decrypt_frame(int callno, struct ast_iax2_full_hdr *fh, struct ast_frame *f, int *datalen)
- {
- int res=-1;
- if (!ast_test_flag64(iaxs[callno], IAX_KEYPOPULATED)) {
- /* Search for possible keys, given secrets */
- struct MD5Context md5;
- unsigned char digest[16];
- char *tmppw, *stringp;
-
- tmppw = ast_strdupa(iaxs[callno]->secret);
- stringp = tmppw;
- while ((tmppw = strsep(&stringp, ";"))) {
- MD5Init(&md5);
- MD5Update(&md5, (unsigned char *)iaxs[callno]->challenge, strlen(iaxs[callno]->challenge));
- MD5Update(&md5, (unsigned char *)tmppw, strlen(tmppw));
- MD5Final(digest, &md5);
- build_encryption_keys(digest, iaxs[callno]);
- res = decode_frame(&iaxs[callno]->dcx, fh, f, datalen);
- if (!res) {
- ast_set_flag64(iaxs[callno], IAX_KEYPOPULATED);
- break;
- }
- }
- } else
- res = decode_frame(&iaxs[callno]->dcx, fh, f, datalen);
- return res;
- }
- static int iax2_send(struct chan_iax2_pvt *pvt, struct ast_frame *f, unsigned int ts, int seqno, int now, int transfer, int final)
- {
- /* Queue a packet for delivery on a given private structure. Use "ts" for
- timestamp, or calculate if ts is 0. Send immediately without retransmission
- or delayed, with retransmission */
- struct ast_iax2_full_hdr *fh;
- struct ast_iax2_mini_hdr *mh;
- struct ast_iax2_video_hdr *vh;
- struct {
- struct iax_frame fr2;
- unsigned char buffer[4096];
- } frb;
- struct iax_frame *fr;
- int res;
- int sendmini=0;
- unsigned int lastsent;
- unsigned int fts;
- frb.fr2.afdatalen = sizeof(frb.buffer);
- if (!pvt) {
- ast_log(LOG_WARNING, "No private structure for packet?\n");
- return -1;
- }
-
- lastsent = pvt->lastsent;
- /* Calculate actual timestamp */
- fts = calc_timestamp(pvt, ts, f);
- /* Bail here if this is an "interp" frame; we don't want or need to send these placeholders out
- * (the endpoint should detect the lost packet itself). But, we want to do this here, so that we
- * increment the "predicted timestamps" for voice, if we're predicting */
- if(f->frametype == AST_FRAME_VOICE && f->datalen == 0)
- return 0;
- #if 0
- ast_log(LOG_NOTICE,
- "f->frametype %c= AST_FRAME_VOICE, %sencrypted, %srotation scheduled...\n",
- *("=!" + (f->frametype == AST_FRAME_VOICE)),
- IAX_CALLENCRYPTED(pvt) ? "" : "not ",
- pvt->keyrotateid != -1 ? "" : "no "
- );
- #endif
- if (pvt->keyrotateid == -1 && f->frametype == AST_FRAME_VOICE && IAX_CALLENCRYPTED(pvt)) {
- iax2_key_rotate(pvt);
- }
- if ((ast_test_flag64(pvt, IAX_TRUNK) ||
- (((fts & 0xFFFF0000L) == (lastsent & 0xFFFF0000L)) ||
- ((fts & 0xFFFF0000L) == ((lastsent + 0x10000) & 0xFFFF0000L))))
- /* High two bytes are the same on timestamp, or sending on a trunk */ &&
- (f->frametype == AST_FRAME_VOICE)
- /* is a voice frame */ &&
- (f->subclass.format.id == ast_format_id_from_old_bitfield(pvt->svoiceformat))
- /* is the same type */ ) {
- /* Force immediate rather than delayed transmission */
- now = 1;
- /* Mark that mini-style frame is appropriate */
- sendmini = 1;
- }
- if ( f->frametype == AST_FRAME_VIDEO ) {
- /*
- * If the lower 15 bits of the timestamp roll over, or if
- * the video format changed then send a full frame.
- * Otherwise send a mini video frame
- */
- if (((fts & 0xFFFF8000L) == (pvt->lastvsent & 0xFFFF8000L)) &&
- ((f->subclass.format.id) == ast_format_id_from_old_bitfield(pvt->svideoformat))
- ) {
- now = 1;
- sendmini = 1;
- } else {
- now = 0;
- sendmini = 0;
- }
- pvt->lastvsent = fts;
- }
- if (f->frametype == AST_FRAME_IAX) {
- /* 0x8000 marks this message as TX:, this bit will be stripped later */
- pvt->last_iax_message = f->subclass.integer | MARK_IAX_SUBCLASS_TX;
- if (!pvt->first_iax_message) {
- pvt->first_iax_message = pvt->last_iax_message;
- }
- }
- /* Allocate an iax_frame */
- if (now) {
- fr = &frb.fr2;
- } else
- fr = iax_frame_new(DIRECTION_OUTGRESS, ast_test_flag64(pvt, IAX_ENCRYPTED) ? f->datalen + 32 : f->datalen, (f->frametype == AST_FRAME_VOICE) || (f->frametype == AST_FRAME_VIDEO));
- if (!fr) {
- ast_log(LOG_WARNING, "Out of memory\n");
- return -1;
- }
- /* Copy our prospective frame into our immediate or retransmitted wrapper */
- iax_frame_wrap(fr, f);
- fr->ts = fts;
- fr->callno = pvt->callno;
- fr->transfer = transfer;
- fr->final = final;
- fr->encmethods = 0;
- if (!sendmini) {
- /* We need a full frame */
- if (seqno > -1)
- fr->oseqno = seqno;
- else
- fr->oseqno = pvt->oseqno++;
- fr->iseqno = pvt->iseqno;
- fh = (struct ast_iax2_full_hdr *)(fr->af.data.ptr - sizeof(struct ast_iax2_full_hdr));
- fh->scallno = htons(fr->callno | IAX_FLAG_FULL);
- fh->ts = htonl(fr->ts);
- fh->oseqno = fr->oseqno;
- if (transfer) {
- fh->iseqno = 0;
- } else
- fh->iseqno = fr->iseqno;
- /* Keep track of the last thing we've acknowledged */
- if (!transfer)
- pvt->aseqno = fr->iseqno;
- fh->type = fr->af.frametype & 0xFF;
- if (fr->af.frametype == AST_FRAME_VIDEO) {
- iax2_format tmpfmt = ast_format_to_old_bitfield(&fr->af.subclass.format);
- tmpfmt |= ast_format_get_video_mark(&fr->af.subclass.format) ? 0x1LL : 0;
- fh->csub = compress_subclass(tmpfmt | ((tmpfmt & 0x1LL) << 6));
- } else if (fr->af.frametype == AST_FRAME_VOICE) {
- fh->csub = compress_subclass(ast_format_to_old_bitfield(&fr->af.subclass.format));
- } else {
- fh->csub = compress_subclass(fr->af.subclass.integer);
- }
- if (transfer) {
- fr->dcallno = pvt->transfercallno;
- } else
- fr->dcallno = pvt->peercallno;
- fh->dcallno = htons(fr->dcallno);
- fr->datalen = fr->af.datalen + sizeof(struct ast_iax2_full_hdr);
- fr->data = fh;
- fr->retries = 0;
- /* Retry after 2x the ping time has passed */
- fr->retrytime = pvt->pingtime * 2;
- if (fr->retrytime < MIN_RETRY_TIME)
- fr->retrytime = MIN_RETRY_TIME;
- if (fr->retrytime > MAX_RETRY_TIME)
- fr->retrytime = MAX_RETRY_TIME;
- /* Acks' don't get retried */
- if ((f->frametype == AST_FRAME_IAX) && (f->subclass.integer == IAX_COMMAND_ACK))
- fr->retries = -1;
- else if (f->frametype == AST_FRAME_VOICE)
- pvt->svoiceformat = ast_format_to_old_bitfield(&f->subclass.format);
- else if (f->frametype == AST_FRAME_VIDEO)
- pvt->svideoformat = ast_format_to_old_bitfield(&f->subclass.format);
- if (ast_test_flag64(pvt, IAX_ENCRYPTED)) {
- if (ast_test_flag64(pvt, IAX_KEYPOPULATED)) {
- if (fr->transfer)
- iax_outputframe(fr, NULL, 2, &pvt->transfer, fr->datalen - sizeof(struct ast_iax2_full_hdr));
- else
- iax_outputframe(fr, NULL, 2, &pvt->addr, fr->datalen - sizeof(struct ast_iax2_full_hdr));
- encrypt_frame(&pvt->ecx, fh, pvt->semirand, &fr->datalen);
- fr->encmethods = pvt->encmethods;
- fr->ecx = pvt->ecx;
- fr->mydcx = pvt->mydcx;
- memcpy(fr->semirand, pvt->semirand, sizeof(fr->semirand));
- } else
- ast_log(LOG_WARNING, "Supposed to send packet encrypted, but no key?\n");
- }
- if (now) {
- res = send_packet(fr);
- } else
- res = iax2_transmit(fr);
- } else {
- if (ast_test_flag64(pvt, IAX_TRUNK)) {
- iax2_trunk_queue(pvt, fr);
- res = 0;
- } else if (fr->af.frametype == AST_FRAME_VIDEO) {
- /* Video frame have no sequence number */
- fr->oseqno = -1;
- fr->iseqno = -1;
- vh = (struct ast_iax2_video_hdr *)(fr->af.data.ptr - sizeof(struct ast_iax2_video_hdr));
- vh->zeros = 0;
- vh->callno = htons(0x8000 | fr->callno);
- vh->ts = htons((fr->ts & 0x7FFF) | (ast_format_get_video_mark(&fr->af.subclass.format) ? 0x8000 : 0));
- fr->datalen = fr->af.datalen + sizeof(struct ast_iax2_video_hdr);
- fr->data = vh;
- fr->retries = -1;
- res = send_packet(fr);
- } else {
- /* Mini-frames have no sequence number */
- fr->oseqno = -1;
- fr->iseqno = -1;
- /* Mini frame will do */
- mh = (struct ast_iax2_mini_hdr *)(fr->af.data.ptr - sizeof(struct ast_iax2_mini_hdr));
- mh->callno = htons(fr->callno);
- mh->ts = htons(fr->ts & 0xFFFF);
- fr->datalen = fr->af.datalen + sizeof(struct ast_iax2_mini_hdr);
- fr->data = mh;
- fr->retries = -1;
- if (pvt->transferring == TRANSFER_MEDIAPASS)
- fr->transfer = 1;
- if (ast_test_flag64(pvt, IAX_ENCRYPTED)) {
- if (ast_test_flag64(pvt, IAX_KEYPOPULATED)) {
- encrypt_frame(&pvt->ecx, (struct ast_iax2_full_hdr *)mh, pvt->semirand, &fr->datalen);
- } else
- ast_log(LOG_WARNING, "Supposed to send packet encrypted, but no key?\n");
- }
- res = send_packet(fr);
- }
- }
- return res;
- }
- static char *handle_cli_iax2_show_users(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- regex_t regexbuf;
- int havepattern = 0;
- #define FORMAT "%-15.15s %-20.20s %-15.15s %-15.15s %-5.5s %-5.10s\n"
- #define FORMAT2 "%-15.15s %-20.20s %-15.15d %-15.15s %-5.5s %-5.10s\n"
- struct iax2_user *user = NULL;
- char auth[90];
- char *pstr = "";
- struct ao2_iterator i;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show users [like]";
- e->usage =
- "Usage: iax2 show users [like <pattern>]\n"
- " Lists all known IAX2 users.\n"
- " Optional regular expression pattern is used to filter the user list.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- switch (a->argc) {
- case 5:
- if (!strcasecmp(a->argv[3], "like")) {
- if (regcomp(®exbuf, a->argv[4], REG_EXTENDED | REG_NOSUB))
- return CLI_SHOWUSAGE;
- havepattern = 1;
- } else
- return CLI_SHOWUSAGE;
- case 3:
- break;
- default:
- return CLI_SHOWUSAGE;
- }
- ast_cli(a->fd, FORMAT, "Username", "Secret", "Authen", "Def.Context", "A/C","Codec Pref");
- i = ao2_iterator_init(users, 0);
- for (; (user = ao2_iterator_next(&i)); user_unref(user)) {
- if (havepattern && regexec(®exbuf, user->name, 0, NULL, 0))
- continue;
- if (!ast_strlen_zero(user->secret)) {
- ast_copy_string(auth,user->secret, sizeof(auth));
- } else if (!ast_strlen_zero(user->inkeys)) {
- snprintf(auth, sizeof(auth), "Key: %-15.15s ", user->inkeys);
- } else
- ast_copy_string(auth, "-no secret-", sizeof(auth));
- if(ast_test_flag64(user, IAX_CODEC_NOCAP))
- pstr = "REQ Only";
- else if(ast_test_flag64(user, IAX_CODEC_NOPREFS))
- pstr = "Disabled";
- else
- pstr = ast_test_flag64(user, IAX_CODEC_USER_FIRST) ? "Caller" : "Host";
- ast_cli(a->fd, FORMAT2, user->name, auth, user->authmethods,
- user->contexts ? user->contexts->context : DEFAULT_CONTEXT,
- ast_acl_list_is_empty(user->acl) ? "No" : "Yes", pstr);
- }
- ao2_iterator_destroy(&i);
- if (havepattern)
- regfree(®exbuf);
- return CLI_SUCCESS;
- #undef FORMAT
- #undef FORMAT2
- }
- static int __iax2_show_peers(int fd, int *total, struct mansession *s, const int argc, const char * const argv[])
- {
- regex_t regexbuf;
- int havepattern = 0;
- int total_peers = 0;
- int online_peers = 0;
- int offline_peers = 0;
- int unmonitored_peers = 0;
- struct ao2_iterator i;
- #define FORMAT2 "%-15.15s %-15.15s %s %-15.15s %-8s %s %-11s %-32.32s\n"
- #define FORMAT "%-15.15s %-15.15s %s %-15.15s %-5d%s %s %-11s %-32.32s\n"
- struct iax2_peer *peer = NULL;
- char name[256];
- struct ast_str *encmethods = ast_str_alloca(256);
- int registeredonly=0;
- char idtext[256] = "";
- switch (argc) {
- case 6:
- if (!strcasecmp(argv[3], "registered"))
- registeredonly = 1;
- else
- return RESULT_SHOWUSAGE;
- if (!strcasecmp(argv[4], "like")) {
- if (regcomp(®exbuf, argv[5], REG_EXTENDED | REG_NOSUB))
- return RESULT_SHOWUSAGE;
- havepattern = 1;
- } else
- return RESULT_SHOWUSAGE;
- break;
- case 5:
- if (!strcasecmp(argv[3], "like")) {
- if (regcomp(®exbuf, argv[4], REG_EXTENDED | REG_NOSUB))
- return RESULT_SHOWUSAGE;
- havepattern = 1;
- } else
- return RESULT_SHOWUSAGE;
- break;
- case 4:
- if (!strcasecmp(argv[3], "registered"))
- registeredonly = 1;
- else
- return RESULT_SHOWUSAGE;
- break;
- case 3:
- break;
- default:
- return RESULT_SHOWUSAGE;
- }
- if (!s)
- ast_cli(fd, FORMAT2, "Name/Username", "Host", " ", "Mask", "Port", " ", "Status", "Description");
- i = ao2_iterator_init(peers, 0);
- for (; (peer = ao2_iterator_next(&i)); peer_unref(peer)) {
- char nm[20];
- char status[20];
- int retstatus;
- struct sockaddr_in peer_addr;
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- if (registeredonly && !peer_addr.sin_addr.s_addr) {
- continue;
- }
- if (havepattern && regexec(®exbuf, peer->name, 0, NULL, 0)) {
- continue;
- }
- if (!ast_strlen_zero(peer->username))
- snprintf(name, sizeof(name), "%s/%s", peer->name, peer->username);
- else
- ast_copy_string(name, peer->name, sizeof(name));
- encmethods_to_str(peer->encmethods, &encmethods);
- retstatus = peer_status(peer, status, sizeof(status));
- if (retstatus > 0)
- online_peers++;
- else if (!retstatus)
- offline_peers++;
- else
- unmonitored_peers++;
- ast_copy_string(nm, ast_inet_ntoa(peer->mask), sizeof(nm));
- if (s) {
- astman_append(s,
- "Event: PeerEntry\r\n%s"
- "Channeltype: IAX2\r\n"
- "ObjectName: %s\r\n"
- "ChanObjectType: peer\r\n"
- "IPaddress: %s\r\n"
- "IPport: %d\r\n"
- "Dynamic: %s\r\n"
- "Trunk: %s\r\n"
- "Encryption: %s\r\n"
- "Status: %s\r\n"
- "Description: %s\r\n\r\n",
- idtext,
- name,
- ast_sockaddr_stringify_addr(&peer->addr),
- ast_sockaddr_port(&peer->addr),
- ast_test_flag64(peer, IAX_DYNAMIC) ? "yes" : "no",
- ast_test_flag64(peer, IAX_TRUNK) ? "yes" : "no",
- peer->encmethods ? ast_str_buffer(encmethods) : "no",
- status,
- peer->description);
- } else {
- ast_cli(fd, FORMAT, name,
- ast_sockaddr_stringify_addr(&peer->addr),
- ast_test_flag64(peer, IAX_DYNAMIC) ? "(D)" : "(S)",
- nm,
- ast_sockaddr_port(&peer->addr),
- ast_test_flag64(peer, IAX_TRUNK) ? "(T)" : " ",
- peer->encmethods ? "(E)" : " ",
- status,
- peer->description);
- }
- total_peers++;
- }
- ao2_iterator_destroy(&i);
- if (!s)
- ast_cli(fd,"%d iax2 peers [%d online, %d offline, %d unmonitored]\n",
- total_peers, online_peers, offline_peers, unmonitored_peers);
- if (havepattern)
- regfree(®exbuf);
- if (total)
- *total = total_peers;
- return RESULT_SUCCESS;
- #undef FORMAT
- #undef FORMAT2
- }
- static char *handle_cli_iax2_show_threads(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct iax2_thread *thread = NULL;
- time_t t;
- int threadcount = 0, dynamiccount = 0;
- char type;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show threads";
- e->usage =
- "Usage: iax2 show threads\n"
- " Lists status of IAX helper threads\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 3)
- return CLI_SHOWUSAGE;
-
- ast_cli(a->fd, "IAX2 Thread Information\n");
- time(&t);
- ast_cli(a->fd, "Idle Threads:\n");
- AST_LIST_LOCK(&idle_list);
- AST_LIST_TRAVERSE(&idle_list, thread, list) {
- #ifdef DEBUG_SCHED_MULTITHREAD
- ast_cli(a->fd, "Thread %d: state=%u, update=%d, actions=%d, func='%s'\n",
- thread->threadnum, thread->iostate, (int)(t - thread->checktime), thread->actions, thread->curfunc);
- #else
- ast_cli(a->fd, "Thread %d: state=%u, update=%d, actions=%d\n",
- thread->threadnum, thread->iostate, (int)(t - thread->checktime), thread->actions);
- #endif
- threadcount++;
- }
- AST_LIST_UNLOCK(&idle_list);
- ast_cli(a->fd, "Active Threads:\n");
- AST_LIST_LOCK(&active_list);
- AST_LIST_TRAVERSE(&active_list, thread, list) {
- if (thread->type == IAX_THREAD_TYPE_DYNAMIC)
- type = 'D';
- else
- type = 'P';
- #ifdef DEBUG_SCHED_MULTITHREAD
- ast_cli(a->fd, "Thread %c%d: state=%u, update=%d, actions=%d, func='%s'\n",
- type, thread->threadnum, thread->iostate, (int)(t - thread->checktime), thread->actions, thread->curfunc);
- #else
- ast_cli(a->fd, "Thread %c%d: state=%u, update=%d, actions=%d\n",
- type, thread->threadnum, thread->iostate, (int)(t - thread->checktime), thread->actions);
- #endif
- threadcount++;
- }
- AST_LIST_UNLOCK(&active_list);
- ast_cli(a->fd, "Dynamic Threads:\n");
- AST_LIST_LOCK(&dynamic_list);
- AST_LIST_TRAVERSE(&dynamic_list, thread, list) {
- #ifdef DEBUG_SCHED_MULTITHREAD
- ast_cli(a->fd, "Thread %d: state=%u, update=%d, actions=%d, func='%s'\n",
- thread->threadnum, thread->iostate, (int)(t - thread->checktime), thread->actions, thread->curfunc);
- #else
- ast_cli(a->fd, "Thread %d: state=%u, update=%d, actions=%d\n",
- thread->threadnum, thread->iostate, (int)(t - thread->checktime), thread->actions);
- #endif
- dynamiccount++;
- }
- AST_LIST_UNLOCK(&dynamic_list);
- ast_cli(a->fd, "%d of %d threads accounted for with %d dynamic threads\n", threadcount, iaxthreadcount, dynamiccount);
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_unregister(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct iax2_peer *p;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 unregister";
- e->usage =
- "Usage: iax2 unregister <peername>\n"
- " Unregister (force expiration) an IAX2 peer from the registry.\n";
- return NULL;
- case CLI_GENERATE:
- return complete_iax2_unregister(a->line, a->word, a->pos, a->n);
- }
- if (a->argc != 3)
- return CLI_SHOWUSAGE;
- p = find_peer(a->argv[2], 1);
- if (p) {
- if (p->expire > 0) {
- struct iax2_peer *peer;
- peer = ao2_find(peers, a->argv[2], OBJ_KEY);
- if (peer) {
- expire_registry(peer_ref(peer)); /* will release its own reference when done */
- peer_unref(peer); /* ref from ao2_find() */
- ast_cli(a->fd, "Peer %s unregistered\n", a->argv[2]);
- } else {
- ast_cli(a->fd, "Peer %s not found\n", a->argv[2]);
- }
- } else {
- ast_cli(a->fd, "Peer %s not registered\n", a->argv[2]);
- }
- peer_unref(p);
- } else {
- ast_cli(a->fd, "Peer unknown: %s. Not unregistered\n", a->argv[2]);
- }
- return CLI_SUCCESS;
- }
- static char *complete_iax2_unregister(const char *line, const char *word, int pos, int state)
- {
- int which = 0;
- struct iax2_peer *p = NULL;
- char *res = NULL;
- int wordlen = strlen(word);
- /* 0 - iax2; 1 - unregister; 2 - <peername> */
- if (pos == 2) {
- struct ao2_iterator i = ao2_iterator_init(peers, 0);
- while ((p = ao2_iterator_next(&i))) {
- if (!strncasecmp(p->name, word, wordlen) &&
- ++which > state && p->expire > 0) {
- res = ast_strdup(p->name);
- peer_unref(p);
- break;
- }
- peer_unref(p);
- }
- ao2_iterator_destroy(&i);
- }
- return res;
- }
- static char *handle_cli_iax2_show_peers(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show peers";
- e->usage =
- "Usage: iax2 show peers [registered] [like <pattern>]\n"
- " Lists all known IAX2 peers.\n"
- " Optional 'registered' argument lists only peers with known addresses.\n"
- " Optional regular expression pattern is used to filter the peer list.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- switch (__iax2_show_peers(a->fd, NULL, NULL, a->argc, a->argv)) {
- case RESULT_SHOWUSAGE:
- return CLI_SHOWUSAGE;
- case RESULT_FAILURE:
- return CLI_FAILURE;
- default:
- return CLI_SUCCESS;
- }
- }
- static int manager_iax2_show_netstats(struct mansession *s, const struct message *m)
- {
- ast_cli_netstats(s, -1, 0);
- astman_append(s, "\r\n");
- return RESULT_SUCCESS;
- }
- static char *handle_cli_iax2_show_firmware(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- struct iax_firmware *cur = NULL;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show firmware";
- e->usage =
- "Usage: iax2 show firmware\n"
- " Lists all known IAX firmware images.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 3 && a->argc != 4)
- return CLI_SHOWUSAGE;
- ast_cli(a->fd, "%-15.15s %-15.15s %-15.15s\n", "Device", "Version", "Size");
- AST_LIST_LOCK(&firmwares);
- AST_LIST_TRAVERSE(&firmwares, cur, list) {
- if ((a->argc == 3) || (!strcasecmp(a->argv[3], (char *) cur->fwh->devname))) {
- ast_cli(a->fd, "%-15.15s %-15d %-15d\n", cur->fwh->devname,
- ntohs(cur->fwh->version), (int)ntohl(cur->fwh->datalen));
- }
- }
- AST_LIST_UNLOCK(&firmwares);
- return CLI_SUCCESS;
- }
- /*! \brief callback to display iax peers in manager */
- static int manager_iax2_show_peers(struct mansession *s, const struct message *m)
- {
- static const char * const a[] = { "iax2", "show", "peers" };
- const char *id = astman_get_header(m,"ActionID");
- char idtext[256] = "";
- int total = 0;
- if (!ast_strlen_zero(id))
- snprintf(idtext, sizeof(idtext), "ActionID: %s\r\n", id);
- astman_send_listack(s, m, "Peer status list will follow", "start");
- /* List the peers in separate manager events */
- __iax2_show_peers(-1, &total, s, 3, a);
- /* Send final confirmation */
- astman_append(s,
- "Event: PeerlistComplete\r\n"
- "EventList: Complete\r\n"
- "ListItems: %d\r\n"
- "%s"
- "\r\n", total, idtext);
- return 0;
- }
- /*! \brief callback to display iax peers in manager format */
- static int manager_iax2_show_peer_list(struct mansession *s, const struct message *m)
- {
- struct iax2_peer *peer = NULL;
- int peer_count = 0;
- char nm[20];
- char status[20];
- const char *id = astman_get_header(m,"ActionID");
- char idtext[256] = "";
- struct ast_str *encmethods = ast_str_alloca(256);
- struct ao2_iterator i;
- if (!ast_strlen_zero(id))
- snprintf(idtext, sizeof(idtext), "ActionID: %s\r\n", id);
- astman_append(s, "Response: Success\r\n%sMessage: IAX Peer status list will follow\r\n\r\n", idtext);
- i = ao2_iterator_init(peers, 0);
- for (; (peer = ao2_iterator_next(&i)); peer_unref(peer)) {
- encmethods_to_str(peer->encmethods, &encmethods);
- astman_append(s, "Event: PeerEntry\r\n%sChanneltype: IAX\r\n", idtext);
- if (!ast_strlen_zero(peer->username)) {
- astman_append(s, "ObjectName: %s\r\nObjectUsername: %s\r\n", peer->name, peer->username);
- } else {
- astman_append(s, "ObjectName: %s\r\n", peer->name);
- }
- astman_append(s, "ChanObjectType: peer\r\n");
- astman_append(s, "IPaddress: %s\r\n", ast_sockaddr_stringify_addr(&peer->addr));
- ast_copy_string(nm, ast_inet_ntoa(peer->mask), sizeof(nm));
- astman_append(s, "Mask: %s\r\n", nm);
- astman_append(s, "Port: %d\r\n", ast_sockaddr_port(&peer->addr));
- astman_append(s, "Dynamic: %s\r\n", ast_test_flag64(peer, IAX_DYNAMIC) ? "Yes" : "No");
- astman_append(s, "Trunk: %s\r\n", ast_test_flag64(peer, IAX_TRUNK) ? "Yes" : "No");
- astman_append(s, "Encryption: %s\r\n", peer->encmethods ? ast_str_buffer(encmethods) : "No");
- peer_status(peer, status, sizeof(status));
- astman_append(s, "Status: %s\r\n\r\n", status);
- peer_count++;
- }
- ao2_iterator_destroy(&i);
- astman_append(s, "Event: PeerlistComplete\r\n%sListItems: %d\r\n\r\n", idtext, peer_count);
- return RESULT_SUCCESS;
- }
- static char *regstate2str(int regstate)
- {
- switch(regstate) {
- case REG_STATE_UNREGISTERED:
- return "Unregistered";
- case REG_STATE_REGSENT:
- return "Request Sent";
- case REG_STATE_AUTHSENT:
- return "Auth. Sent";
- case REG_STATE_REGISTERED:
- return "Registered";
- case REG_STATE_REJECTED:
- return "Rejected";
- case REG_STATE_TIMEOUT:
- return "Timeout";
- case REG_STATE_NOAUTH:
- return "No Authentication";
- default:
- return "Unknown";
- }
- }
- static char *handle_cli_iax2_show_registry(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- #define FORMAT2 "%-20.20s %-6.6s %-10.10s %-20.20s %8.8s %s\n"
- #define FORMAT "%-20.20s %-6.6s %-10.10s %-20.20s %8d %s\n"
- struct iax2_registry *reg = NULL;
- char host[80];
- char perceived[80];
- int counter = 0;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show registry";
- e->usage =
- "Usage: iax2 show registry\n"
- " Lists all registration requests and status.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 3)
- return CLI_SHOWUSAGE;
- ast_cli(a->fd, FORMAT2, "Host", "dnsmgr", "Username", "Perceived", "Refresh", "State");
- AST_LIST_LOCK(®istrations);
- AST_LIST_TRAVERSE(®istrations, reg, entry) {
- snprintf(host, sizeof(host), "%s", ast_sockaddr_stringify(®->addr));
- if (reg->us.sin_addr.s_addr)
- snprintf(perceived, sizeof(perceived), "%s:%d", ast_inet_ntoa(reg->us.sin_addr), ntohs(reg->us.sin_port));
- else
- ast_copy_string(perceived, "<Unregistered>", sizeof(perceived));
- ast_cli(a->fd, FORMAT, host,
- (reg->dnsmgr) ? "Y" : "N",
- reg->username, perceived, reg->refresh, regstate2str(reg->regstate));
- counter++;
- }
- AST_LIST_UNLOCK(®istrations);
- ast_cli(a->fd, "%d IAX2 registrations.\n", counter);
- return CLI_SUCCESS;
- #undef FORMAT
- #undef FORMAT2
- }
- static int manager_iax2_show_registry(struct mansession *s, const struct message *m)
- {
- const char *id = astman_get_header(m, "ActionID");
- struct iax2_registry *reg = NULL;
- char idtext[256] = "";
- char host[80] = "";
- char perceived[80] = "";
- int total = 0;
- if (!ast_strlen_zero(id))
- snprintf(idtext, sizeof(idtext), "ActionID: %s\r\n", id);
- astman_send_listack(s, m, "Registrations will follow", "start");
- AST_LIST_LOCK(®istrations);
- AST_LIST_TRAVERSE(®istrations, reg, entry) {
- snprintf(host, sizeof(host), "%s", ast_sockaddr_stringify(®->addr));
-
- if (reg->us.sin_addr.s_addr) {
- snprintf(perceived, sizeof(perceived), "%s:%d", ast_inet_ntoa(reg->us.sin_addr), ntohs(reg->us.sin_port));
- } else {
- ast_copy_string(perceived, "<Unregistered>", sizeof(perceived));
- }
-
- astman_append(s,
- "Event: RegistryEntry\r\n"
- "%s"
- "Host: %s\r\n"
- "DNSmanager: %s\r\n"
- "Username: %s\r\n"
- "Perceived: %s\r\n"
- "Refresh: %d\r\n"
- "State: %s\r\n"
- "\r\n", idtext, host, (reg->dnsmgr) ? "Y" : "N", reg->username, perceived,
- reg->refresh, regstate2str(reg->regstate));
- total++;
- }
- AST_LIST_UNLOCK(®istrations);
- astman_append(s,
- "Event: RegistrationsComplete\r\n"
- "EventList: Complete\r\n"
- "ListItems: %d\r\n"
- "%s"
- "\r\n", total, idtext);
-
- return 0;
- }
- static char *handle_cli_iax2_show_channels(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- #define FORMAT2 "%-20.20s %-15.15s %-10.10s %-11.11s %-11.11s %-7.7s %-6.6s %-6.6s %s %s %9s\n"
- #define FORMAT "%-20.20s %-15.15s %-10.10s %5.5d/%5.5d %5.5d/%5.5d %-5.5dms %-4.4dms %-4.4dms %-6.6s %s%s %3s%s\n"
- #define FORMATB "%-20.20s %-15.15s %-10.10s %5.5d/%5.5d %5.5d/%5.5d [Native Bridged to ID=%5.5d]\n"
- int x;
- int numchans = 0;
- char first_message[10] = { 0, };
- char last_message[10] = { 0, };
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show channels";
- e->usage =
- "Usage: iax2 show channels\n"
- " Lists all currently active IAX channels.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 3)
- return CLI_SHOWUSAGE;
- ast_cli(a->fd, FORMAT2, "Channel", "Peer", "Username", "ID (Lo/Rem)", "Seq (Tx/Rx)", "Lag", "Jitter", "JitBuf", "Format", "FirstMsg", "LastMsg");
- for (x = 0; x < ARRAY_LEN(iaxs); x++) {
- ast_mutex_lock(&iaxsl[x]);
- if (iaxs[x]) {
- int lag, jitter, localdelay;
- jb_info jbinfo;
- if (ast_test_flag64(iaxs[x], IAX_USEJITTERBUF)) {
- jb_getinfo(iaxs[x]->jb, &jbinfo);
- jitter = jbinfo.jitter;
- localdelay = jbinfo.current - jbinfo.min;
- } else {
- jitter = -1;
- localdelay = 0;
- }
- iax_frame_subclass2str(iaxs[x]->first_iax_message & ~MARK_IAX_SUBCLASS_TX, first_message, sizeof(first_message));
- iax_frame_subclass2str(iaxs[x]->last_iax_message & ~MARK_IAX_SUBCLASS_TX, last_message, sizeof(last_message));
- lag = iaxs[x]->remote_rr.delay;
- ast_cli(a->fd, FORMAT,
- iaxs[x]->owner ? ast_channel_name(iaxs[x]->owner) : "(None)",
- ast_inet_ntoa(iaxs[x]->addr.sin_addr),
- S_OR(iaxs[x]->username, "(None)"),
- iaxs[x]->callno, iaxs[x]->peercallno,
- iaxs[x]->oseqno, iaxs[x]->iseqno,
- lag,
- jitter,
- localdelay,
- iax2_getformatname(iaxs[x]->voiceformat),
- (iaxs[x]->first_iax_message & MARK_IAX_SUBCLASS_TX) ? "Tx:" : "Rx:",
- first_message,
- (iaxs[x]->last_iax_message & MARK_IAX_SUBCLASS_TX) ? "Tx:" : "Rx:",
- last_message);
- numchans++;
- }
- ast_mutex_unlock(&iaxsl[x]);
- }
- ast_cli(a->fd, "%d active IAX channel%s\n", numchans, (numchans != 1) ? "s" : "");
- return CLI_SUCCESS;
- #undef FORMAT
- #undef FORMAT2
- #undef FORMATB
- }
- static int ast_cli_netstats(struct mansession *s, int fd, int limit_fmt)
- {
- int x;
- int numchans = 0;
- char first_message[10] = { 0, };
- char last_message[10] = { 0, };
- #define ACN_FORMAT1 "%-20.25s %4u %4d %4d %5d %3d %5d %4d %6d %4d %4d %5d %3d %5d %4d %6d %s%s %4s%s\n"
- #define ACN_FORMAT2 "%s %u %d %d %d %d %d %d %d %d %d %d %d %d %d %d %s%s %s%s\n"
- for (x = 0; x < ARRAY_LEN(iaxs); x++) {
- ast_mutex_lock(&iaxsl[x]);
- if (iaxs[x]) {
- int localjitter, localdelay, locallost, locallosspct, localdropped, localooo;
- jb_info jbinfo;
- iax_frame_subclass2str(iaxs[x]->first_iax_message & ~MARK_IAX_SUBCLASS_TX, first_message, sizeof(first_message));
- iax_frame_subclass2str(iaxs[x]->last_iax_message & ~MARK_IAX_SUBCLASS_TX, last_message, sizeof(last_message));
- if(ast_test_flag64(iaxs[x], IAX_USEJITTERBUF)) {
- jb_getinfo(iaxs[x]->jb, &jbinfo);
- localjitter = jbinfo.jitter;
- localdelay = jbinfo.current - jbinfo.min;
- locallost = jbinfo.frames_lost;
- locallosspct = jbinfo.losspct/1000;
- localdropped = jbinfo.frames_dropped;
- localooo = jbinfo.frames_ooo;
- } else {
- localjitter = -1;
- localdelay = 0;
- locallost = -1;
- locallosspct = -1;
- localdropped = 0;
- localooo = -1;
- }
- if (s)
- astman_append(s, limit_fmt ? ACN_FORMAT1 : ACN_FORMAT2,
- iaxs[x]->owner ? ast_channel_name(iaxs[x]->owner) : "(None)",
- iaxs[x]->pingtime,
- localjitter,
- localdelay,
- locallost,
- locallosspct,
- localdropped,
- localooo,
- iaxs[x]->frames_received/1000,
- iaxs[x]->remote_rr.jitter,
- iaxs[x]->remote_rr.delay,
- iaxs[x]->remote_rr.losscnt,
- iaxs[x]->remote_rr.losspct,
- iaxs[x]->remote_rr.dropped,
- iaxs[x]->remote_rr.ooo,
- iaxs[x]->remote_rr.packets/1000,
- (iaxs[x]->first_iax_message & MARK_IAX_SUBCLASS_TX) ? "Tx:" : "Rx:",
- first_message,
- (iaxs[x]->last_iax_message & MARK_IAX_SUBCLASS_TX) ? "Tx:" : "Rx:",
- last_message);
- else
- ast_cli(fd, limit_fmt ? ACN_FORMAT1 : ACN_FORMAT2,
- iaxs[x]->owner ? ast_channel_name(iaxs[x]->owner) : "(None)",
- iaxs[x]->pingtime,
- localjitter,
- localdelay,
- locallost,
- locallosspct,
- localdropped,
- localooo,
- iaxs[x]->frames_received/1000,
- iaxs[x]->remote_rr.jitter,
- iaxs[x]->remote_rr.delay,
- iaxs[x]->remote_rr.losscnt,
- iaxs[x]->remote_rr.losspct,
- iaxs[x]->remote_rr.dropped,
- iaxs[x]->remote_rr.ooo,
- iaxs[x]->remote_rr.packets/1000,
- (iaxs[x]->first_iax_message & MARK_IAX_SUBCLASS_TX) ? "Tx:" : "Rx:",
- first_message,
- (iaxs[x]->last_iax_message & MARK_IAX_SUBCLASS_TX) ? "Tx:" : "Rx:",
- last_message);
- numchans++;
- }
- ast_mutex_unlock(&iaxsl[x]);
- }
- return numchans;
- }
- static char *handle_cli_iax2_show_netstats(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- int numchans = 0;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 show netstats";
- e->usage =
- "Usage: iax2 show netstats\n"
- " Lists network status for all currently active IAX channels.\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != 3)
- return CLI_SHOWUSAGE;
- ast_cli(a->fd, " -------- LOCAL --------------------- -------- REMOTE --------------------\n");
- ast_cli(a->fd, "Channel RTT Jit Del Lost %% Drop OOO Kpkts Jit Del Lost %% Drop OOO Kpkts FirstMsg LastMsg\n");
- numchans = ast_cli_netstats(NULL, a->fd, 1);
- ast_cli(a->fd, "%d active IAX channel%s\n", numchans, (numchans != 1) ? "s" : "");
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_set_debug(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 set debug {on|off|peer}";
- e->usage =
- "Usage: iax2 set debug {on|off|peer peername}\n"
- " Enables/Disables dumping of IAX packets for debugging purposes.\n";
- return NULL;
- case CLI_GENERATE:
- if (a->pos == 4 && !strcasecmp(a->argv[3], "peer"))
- return complete_iax2_peers(a->line, a->word, a->pos, a->n, 0);
- return NULL;
- }
- if (a->argc < e->args || a->argc > e->args + 1)
- return CLI_SHOWUSAGE;
- if (!strcasecmp(a->argv[3], "peer")) {
- struct iax2_peer *peer;
- struct sockaddr_in peer_addr;
- if (a->argc != e->args + 1)
- return CLI_SHOWUSAGE;
- peer = find_peer(a->argv[4], 1);
- if (!peer) {
- ast_cli(a->fd, "IAX2 peer '%s' does not exist\n", a->argv[e->args-1]);
- return CLI_FAILURE;
- }
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- debugaddr.sin_addr = peer_addr.sin_addr;
- debugaddr.sin_port = peer_addr.sin_port;
- ast_cli(a->fd, "IAX2 Debugging Enabled for IP: %s:%d\n",
- ast_inet_ntoa(debugaddr.sin_addr), ntohs(debugaddr.sin_port));
- ao2_ref(peer, -1);
- } else if (!strncasecmp(a->argv[3], "on", 2)) {
- iaxdebug = 1;
- ast_cli(a->fd, "IAX2 Debugging Enabled\n");
- } else {
- iaxdebug = 0;
- memset(&debugaddr, 0, sizeof(debugaddr));
- ast_cli(a->fd, "IAX2 Debugging Disabled\n");
- }
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_set_debug_trunk(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 set debug trunk {on|off}";
- e->usage =
- "Usage: iax2 set debug trunk {on|off}\n"
- " Enables/Disables debugging of IAX trunking\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != e->args)
- return CLI_SHOWUSAGE;
- if (!strncasecmp(a->argv[e->args - 1], "on", 2)) {
- iaxtrunkdebug = 1;
- ast_cli(a->fd, "IAX2 Trunk Debugging Enabled\n");
- } else {
- iaxtrunkdebug = 0;
- ast_cli(a->fd, "IAX2 Trunk Debugging Disabled\n");
- }
- return CLI_SUCCESS;
- }
- static char *handle_cli_iax2_set_debug_jb(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 set debug jb {on|off}";
- e->usage =
- "Usage: iax2 set debug jb {on|off}\n"
- " Enables/Disables jitterbuffer debugging information\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- if (a->argc != e->args)
- return CLI_SHOWUSAGE;
-
- if (!strncasecmp(a->argv[e->args -1], "on", 2)) {
- jb_setoutput(jb_error_output, jb_warning_output, jb_debug_output);
- ast_cli(a->fd, "IAX2 Jitterbuffer Debugging Enabled\n");
- } else {
- jb_setoutput(jb_error_output, jb_warning_output, NULL);
- ast_cli(a->fd, "IAX2 Jitterbuffer Debugging Disabled\n");
- }
- return CLI_SUCCESS;
- }
- static int iax2_write(struct ast_channel *c, struct ast_frame *f)
- {
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(c));
- int res = -1;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- /* If there's an outstanding error, return failure now */
- if (!iaxs[callno]->error) {
- if (ast_test_flag64(iaxs[callno], IAX_ALREADYGONE))
- res = 0;
- /* Don't waste bandwidth sending null frames */
- else if (f->frametype == AST_FRAME_NULL)
- res = 0;
- else if ((f->frametype == AST_FRAME_VOICE) && ast_test_flag64(iaxs[callno], IAX_QUELCH))
- res = 0;
- else if (!ast_test_flag(&iaxs[callno]->state, IAX_STATE_STARTED))
- res = 0;
- else
- /* Simple, just queue for transmission */
- res = iax2_send(iaxs[callno], f, 0, -1, 0, 0, 0);
- } else {
- ast_debug(1, "Write error: %s\n", strerror(errno));
- }
- }
- /* If it's already gone, just return */
- ast_mutex_unlock(&iaxsl[callno]);
- return res;
- }
- static int __send_command(struct chan_iax2_pvt *i, char type, int command, unsigned int ts, const unsigned char *data, int datalen, int seqno,
- int now, int transfer, int final)
- {
- struct ast_frame f = { 0, };
- int res = 0;
- f.frametype = type;
- f.subclass.integer = command;
- f.datalen = datalen;
- f.src = __FUNCTION__;
- f.data.ptr = (void *) data;
- if ((res = queue_signalling(i, &f)) <= 0) {
- return res;
- }
- return iax2_send(i, &f, ts, seqno, now, transfer, final);
- }
- static int send_command(struct chan_iax2_pvt *i, char type, int command, unsigned int ts, const unsigned char *data, int datalen, int seqno)
- {
- if (type == AST_FRAME_CONTROL && !iax2_is_control_frame_allowed(command)) {
- /* Control frame should not go out on the wire. */
- ast_debug(2, "Callno %d: Blocked sending control frame %d.\n",
- i->callno, command);
- return 0;
- }
- return __send_command(i, type, command, ts, data, datalen, seqno, 0, 0, 0);
- }
- static int send_command_locked(unsigned short callno, char type, int command, unsigned int ts, const unsigned char *data, int datalen, int seqno)
- {
- int res;
- ast_mutex_lock(&iaxsl[callno]);
- res = send_command(iaxs[callno], type, command, ts, data, datalen, seqno);
- ast_mutex_unlock(&iaxsl[callno]);
- return res;
- }
- /*!
- * \note Since this function calls iax2_predestroy() -> iax2_queue_hangup(),
- * the pvt struct for the given call number may disappear during its
- * execution.
- */
- static int send_command_final(struct chan_iax2_pvt *i, char type, int command, unsigned int ts, const unsigned char *data, int datalen, int seqno)
- {
- int call_num = i->callno;
- /* It is assumed that the callno has already been locked */
- iax2_predestroy(i->callno);
- if (!iaxs[call_num])
- return -1;
- return __send_command(i, type, command, ts, data, datalen, seqno, 0, 0, 1);
- }
- static int send_command_immediate(struct chan_iax2_pvt *i, char type, int command, unsigned int ts, const unsigned char *data, int datalen, int seqno)
- {
- return __send_command(i, type, command, ts, data, datalen, seqno, 1, 0, 0);
- }
- static int send_command_transfer(struct chan_iax2_pvt *i, char type, int command, unsigned int ts, const unsigned char *data, int datalen)
- {
- return __send_command(i, type, command, ts, data, datalen, 0, 0, 1, 0);
- }
- static int apply_context(struct iax2_context *con, const char *context)
- {
- while(con) {
- if (!strcmp(con->context, context) || !strcmp(con->context, "*"))
- return -1;
- con = con->next;
- }
- return 0;
- }
- static int check_access(int callno, struct sockaddr_in *sin, struct iax_ies *ies)
- {
- /* Start pessimistic */
- int res = -1;
- int version = 2;
- struct iax2_user *user = NULL, *best = NULL;
- int bestscore = 0;
- int gotcapability = 0;
- struct ast_variable *v = NULL, *tmpvar = NULL;
- struct ao2_iterator i;
- struct ast_sockaddr addr;
- if (!iaxs[callno])
- return res;
- if (ies->called_number)
- ast_string_field_set(iaxs[callno], exten, ies->called_number);
- if (ies->calling_number) {
- if (ast_test_flag64(&globalflags, IAX_SHRINKCALLERID)) {
- ast_shrink_phone_number(ies->calling_number);
- }
- ast_string_field_set(iaxs[callno], cid_num, ies->calling_number);
- }
- if (ies->calling_name)
- ast_string_field_set(iaxs[callno], cid_name, ies->calling_name);
- if (ies->calling_ani)
- ast_string_field_set(iaxs[callno], ani, ies->calling_ani);
- if (ies->dnid)
- ast_string_field_set(iaxs[callno], dnid, ies->dnid);
- if (ies->rdnis)
- ast_string_field_set(iaxs[callno], rdnis, ies->rdnis);
- if (ies->called_context)
- ast_string_field_set(iaxs[callno], context, ies->called_context);
- if (ies->language)
- ast_string_field_set(iaxs[callno], language, ies->language);
- if (ies->username)
- ast_string_field_set(iaxs[callno], username, ies->username);
- if (ies->calling_ton > -1)
- iaxs[callno]->calling_ton = ies->calling_ton;
- if (ies->calling_tns > -1)
- iaxs[callno]->calling_tns = ies->calling_tns;
- if (ies->calling_pres > -1)
- iaxs[callno]->calling_pres = ies->calling_pres;
- if (ies->format)
- iaxs[callno]->peerformat = ies->format;
- if (ies->adsicpe)
- iaxs[callno]->peeradsicpe = ies->adsicpe;
- if (ies->capability) {
- gotcapability = 1;
- iaxs[callno]->peercapability = ies->capability;
- }
- if (ies->version)
- version = ies->version;
- /* Use provided preferences until told otherwise for actual preferences */
- if (ies->codec_prefs) {
- ast_codec_pref_convert(&iaxs[callno]->rprefs, ies->codec_prefs, 32, 0);
- ast_codec_pref_convert(&iaxs[callno]->prefs, ies->codec_prefs, 32, 0);
- }
- if (!gotcapability)
- iaxs[callno]->peercapability = iaxs[callno]->peerformat;
- if (version > IAX_PROTO_VERSION) {
- ast_log(LOG_WARNING, "Peer '%s' has too new a protocol version (%d) for me\n",
- ast_inet_ntoa(sin->sin_addr), version);
- return res;
- }
- /* Search the userlist for a compatible entry, and fill in the rest */
- ast_sockaddr_from_sin(&addr, sin);
- i = ao2_iterator_init(users, 0);
- while ((user = ao2_iterator_next(&i))) {
- if ((ast_strlen_zero(iaxs[callno]->username) || /* No username specified */
- !strcmp(iaxs[callno]->username, user->name)) /* Or this username specified */
- && (ast_apply_acl(user->acl, &addr, "IAX2 user ACL: ") == AST_SENSE_ALLOW) /* Access is permitted from this IP */
- && (ast_strlen_zero(iaxs[callno]->context) || /* No context specified */
- apply_context(user->contexts, iaxs[callno]->context))) { /* Context is permitted */
- if (!ast_strlen_zero(iaxs[callno]->username)) {
- /* Exact match, stop right now. */
- if (best)
- user_unref(best);
- best = user;
- break;
- } else if (ast_strlen_zero(user->secret) && ast_strlen_zero(user->dbsecret) && ast_strlen_zero(user->inkeys)) {
- /* No required authentication */
- if (user->acl) {
- /* There was host authentication and we passed, bonus! */
- if (bestscore < 4) {
- bestscore = 4;
- if (best)
- user_unref(best);
- best = user;
- continue;
- }
- } else {
- /* No host access, but no secret, either, not bad */
- if (bestscore < 3) {
- bestscore = 3;
- if (best)
- user_unref(best);
- best = user;
- continue;
- }
- }
- } else {
- if (user->acl) {
- /* Authentication, but host access too, eh, it's something.. */
- if (bestscore < 2) {
- bestscore = 2;
- if (best)
- user_unref(best);
- best = user;
- continue;
- }
- } else {
- /* Authentication and no host access... This is our baseline */
- if (bestscore < 1) {
- bestscore = 1;
- if (best)
- user_unref(best);
- best = user;
- continue;
- }
- }
- }
- }
- user_unref(user);
- }
- ao2_iterator_destroy(&i);
- user = best;
- if (!user && !ast_strlen_zero(iaxs[callno]->username)) {
- user = realtime_user(iaxs[callno]->username, sin);
- if (user && (ast_apply_acl(user->acl, &addr, "IAX2 user ACL: ") == AST_SENSE_DENY /* Access is denied from this IP */
- || (!ast_strlen_zero(iaxs[callno]->context) && /* No context specified */
- !apply_context(user->contexts, iaxs[callno]->context)))) { /* Context is permitted */
- user = user_unref(user);
- }
- }
- if (user) {
- /* We found our match (use the first) */
- /* copy vars */
- for (v = user->vars ; v ; v = v->next) {
- if((tmpvar = ast_variable_new(v->name, v->value, v->file))) {
- tmpvar->next = iaxs[callno]->vars;
- iaxs[callno]->vars = tmpvar;
- }
- }
- /* If a max AUTHREQ restriction is in place, activate it */
- if (user->maxauthreq > 0)
- ast_set_flag64(iaxs[callno], IAX_MAXAUTHREQ);
- iaxs[callno]->prefs = user->prefs;
- ast_copy_flags64(iaxs[callno], user, IAX_CODEC_USER_FIRST | IAX_IMMEDIATE | IAX_CODEC_NOPREFS | IAX_CODEC_NOCAP | IAX_FORCE_ENCRYPT);
- iaxs[callno]->encmethods = user->encmethods;
- /* Store the requested username if not specified */
- if (ast_strlen_zero(iaxs[callno]->username))
- ast_string_field_set(iaxs[callno], username, user->name);
- /* Store whether this is a trunked call, too, of course, and move if appropriate */
- ast_copy_flags64(iaxs[callno], user, IAX_TRUNK);
- iaxs[callno]->capability = user->capability;
- /* And use the default context */
- if (ast_strlen_zero(iaxs[callno]->context)) {
- if (user->contexts)
- ast_string_field_set(iaxs[callno], context, user->contexts->context);
- else
- ast_string_field_set(iaxs[callno], context, DEFAULT_CONTEXT);
- }
- /* And any input keys */
- ast_string_field_set(iaxs[callno], inkeys, user->inkeys);
- /* And the permitted authentication methods */
- iaxs[callno]->authmethods = user->authmethods;
- iaxs[callno]->adsi = user->adsi;
- /* If the user has callerid, override the remote caller id. */
- if (ast_test_flag64(user, IAX_HASCALLERID)) {
- iaxs[callno]->calling_tns = 0;
- iaxs[callno]->calling_ton = 0;
- ast_string_field_set(iaxs[callno], cid_num, user->cid_num);
- ast_string_field_set(iaxs[callno], cid_name, user->cid_name);
- ast_string_field_set(iaxs[callno], ani, user->cid_num);
- iaxs[callno]->calling_pres = AST_PRES_ALLOWED_USER_NUMBER_PASSED_SCREEN;
- } else if (ast_strlen_zero(iaxs[callno]->cid_num) && ast_strlen_zero(iaxs[callno]->cid_name)) {
- iaxs[callno]->calling_pres = AST_PRES_NUMBER_NOT_AVAILABLE;
- } /* else user is allowed to set their own CID settings */
- if (!ast_strlen_zero(user->accountcode))
- ast_string_field_set(iaxs[callno], accountcode, user->accountcode);
- if (!ast_strlen_zero(user->mohinterpret))
- ast_string_field_set(iaxs[callno], mohinterpret, user->mohinterpret);
- if (!ast_strlen_zero(user->mohsuggest))
- ast_string_field_set(iaxs[callno], mohsuggest, user->mohsuggest);
- if (!ast_strlen_zero(user->parkinglot))
- ast_string_field_set(iaxs[callno], parkinglot, user->parkinglot);
- if (user->amaflags)
- iaxs[callno]->amaflags = user->amaflags;
- if (!ast_strlen_zero(user->language))
- ast_string_field_set(iaxs[callno], language, user->language);
- ast_copy_flags64(iaxs[callno], user, IAX_NOTRANSFER | IAX_TRANSFERMEDIA | IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- /* Keep this check last */
- if (!ast_strlen_zero(user->dbsecret)) {
- char *family, *key=NULL;
- char buf[80];
- family = ast_strdupa(user->dbsecret);
- key = strchr(family, '/');
- if (key) {
- *key = '\0';
- key++;
- }
- if (!key || ast_db_get(family, key, buf, sizeof(buf)))
- ast_log(LOG_WARNING, "Unable to retrieve database password for family/key '%s'!\n", user->dbsecret);
- else
- ast_string_field_set(iaxs[callno], secret, buf);
- } else
- ast_string_field_set(iaxs[callno], secret, user->secret);
- res = 0;
- user = user_unref(user);
- } else {
- /* user was not found, but we should still fake an AUTHREQ.
- * Set authmethods to the last known authmethod used by the system
- * Set a fake secret, it's not looked at, just required to attempt authentication.
- * Set authrej so the AUTHREP is rejected without even looking at its contents */
- iaxs[callno]->authmethods = last_authmethod ? last_authmethod : (IAX_AUTH_MD5 | IAX_AUTH_PLAINTEXT);
- ast_string_field_set(iaxs[callno], secret, "badsecret");
- iaxs[callno]->authrej = 1;
- if (!ast_strlen_zero(iaxs[callno]->username)) {
- /* only send the AUTHREQ if a username was specified. */
- res = 0;
- }
- }
- ast_set2_flag64(iaxs[callno], iax2_getpeertrunk(*sin), IAX_TRUNK);
- return res;
- }
- static int raw_hangup(struct sockaddr_in *sin, unsigned short src, unsigned short dst, int sockfd)
- {
- struct ast_iax2_full_hdr fh;
- fh.scallno = htons(src | IAX_FLAG_FULL);
- fh.dcallno = htons(dst);
- fh.ts = 0;
- fh.oseqno = 0;
- fh.iseqno = 0;
- fh.type = AST_FRAME_IAX;
- fh.csub = compress_subclass(IAX_COMMAND_INVAL);
- iax_outputframe(NULL, &fh, 0, sin, 0);
- ast_debug(1, "Raw Hangup %s:%d, src=%d, dst=%d\n",
- ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port), src, dst);
- return sendto(sockfd, &fh, sizeof(fh), 0, (struct sockaddr *)sin, sizeof(*sin));
- }
- static void merge_encryption(struct chan_iax2_pvt *p, unsigned int enc)
- {
- /* Select exactly one common encryption if there are any */
- p->encmethods &= enc;
- if (p->encmethods) {
- if (!(p->encmethods & IAX_ENCRYPT_KEYROTATE)){ /* if key rotation is not supported, turn off keyrotation. */
- p->keyrotateid = -2;
- }
- if (p->encmethods & IAX_ENCRYPT_AES128)
- p->encmethods = IAX_ENCRYPT_AES128;
- else
- p->encmethods = 0;
- }
- }
- /*!
- * \pre iaxsl[call_num] is locked
- *
- * \note Since this function calls send_command_final(), the pvt struct for the given
- * call number may disappear while executing this function.
- */
- static int authenticate_request(int call_num)
- {
- struct iax_ie_data ied;
- int res = -1, authreq_restrict = 0;
- char challenge[10];
- struct chan_iax2_pvt *p = iaxs[call_num];
- memset(&ied, 0, sizeof(ied));
- /* If an AUTHREQ restriction is in place, make sure we can send an AUTHREQ back */
- if (ast_test_flag64(p, IAX_MAXAUTHREQ)) {
- struct iax2_user *user;
- user = ao2_find(users, p->username, OBJ_KEY);
- if (user) {
- if (user->curauthreq == user->maxauthreq)
- authreq_restrict = 1;
- else
- user->curauthreq++;
- user = user_unref(user);
- }
- }
- /* If the AUTHREQ limit test failed, send back an error */
- if (authreq_restrict) {
- iax_ie_append_str(&ied, IAX_IE_CAUSE, "Unauthenticated call limit reached");
- iax_ie_append_byte(&ied, IAX_IE_CAUSECODE, AST_CAUSE_CALL_REJECTED);
- send_command_final(p, AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied.buf, ied.pos, -1);
- return 0;
- }
- iax_ie_append_short(&ied, IAX_IE_AUTHMETHODS, p->authmethods);
- if (p->authmethods & (IAX_AUTH_MD5 | IAX_AUTH_RSA)) {
- snprintf(challenge, sizeof(challenge), "%d", (int)ast_random());
- ast_string_field_set(p, challenge, challenge);
- /* snprintf(p->challenge, sizeof(p->challenge), "%d", (int)ast_random()); */
- iax_ie_append_str(&ied, IAX_IE_CHALLENGE, p->challenge);
- }
- if (p->encmethods)
- iax_ie_append_short(&ied, IAX_IE_ENCRYPTION, p->encmethods);
- iax_ie_append_str(&ied,IAX_IE_USERNAME, p->username);
- res = send_command(p, AST_FRAME_IAX, IAX_COMMAND_AUTHREQ, 0, ied.buf, ied.pos, -1);
- if (p->encmethods)
- ast_set_flag64(p, IAX_ENCRYPTED);
- return res;
- }
- static int authenticate_verify(struct chan_iax2_pvt *p, struct iax_ies *ies)
- {
- char requeststr[256];
- char md5secret[256] = "";
- char secret[256] = "";
- char rsasecret[256] = "";
- int res = -1;
- int x;
- struct iax2_user *user;
- if (p->authrej) {
- return res;
- }
- user = ao2_find(users, p->username, OBJ_KEY);
- if (user) {
- if (ast_test_flag64(p, IAX_MAXAUTHREQ)) {
- ast_atomic_fetchadd_int(&user->curauthreq, -1);
- ast_clear_flag64(p, IAX_MAXAUTHREQ);
- }
- ast_string_field_set(p, host, user->name);
- user = user_unref(user);
- }
- if (ast_test_flag64(p, IAX_FORCE_ENCRYPT) && !p->encmethods) {
- ast_log(LOG_NOTICE, "Call Terminated, Incoming call is unencrypted while force encrypt is enabled.\n");
- return res;
- }
- if (!ast_test_flag(&p->state, IAX_STATE_AUTHENTICATED))
- return res;
- if (ies->password)
- ast_copy_string(secret, ies->password, sizeof(secret));
- if (ies->md5_result)
- ast_copy_string(md5secret, ies->md5_result, sizeof(md5secret));
- if (ies->rsa_result)
- ast_copy_string(rsasecret, ies->rsa_result, sizeof(rsasecret));
- if ((p->authmethods & IAX_AUTH_RSA) && !ast_strlen_zero(rsasecret) && !ast_strlen_zero(p->inkeys)) {
- struct ast_key *key;
- char *keyn;
- char tmpkey[256];
- char *stringp=NULL;
- ast_copy_string(tmpkey, p->inkeys, sizeof(tmpkey));
- stringp=tmpkey;
- keyn = strsep(&stringp, ":");
- while(keyn) {
- key = ast_key_get(keyn, AST_KEY_PUBLIC);
- if (key && !ast_check_signature(key, p->challenge, rsasecret)) {
- res = 0;
- break;
- } else if (!key)
- ast_log(LOG_WARNING, "requested inkey '%s' for RSA authentication does not exist\n", keyn);
- keyn = strsep(&stringp, ":");
- }
- } else if (p->authmethods & IAX_AUTH_MD5) {
- struct MD5Context md5;
- unsigned char digest[16];
- char *tmppw, *stringp;
-
- tmppw = ast_strdupa(p->secret);
- stringp = tmppw;
- while((tmppw = strsep(&stringp, ";"))) {
- MD5Init(&md5);
- MD5Update(&md5, (unsigned char *)p->challenge, strlen(p->challenge));
- MD5Update(&md5, (unsigned char *)tmppw, strlen(tmppw));
- MD5Final(digest, &md5);
- /* If they support md5, authenticate with it. */
- for (x=0;x<16;x++)
- sprintf(requeststr + (x << 1), "%02hhx", digest[x]); /* safe */
- if (!strcasecmp(requeststr, md5secret)) {
- res = 0;
- break;
- }
- }
- } else if (p->authmethods & IAX_AUTH_PLAINTEXT) {
- if (!strcmp(secret, p->secret))
- res = 0;
- }
- return res;
- }
- /*! \brief Verify inbound registration */
- static int register_verify(int callno, struct sockaddr_in *sin, struct iax_ies *ies)
- {
- char requeststr[256] = "";
- char peer[256] = "";
- char md5secret[256] = "";
- char rsasecret[256] = "";
- char secret[256] = "";
- struct iax2_peer *p = NULL;
- struct ast_key *key;
- char *keyn;
- int x;
- int expire = 0;
- int res = -1;
- struct ast_sockaddr addr;
- ast_clear_flag(&iaxs[callno]->state, IAX_STATE_AUTHENTICATED);
- /* iaxs[callno]->peer[0] = '\0'; not necc. any more-- stringfield is pre-inited to null string */
- if (ies->username)
- ast_copy_string(peer, ies->username, sizeof(peer));
- if (ies->password)
- ast_copy_string(secret, ies->password, sizeof(secret));
- if (ies->md5_result)
- ast_copy_string(md5secret, ies->md5_result, sizeof(md5secret));
- if (ies->rsa_result)
- ast_copy_string(rsasecret, ies->rsa_result, sizeof(rsasecret));
- if (ies->refresh)
- expire = ies->refresh;
- if (ast_strlen_zero(peer)) {
- ast_log(LOG_NOTICE, "Empty registration from %s\n", ast_inet_ntoa(sin->sin_addr));
- return -1;
- }
- /* SLD: first call to lookup peer during registration */
- ast_mutex_unlock(&iaxsl[callno]);
- p = find_peer(peer, 1);
- ast_mutex_lock(&iaxsl[callno]);
- if (!p || !iaxs[callno]) {
- if (iaxs[callno]) {
- int plaintext = ((last_authmethod & IAX_AUTH_PLAINTEXT) | (iaxs[callno]->authmethods & IAX_AUTH_PLAINTEXT));
- /* Anything, as long as it's non-blank */
- ast_string_field_set(iaxs[callno], secret, "badsecret");
- /* An AUTHREQ must be sent in response to a REGREQ of an invalid peer unless
- * 1. A challenge already exists indicating a AUTHREQ was already sent out.
- * 2. A plaintext secret is present in ie as result of a previous AUTHREQ requesting it.
- * 3. A plaintext secret is present in the ie and the last_authmethod used by a peer happened
- * to be plaintext, indicating it is an authmethod used by other peers on the system.
- *
- * If none of these cases exist, res will be returned as 0 without authentication indicating
- * an AUTHREQ needs to be sent out. */
- if (ast_strlen_zero(iaxs[callno]->challenge) &&
- !(!ast_strlen_zero(secret) && plaintext)) {
- /* by setting res to 0, an REGAUTH will be sent */
- res = 0;
- }
- }
- if (authdebug && !p)
- ast_log(LOG_NOTICE, "No registration for peer '%s' (from %s)\n", peer, ast_inet_ntoa(sin->sin_addr));
- goto return_unref;
- }
- if (!ast_test_flag64(p, IAX_DYNAMIC)) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Peer '%s' is not dynamic (from %s)\n", peer, ast_inet_ntoa(sin->sin_addr));
- goto return_unref;
- }
- ast_sockaddr_from_sin(&addr, sin);
- if (!ast_apply_acl(p->acl, &addr, "IAX2 Peer ACL: ")) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Host %s denied access to register peer '%s'\n", ast_inet_ntoa(sin->sin_addr), p->name);
- goto return_unref;
- }
- ast_string_field_set(iaxs[callno], secret, p->secret);
- ast_string_field_set(iaxs[callno], inkeys, p->inkeys);
- /* Check secret against what we have on file */
- if (!ast_strlen_zero(rsasecret) && (p->authmethods & IAX_AUTH_RSA) && !ast_strlen_zero(iaxs[callno]->challenge)) {
- if (!ast_strlen_zero(p->inkeys)) {
- char tmpkeys[256];
- char *stringp=NULL;
- ast_copy_string(tmpkeys, p->inkeys, sizeof(tmpkeys));
- stringp=tmpkeys;
- keyn = strsep(&stringp, ":");
- while(keyn) {
- key = ast_key_get(keyn, AST_KEY_PUBLIC);
- if (key && !ast_check_signature(key, iaxs[callno]->challenge, rsasecret)) {
- ast_set_flag(&iaxs[callno]->state, IAX_STATE_AUTHENTICATED);
- break;
- } else if (!key)
- ast_log(LOG_WARNING, "requested inkey '%s' does not exist\n", keyn);
- keyn = strsep(&stringp, ":");
- }
- if (!keyn) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Host %s failed RSA authentication with inkeys '%s'\n", peer, p->inkeys);
- goto return_unref;
- }
- } else {
- if (authdebug)
- ast_log(LOG_NOTICE, "Host '%s' trying to do RSA authentication, but we have no inkeys\n", peer);
- goto return_unref;
- }
- } else if (!ast_strlen_zero(md5secret) && (p->authmethods & IAX_AUTH_MD5) && !ast_strlen_zero(iaxs[callno]->challenge)) {
- struct MD5Context md5;
- unsigned char digest[16];
- char *tmppw, *stringp;
- tmppw = ast_strdupa(p->secret);
- stringp = tmppw;
- while((tmppw = strsep(&stringp, ";"))) {
- MD5Init(&md5);
- MD5Update(&md5, (unsigned char *)iaxs[callno]->challenge, strlen(iaxs[callno]->challenge));
- MD5Update(&md5, (unsigned char *)tmppw, strlen(tmppw));
- MD5Final(digest, &md5);
- for (x=0;x<16;x++)
- sprintf(requeststr + (x << 1), "%02hhx", digest[x]); /* safe */
- if (!strcasecmp(requeststr, md5secret))
- break;
- }
- if (tmppw) {
- ast_set_flag(&iaxs[callno]->state, IAX_STATE_AUTHENTICATED);
- } else {
- if (authdebug)
- ast_log(LOG_NOTICE, "Host %s failed MD5 authentication for '%s' (%s != %s)\n", ast_inet_ntoa(sin->sin_addr), p->name, requeststr, md5secret);
- goto return_unref;
- }
- } else if (!ast_strlen_zero(secret) && (p->authmethods & IAX_AUTH_PLAINTEXT)) {
- /* They've provided a plain text password and we support that */
- if (strcmp(secret, p->secret)) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Host %s did not provide proper plaintext password for '%s'\n", ast_inet_ntoa(sin->sin_addr), p->name);
- goto return_unref;
- } else
- ast_set_flag(&iaxs[callno]->state, IAX_STATE_AUTHENTICATED);
- } else if (!ast_strlen_zero(iaxs[callno]->challenge) && ast_strlen_zero(md5secret) && ast_strlen_zero(rsasecret)) {
- /* if challenge has been sent, but no challenge response if given, reject. */
- goto return_unref;
- }
- ast_devstate_changed(AST_DEVICE_UNKNOWN, AST_DEVSTATE_CACHABLE, "IAX2/%s", p->name); /* Activate notification */
- /* either Authentication has taken place, or a REGAUTH must be sent before verifying registration */
- res = 0;
- return_unref:
- if (iaxs[callno]) {
- ast_string_field_set(iaxs[callno], peer, peer);
- /* Choose lowest expiry number */
- if (expire && (expire < iaxs[callno]->expiry)) {
- iaxs[callno]->expiry = expire;
- }
- }
- if (p) {
- peer_unref(p);
- }
- return res;
- }
- static int authenticate(const char *challenge, const char *secret, const char *keyn, int authmethods, struct iax_ie_data *ied, struct sockaddr_in *sin, struct chan_iax2_pvt *pvt)
- {
- int res = -1;
- int x;
- if (!ast_strlen_zero(keyn)) {
- if (!(authmethods & IAX_AUTH_RSA)) {
- if (ast_strlen_zero(secret))
- ast_log(LOG_NOTICE, "Asked to authenticate to %s with an RSA key, but they don't allow RSA authentication\n", ast_inet_ntoa(sin->sin_addr));
- } else if (ast_strlen_zero(challenge)) {
- ast_log(LOG_NOTICE, "No challenge provided for RSA authentication to %s\n", ast_inet_ntoa(sin->sin_addr));
- } else {
- char sig[256];
- struct ast_key *key;
- key = ast_key_get(keyn, AST_KEY_PRIVATE);
- if (!key) {
- ast_log(LOG_NOTICE, "Unable to find private key '%s'\n", keyn);
- } else {
- if (ast_sign(key, (char*)challenge, sig)) {
- ast_log(LOG_NOTICE, "Unable to sign challenge with key\n");
- res = -1;
- } else {
- iax_ie_append_str(ied, IAX_IE_RSA_RESULT, sig);
- res = 0;
- }
- }
- }
- }
- /* Fall back */
- if (res && !ast_strlen_zero(secret)) {
- if ((authmethods & IAX_AUTH_MD5) && !ast_strlen_zero(challenge)) {
- struct MD5Context md5;
- unsigned char digest[16];
- char digres[128];
- MD5Init(&md5);
- MD5Update(&md5, (unsigned char *)challenge, strlen(challenge));
- MD5Update(&md5, (unsigned char *)secret, strlen(secret));
- MD5Final(digest, &md5);
- /* If they support md5, authenticate with it. */
- for (x=0;x<16;x++)
- sprintf(digres + (x << 1), "%02hhx", digest[x]); /* safe */
- if (pvt) {
- build_encryption_keys(digest, pvt);
- }
- iax_ie_append_str(ied, IAX_IE_MD5_RESULT, digres);
- res = 0;
- } else if (authmethods & IAX_AUTH_PLAINTEXT) {
- iax_ie_append_str(ied, IAX_IE_PASSWORD, secret);
- res = 0;
- } else
- ast_log(LOG_NOTICE, "No way to send secret to peer '%s' (their methods: %d)\n", ast_inet_ntoa(sin->sin_addr), authmethods);
- }
- return res;
- }
- /*!
- * \note This function calls realtime_peer -> reg_source_db -> iax2_poke_peer -> find_callno,
- * so do not call this function with a pvt lock held.
- */
- static int authenticate_reply(struct chan_iax2_pvt *p, struct sockaddr_in *sin, struct iax_ies *ies, const char *override, const char *okey)
- {
- struct iax2_peer *peer = NULL;
- /* Start pessimistic */
- int res = -1;
- int authmethods = 0;
- struct iax_ie_data ied;
- uint16_t callno = p->callno;
- memset(&ied, 0, sizeof(ied));
-
- if (ies->username)
- ast_string_field_set(p, username, ies->username);
- if (ies->challenge)
- ast_string_field_set(p, challenge, ies->challenge);
- if (ies->authmethods)
- authmethods = ies->authmethods;
- if (authmethods & IAX_AUTH_MD5)
- merge_encryption(p, ies->encmethods);
- else
- p->encmethods = 0;
- /* Check for override RSA authentication first */
- if (!ast_strlen_zero(override) || !ast_strlen_zero(okey)) {
- /* Normal password authentication */
- res = authenticate(p->challenge, override, okey, authmethods, &ied, sin, p);
- } else {
- struct ao2_iterator i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- struct sockaddr_in peer_addr;
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- if ((ast_strlen_zero(p->peer) || !strcmp(p->peer, peer->name))
- /* No peer specified at our end, or this is the peer */
- && (ast_strlen_zero(peer->username) || (!strcmp(peer->username, p->username)))
- /* No username specified in peer rule, or this is the right username */
- && (!peer_addr.sin_addr.s_addr || ((sin->sin_addr.s_addr & peer->mask.s_addr) == (peer_addr.sin_addr.s_addr & peer->mask.s_addr)))
- /* No specified host, or this is our host */
- ) {
- res = authenticate(p->challenge, peer->secret, peer->outkey, authmethods, &ied, sin, p);
- if (!res) {
- peer_unref(peer);
- break;
- }
- }
- peer_unref(peer);
- }
- ao2_iterator_destroy(&i);
- if (!peer) {
- /* We checked our list and didn't find one. It's unlikely, but possible,
- that we're trying to authenticate *to* a realtime peer */
- const char *peer_name = ast_strdupa(p->peer);
- ast_mutex_unlock(&iaxsl[callno]);
- if ((peer = realtime_peer(peer_name, NULL))) {
- ast_mutex_lock(&iaxsl[callno]);
- if (!(p = iaxs[callno])) {
- peer_unref(peer);
- return -1;
- }
- res = authenticate(p->challenge, peer->secret,peer->outkey, authmethods, &ied, sin, p);
- peer_unref(peer);
- }
- if (!peer) {
- ast_mutex_lock(&iaxsl[callno]);
- if (!(p = iaxs[callno]))
- return -1;
- }
- }
- }
- if (ies->encmethods) {
- ast_set_flag64(p, IAX_ENCRYPTED | IAX_KEYPOPULATED);
- } else if (ast_test_flag64(iaxs[callno], IAX_FORCE_ENCRYPT)) {
- ast_log(LOG_NOTICE, "Call initiated without encryption while forceencryption=yes option is set\n");
- return -1; /* if force encryption is yes, and no encryption methods, then return -1 to hangup */
- }
- if (!res) {
- struct ast_datastore *variablestore;
- struct ast_variable *var, *prev = NULL;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- varlist = ast_calloc(1, sizeof(*varlist));
- variablestore = ast_datastore_alloc(&iax2_variable_datastore_info, NULL);
- if (variablestore && varlist && p->owner) {
- variablestore->data = varlist;
- variablestore->inheritance = DATASTORE_INHERIT_FOREVER;
- AST_LIST_HEAD_INIT(varlist);
- for (var = ies->vars; var; var = var->next) {
- struct ast_var_t *newvar = ast_var_assign(var->name, var->value);
- if (prev)
- ast_free(prev);
- prev = var;
- if (!newvar) {
- /* Don't abort list traversal, as this would leave ies->vars in an inconsistent state. */
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- } else {
- AST_LIST_INSERT_TAIL(varlist, newvar, entries);
- }
- }
- if (prev)
- ast_free(prev);
- ies->vars = NULL;
- ast_channel_datastore_add(p->owner, variablestore);
- } else {
- if (p->owner)
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- if (variablestore)
- ast_datastore_free(variablestore);
- if (varlist)
- ast_free(varlist);
- }
- }
- if (!res)
- res = send_command(p, AST_FRAME_IAX, IAX_COMMAND_AUTHREP, 0, ied.buf, ied.pos, -1);
- return res;
- }
- static int iax2_do_register(struct iax2_registry *reg);
- static void __iax2_do_register_s(const void *data)
- {
- struct iax2_registry *reg = (struct iax2_registry *)data;
- if (ast_sockaddr_isnull(®->addr)) {
- reg->addr.ss.ss_family = AST_AF_UNSPEC;
- ast_dnsmgr_lookup(reg->hostname, ®->addr, ®->dnsmgr, srvlookup ? "_iax._udp" : NULL);
- if (!ast_sockaddr_port(®->addr)) {
- ast_sockaddr_set_port(®->addr, reg->port);
- } else {
- reg->port = ast_sockaddr_port(®->addr);
- }
- }
- reg->expire = -1;
- iax2_do_register(reg);
- }
- static int iax2_do_register_s(const void *data)
- {
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__iax2_do_register_s, data))
- #endif
- __iax2_do_register_s(data);
- return 0;
- }
- static int try_transfer(struct chan_iax2_pvt *pvt, struct iax_ies *ies)
- {
- int newcall = 0;
- char newip[256];
- struct iax_ie_data ied;
- struct sockaddr_in new = { 0, };
- memset(&ied, 0, sizeof(ied));
- if (ies->apparent_addr)
- memmove(&new, ies->apparent_addr, sizeof(new));
- if (ies->callno)
- newcall = ies->callno;
- if (!newcall || !new.sin_addr.s_addr || !new.sin_port) {
- ast_log(LOG_WARNING, "Invalid transfer request\n");
- return -1;
- }
- pvt->transfercallno = newcall;
- memcpy(&pvt->transfer, &new, sizeof(pvt->transfer));
- inet_aton(newip, &pvt->transfer.sin_addr);
- pvt->transfer.sin_family = AF_INET;
- pvt->transferid = ies->transferid;
- /* only store by transfercallno if this is a new transfer,
- * just in case we get a duplicate TXREQ */
- if (pvt->transferring == TRANSFER_NONE) {
- store_by_transfercallno(pvt);
- }
- pvt->transferring = TRANSFER_BEGIN;
- if (ies->transferid)
- iax_ie_append_int(&ied, IAX_IE_TRANSFERID, ies->transferid);
- send_command_transfer(pvt, AST_FRAME_IAX, IAX_COMMAND_TXCNT, 0, ied.buf, ied.pos);
- return 0;
- }
- static int complete_dpreply(struct chan_iax2_pvt *pvt, struct iax_ies *ies)
- {
- char exten[256] = "";
- int status = CACHE_FLAG_UNKNOWN, expiry = iaxdefaultdpcache, x, matchmore = 0;
- struct iax2_dpcache *dp = NULL;
-
- if (ies->called_number)
- ast_copy_string(exten, ies->called_number, sizeof(exten));
-
- if (ies->dpstatus & IAX_DPSTATUS_EXISTS)
- status = CACHE_FLAG_EXISTS;
- else if (ies->dpstatus & IAX_DPSTATUS_CANEXIST)
- status = CACHE_FLAG_CANEXIST;
- else if (ies->dpstatus & IAX_DPSTATUS_NONEXISTENT)
- status = CACHE_FLAG_NONEXISTENT;
- if (ies->refresh)
- expiry = ies->refresh;
- if (ies->dpstatus & IAX_DPSTATUS_MATCHMORE)
- matchmore = CACHE_FLAG_MATCHMORE;
-
- AST_LIST_LOCK(&dpcache);
- AST_LIST_TRAVERSE_SAFE_BEGIN(&dpcache, dp, peer_list) {
- if (strcmp(dp->exten, exten))
- continue;
- AST_LIST_REMOVE_CURRENT(peer_list);
- dp->callno = 0;
- dp->expiry.tv_sec = dp->orig.tv_sec + expiry;
- if (dp->flags & CACHE_FLAG_PENDING) {
- dp->flags &= ~CACHE_FLAG_PENDING;
- dp->flags |= status;
- dp->flags |= matchmore;
- }
- /* Wake up waiters */
- for (x = 0; x < ARRAY_LEN(dp->waiters); x++) {
- if (dp->waiters[x] > -1) {
- if (write(dp->waiters[x], "asdf", 4) < 0) {
- }
- }
- }
- }
- AST_LIST_TRAVERSE_SAFE_END;
- AST_LIST_UNLOCK(&dpcache);
- return 0;
- }
- static int complete_transfer(int callno, struct iax_ies *ies)
- {
- int peercallno = 0;
- struct chan_iax2_pvt *pvt = iaxs[callno];
- struct iax_frame *cur;
- jb_frame frame;
- if (ies->callno)
- peercallno = ies->callno;
- if (peercallno < 1) {
- ast_log(LOG_WARNING, "Invalid transfer request\n");
- return -1;
- }
- remove_by_transfercallno(pvt);
- /* since a transfer has taken place, the address will change.
- * This must be accounted for in the peercnts table. Remove
- * the old address and add the new one */
- peercnt_remove_by_addr(&pvt->addr);
- peercnt_add(&pvt->transfer);
- /* now copy over the new address */
- memcpy(&pvt->addr, &pvt->transfer, sizeof(pvt->addr));
- memset(&pvt->transfer, 0, sizeof(pvt->transfer));
- /* Reset sequence numbers */
- pvt->oseqno = 0;
- pvt->rseqno = 0;
- pvt->iseqno = 0;
- pvt->aseqno = 0;
- if (pvt->peercallno) {
- remove_by_peercallno(pvt);
- }
- pvt->peercallno = peercallno;
- /*this is where the transfering call swiches hash tables */
- store_by_peercallno(pvt);
- pvt->transferring = TRANSFER_NONE;
- pvt->svoiceformat = -1;
- pvt->voiceformat = 0;
- pvt->svideoformat = -1;
- pvt->videoformat = 0;
- pvt->transfercallno = 0;
- memset(&pvt->rxcore, 0, sizeof(pvt->rxcore));
- memset(&pvt->offset, 0, sizeof(pvt->offset));
- /* reset jitterbuffer */
- while(jb_getall(pvt->jb,&frame) == JB_OK)
- iax2_frame_free(frame.data);
- jb_reset(pvt->jb);
- pvt->lag = 0;
- pvt->last = 0;
- pvt->lastsent = 0;
- pvt->nextpred = 0;
- pvt->pingtime = DEFAULT_RETRY_TIME;
- AST_LIST_TRAVERSE(&frame_queue[callno], cur, list) {
- /* We must cancel any packets that would have been transmitted
- because now we're talking to someone new. It's okay, they
- were transmitted to someone that didn't care anyway. */
- cur->retries = -1;
- }
- return 0;
- }
- /*! \brief Acknowledgment received for OUR registration */
- static int iax2_ack_registry(struct iax_ies *ies, struct sockaddr_in *sin, int callno)
- {
- struct iax2_registry *reg;
- /* Start pessimistic */
- char peer[256] = "";
- char msgstatus[60];
- int refresh = 60;
- char ourip[256] = "<Unspecified>";
- struct sockaddr_in oldus;
- struct sockaddr_in us;
- int oldmsgs;
- struct sockaddr_in reg_addr;
- memset(&us, 0, sizeof(us));
- if (ies->apparent_addr) {
- memmove(&us, ies->apparent_addr, sizeof(us));
- }
- if (ies->username) {
- ast_copy_string(peer, ies->username, sizeof(peer));
- }
- if (ies->refresh) {
- refresh = ies->refresh;
- }
- if (ies->calling_number) {
- /* We don't do anything with it really, but maybe we should */
- }
- reg = iaxs[callno]->reg;
- if (!reg) {
- ast_log(LOG_WARNING, "Registry acknowledge on unknown registry '%s'\n", peer);
- return -1;
- }
- memcpy(&oldus, ®->us, sizeof(oldus));
- oldmsgs = reg->messages;
- ast_sockaddr_to_sin(®->addr, ®_addr);
- if (inaddrcmp(®_addr, sin)) {
- ast_log(LOG_WARNING, "Received unsolicited registry ack from '%s'\n", ast_inet_ntoa(sin->sin_addr));
- return -1;
- }
- memcpy(®->us, &us, sizeof(reg->us));
- if (ies->msgcount >= 0) {
- reg->messages = ies->msgcount & 0xffff; /* only low 16 bits are used in the transmission of the IE */
- }
- /* always refresh the registration at the interval requested by the server
- we are registering to
- */
- reg->refresh = refresh;
- reg->expire = iax2_sched_replace(reg->expire, sched,
- (5 * reg->refresh / 6) * 1000, iax2_do_register_s, reg);
- if (inaddrcmp(&oldus, ®->us) || (reg->messages != oldmsgs)) {
- if (reg->messages > 255) {
- snprintf(msgstatus, sizeof(msgstatus), " with %d new and %d old messages waiting", reg->messages & 0xff, reg->messages >> 8);
- } else if (reg->messages > 1) {
- snprintf(msgstatus, sizeof(msgstatus), " with %d new messages waiting", reg->messages);
- } else if (reg->messages > 0) {
- ast_copy_string(msgstatus, " with 1 new message waiting", sizeof(msgstatus));
- } else {
- ast_copy_string(msgstatus, " with no messages waiting", sizeof(msgstatus));
- }
- snprintf(ourip, sizeof(ourip), "%s:%d", ast_inet_ntoa(reg->us.sin_addr), ntohs(reg->us.sin_port));
- ast_verb(3, "Registered IAX2 to '%s', who sees us as %s%s\n", ast_inet_ntoa(sin->sin_addr), ourip, msgstatus);
- manager_event(EVENT_FLAG_SYSTEM, "Registry", "ChannelType: IAX2\r\nDomain: %s\r\nStatus: Registered\r\n", ast_inet_ntoa(sin->sin_addr));
- }
- reg->regstate = REG_STATE_REGISTERED;
- return 0;
- }
- static int iax2_append_register(const char *hostname, const char *username,
- const char *secret, const char *porta)
- {
- struct iax2_registry *reg;
- if (!(reg = ast_calloc(1, sizeof(*reg) + strlen(hostname) + 1))) {
- return -1;
- }
- reg->addr.ss.ss_family = AF_INET;
- if (ast_dnsmgr_lookup(hostname, ®->addr, ®->dnsmgr, srvlookup ? "_iax._udp" : NULL) < 0) {
- ast_free(reg);
- return -1;
- }
- ast_copy_string(reg->username, username, sizeof(reg->username));
- strcpy(reg->hostname, hostname); /* Note: This is safe */
- if (secret) {
- ast_copy_string(reg->secret, secret, sizeof(reg->secret));
- }
- reg->expire = -1;
- reg->refresh = IAX_DEFAULT_REG_EXPIRE;
- reg->port = ast_sockaddr_port(®->addr);
- if (!porta && !reg->port) {
- reg->port = IAX_DEFAULT_PORTNO;
- } else if (porta) {
- sscanf(porta, "%5d", ®->port);
- }
- ast_sockaddr_set_port(®->addr, reg->port);
- AST_LIST_LOCK(®istrations);
- AST_LIST_INSERT_HEAD(®istrations, reg, entry);
- AST_LIST_UNLOCK(®istrations);
- return 0;
- }
- static int iax2_register(const char *value, int lineno)
- {
- char copy[256];
- char *username, *hostname, *secret;
- char *porta;
- char *stringp=NULL;
-
- if (!value)
- return -1;
- ast_copy_string(copy, value, sizeof(copy));
- stringp = copy;
- username = strsep(&stringp, "@");
- hostname = strsep(&stringp, "@");
- if (!hostname) {
- ast_log(LOG_WARNING, "Format for registration is user[:secret]@host[:port] at line %d\n", lineno);
- return -1;
- }
- stringp = username;
- username = strsep(&stringp, ":");
- secret = strsep(&stringp, ":");
- stringp = hostname;
- hostname = strsep(&stringp, ":");
- porta = strsep(&stringp, ":");
-
- if (porta && !atoi(porta)) {
- ast_log(LOG_WARNING, "%s is not a valid port number at line %d\n", porta, lineno);
- return -1;
- }
- return iax2_append_register(hostname, username, secret, porta);
- }
- static void register_peer_exten(struct iax2_peer *peer, int onoff)
- {
- char multi[256];
- char *stringp, *ext;
- if (!ast_strlen_zero(regcontext)) {
- ast_copy_string(multi, S_OR(peer->regexten, peer->name), sizeof(multi));
- stringp = multi;
- while((ext = strsep(&stringp, "&"))) {
- if (onoff) {
- if (!ast_exists_extension(NULL, regcontext, ext, 1, NULL))
- ast_add_extension(regcontext, 1, ext, 1, NULL, NULL,
- "Noop", ast_strdup(peer->name), ast_free_ptr, "IAX2");
- } else
- ast_context_remove_extension(regcontext, ext, 1, NULL);
- }
- }
- }
- static void prune_peers(void);
- static void unlink_peer(struct iax2_peer *peer)
- {
- if (peer->expire > -1) {
- if (!AST_SCHED_DEL(sched, peer->expire)) {
- peer->expire = -1;
- peer_unref(peer);
- }
- }
- if (peer->pokeexpire > -1) {
- if (!AST_SCHED_DEL(sched, peer->pokeexpire)) {
- peer->pokeexpire = -1;
- peer_unref(peer);
- }
- }
- ao2_unlink(peers, peer);
- }
- static void __expire_registry(const void *data)
- {
- struct iax2_peer *peer = (struct iax2_peer *) data;
- if (!peer)
- return;
- if (peer->expire == -1) {
- /* Removed already (possibly through CLI), ignore */
- return;
- }
- peer->expire = -1;
- ast_debug(1, "Expiring registration for peer '%s'\n", peer->name);
- if (ast_test_flag64((&globalflags), IAX_RTUPDATE) && (ast_test_flag64(peer, IAX_TEMPONLY|IAX_RTCACHEFRIENDS)))
- realtime_update_peer(peer->name, &peer->addr, 0);
- manager_event(EVENT_FLAG_SYSTEM, "PeerStatus", "ChannelType: IAX2\r\nPeer: IAX2/%s\r\nPeerStatus: Unregistered\r\nCause: Expired\r\n", peer->name);
- /* modify entry in peercnts table as _not_ registered */
- peercnt_modify((unsigned char) 0, 0, &peer->addr);
- /* Reset the address */
- memset(&peer->addr, 0, sizeof(peer->addr));
- /* Reset expiry value */
- peer->expiry = min_reg_expire;
- if (!ast_test_flag64(peer, IAX_TEMPONLY))
- ast_db_del("IAX/Registry", peer->name);
- register_peer_exten(peer, 0);
- ast_devstate_changed(AST_DEVICE_UNAVAILABLE, AST_DEVSTATE_CACHABLE, "IAX2/%s", peer->name); /* Activate notification */
- if (iax2_regfunk)
- iax2_regfunk(peer->name, 0);
- if (ast_test_flag64(peer, IAX_RTAUTOCLEAR))
- unlink_peer(peer);
- peer_unref(peer);
- }
- static int expire_registry(const void *data)
- {
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__expire_registry, data))
- #endif
- __expire_registry(data);
- return 0;
- }
- static void reg_source_db(struct iax2_peer *p)
- {
- char data[80];
- char *expiry;
- if (ast_test_flag64(p, IAX_TEMPONLY) || ast_db_get("IAX/Registry", p->name, data, sizeof(data))) {
- return;
- }
- expiry = strrchr(data, ':');
- if (!expiry) {
- ast_log(LOG_NOTICE, "IAX/Registry astdb entry missing expiry: '%s'\n", data);
- return;
- }
- *expiry++ = '\0';
- if (!ast_sockaddr_parse(&p->addr, data, PARSE_PORT_REQUIRE)) {
- ast_log(LOG_NOTICE, "IAX/Registry astdb host:port invalid - '%s'\n", data);
- return;
- }
- p->expiry = atoi(expiry);
- ast_verb(3, "Seeding '%s' at %s for %d\n", p->name,
- ast_sockaddr_stringify(&p->addr), p->expiry);
- iax2_poke_peer(p, 0);
- if (p->expire > -1) {
- if (!AST_SCHED_DEL(sched, p->expire)) {
- p->expire = -1;
- peer_unref(p);
- }
- }
- ast_devstate_changed(AST_DEVICE_UNKNOWN, AST_DEVSTATE_CACHABLE, "IAX2/%s", p->name); /* Activate notification */
- p->expire = iax2_sched_add(sched, (p->expiry + 10) * 1000, expire_registry, peer_ref(p));
- if (p->expire == -1) {
- peer_unref(p);
- }
- if (iax2_regfunk) {
- iax2_regfunk(p->name, 1);
- }
- register_peer_exten(p, 1);
- }
- /*!
- * \pre iaxsl[callno] is locked
- *
- * \note Since this function calls send_command_final(), the pvt struct for
- * the given call number may disappear while executing this function.
- */
- static int update_registry(struct sockaddr_in *sin, int callno, char *devtype, int fd, unsigned short refresh)
- {
- /* Called from IAX thread only, with proper iaxsl lock */
- struct iax_ie_data ied = {
- .pos = 0,
- };
- struct iax2_peer *p;
- int msgcount;
- char data[80];
- int version;
- const char *peer_name;
- int res = -1;
- struct ast_sockaddr sockaddr;
- ast_sockaddr_from_sin(&sockaddr, sin);
- peer_name = ast_strdupa(iaxs[callno]->peer);
- /* SLD: Another find_peer call during registration - this time when we are really updating our registration */
- ast_mutex_unlock(&iaxsl[callno]);
- if (!(p = find_peer(peer_name, 1))) {
- ast_mutex_lock(&iaxsl[callno]);
- ast_log(LOG_WARNING, "No such peer '%s'\n", peer_name);
- return -1;
- }
- ast_mutex_lock(&iaxsl[callno]);
- if (!iaxs[callno])
- goto return_unref;
- if (ast_test_flag64((&globalflags), IAX_RTUPDATE) && (ast_test_flag64(p, IAX_TEMPONLY|IAX_RTCACHEFRIENDS))) {
- if (sin->sin_addr.s_addr) {
- time_t nowtime;
- time(&nowtime);
- realtime_update_peer(peer_name, &sockaddr, nowtime);
- } else {
- realtime_update_peer(peer_name, &sockaddr, 0);
- }
- }
- /* treat an unspecified refresh interval as the minimum */
- if (!refresh) {
- refresh = min_reg_expire;
- }
- if (refresh > max_reg_expire) {
- ast_log(LOG_NOTICE, "Restricting registration for peer '%s' to %d seconds (requested %d)\n",
- p->name, max_reg_expire, refresh);
- p->expiry = max_reg_expire;
- } else if (refresh < min_reg_expire) {
- ast_log(LOG_NOTICE, "Restricting registration for peer '%s' to %d seconds (requested %d)\n",
- p->name, min_reg_expire, refresh);
- p->expiry = min_reg_expire;
- } else {
- p->expiry = refresh;
- }
- if (ast_sockaddr_cmp(&p->addr, &sockaddr)) {
- if (iax2_regfunk) {
- iax2_regfunk(p->name, 1);
- }
- /* modify entry in peercnts table as _not_ registered */
- peercnt_modify((unsigned char) 0, 0, &p->addr);
- /* Stash the IP address from which they registered */
- ast_sockaddr_from_sin(&p->addr, sin);
- snprintf(data, sizeof(data), "%s:%d:%d", ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port), p->expiry);
- if (!ast_test_flag64(p, IAX_TEMPONLY) && sin->sin_addr.s_addr) {
- ast_db_put("IAX/Registry", p->name, data);
- ast_verb(3, "Registered IAX2 '%s' (%s) at %s:%d\n", p->name,
- ast_test_flag(&iaxs[callno]->state, IAX_STATE_AUTHENTICATED) ? "AUTHENTICATED" : "UNAUTHENTICATED", ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
- manager_event(EVENT_FLAG_SYSTEM, "PeerStatus", "ChannelType: IAX2\r\nPeer: IAX2/%s\r\nPeerStatus: Registered\r\nAddress: %s\r\nPort: %d\r\n", p->name, ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
- register_peer_exten(p, 1);
- ast_devstate_changed(AST_DEVICE_UNKNOWN, AST_DEVSTATE_CACHABLE, "IAX2/%s", p->name); /* Activate notification */
- } else if (!ast_test_flag64(p, IAX_TEMPONLY)) {
- ast_verb(3, "Unregistered IAX2 '%s' (%s)\n", p->name,
- ast_test_flag(&iaxs[callno]->state, IAX_STATE_AUTHENTICATED) ? "AUTHENTICATED" : "UNAUTHENTICATED");
- manager_event(EVENT_FLAG_SYSTEM, "PeerStatus", "ChannelType: IAX2\r\nPeer: IAX2/%s\r\nPeerStatus: Unregistered\r\n", p->name);
- register_peer_exten(p, 0);
- ast_db_del("IAX/Registry", p->name);
- ast_devstate_changed(AST_DEVICE_UNAVAILABLE, AST_DEVSTATE_CACHABLE, "IAX2/%s", p->name); /* Activate notification */
- }
- /* Update the host */
- /* Verify that the host is really there */
- iax2_poke_peer(p, callno);
- }
- /* modify entry in peercnts table as registered */
- if (p->maxcallno) {
- peercnt_modify((unsigned char) 1, p->maxcallno, &p->addr);
- }
- /* Make sure our call still exists, an INVAL at the right point may make it go away */
- if (!iaxs[callno]) {
- res = -1;
- goto return_unref;
- }
- /* Store socket fd */
- p->sockfd = fd;
- /* Setup the expiry */
- if (p->expire > -1) {
- if (!AST_SCHED_DEL(sched, p->expire)) {
- p->expire = -1;
- peer_unref(p);
- }
- }
- if (p->expiry && sin->sin_addr.s_addr) {
- p->expire = iax2_sched_add(sched, (p->expiry + 10) * 1000, expire_registry, peer_ref(p));
- if (p->expire == -1)
- peer_unref(p);
- }
- iax_ie_append_str(&ied, IAX_IE_USERNAME, p->name);
- iax_ie_append_int(&ied, IAX_IE_DATETIME, iax2_datetime(p->zonetag));
- if (sin->sin_addr.s_addr) {
- struct sockaddr_in peer_addr;
- ast_sockaddr_to_sin(&p->addr, &peer_addr);
- iax_ie_append_short(&ied, IAX_IE_REFRESH, p->expiry);
- iax_ie_append_addr(&ied, IAX_IE_APPARENT_ADDR, &peer_addr);
- if (!ast_strlen_zero(p->mailbox)) {
- struct ast_event *event;
- int new, old;
- char *mailbox, *context;
- context = mailbox = ast_strdupa(p->mailbox);
- strsep(&context, "@");
- if (ast_strlen_zero(context))
- context = "default";
- event = ast_event_get_cached(AST_EVENT_MWI,
- AST_EVENT_IE_MAILBOX, AST_EVENT_IE_PLTYPE_STR, mailbox,
- AST_EVENT_IE_CONTEXT, AST_EVENT_IE_PLTYPE_STR, context,
- AST_EVENT_IE_END);
- if (event) {
- new = ast_event_get_ie_uint(event, AST_EVENT_IE_NEWMSGS);
- old = ast_event_get_ie_uint(event, AST_EVENT_IE_OLDMSGS);
- ast_event_destroy(event);
- } else { /* Fall back on checking the mailbox directly */
- ast_app_inboxcount(p->mailbox, &new, &old);
- }
- if (new > 255) {
- new = 255;
- }
- if (old > 255) {
- old = 255;
- }
- msgcount = (old << 8) | new;
- iax_ie_append_short(&ied, IAX_IE_MSGCOUNT, msgcount);
- }
- if (ast_test_flag64(p, IAX_HASCALLERID)) {
- iax_ie_append_str(&ied, IAX_IE_CALLING_NUMBER, p->cid_num);
- iax_ie_append_str(&ied, IAX_IE_CALLING_NAME, p->cid_name);
- }
- }
- version = iax_check_version(devtype);
- if (version)
- iax_ie_append_short(&ied, IAX_IE_FIRMWAREVER, version);
- res = 0;
- return_unref:
- peer_unref(p);
- return res ? res : send_command_final(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_REGACK, 0, ied.buf, ied.pos, -1);
- }
- static int registry_authrequest(int callno)
- {
- struct iax_ie_data ied;
- struct iax2_peer *p;
- char challenge[10];
- const char *peer_name;
- int sentauthmethod;
- peer_name = ast_strdupa(iaxs[callno]->peer);
- /* SLD: third call to find_peer in registration */
- ast_mutex_unlock(&iaxsl[callno]);
- if ((p = find_peer(peer_name, 1))) {
- last_authmethod = p->authmethods;
- }
- ast_mutex_lock(&iaxsl[callno]);
- if (!iaxs[callno])
- goto return_unref;
- memset(&ied, 0, sizeof(ied));
- /* The selection of which delayed reject is sent may leak information,
- * if it sets a static response. For example, if a host is known to only
- * use MD5 authentication, then an RSA response would indicate that the
- * peer does not exist, and vice-versa.
- * Therefore, we use whatever the last peer used (which may vary over the
- * course of a server, which should leak minimal information). */
- sentauthmethod = p ? p->authmethods : last_authmethod ? last_authmethod : (IAX_AUTH_MD5 | IAX_AUTH_PLAINTEXT);
- if (!p) {
- iaxs[callno]->authmethods = sentauthmethod;
- }
- iax_ie_append_short(&ied, IAX_IE_AUTHMETHODS, sentauthmethod);
- if (sentauthmethod & (IAX_AUTH_RSA | IAX_AUTH_MD5)) {
- /* Build the challenge */
- snprintf(challenge, sizeof(challenge), "%d", (int)ast_random());
- ast_string_field_set(iaxs[callno], challenge, challenge);
- iax_ie_append_str(&ied, IAX_IE_CHALLENGE, iaxs[callno]->challenge);
- }
- iax_ie_append_str(&ied, IAX_IE_USERNAME, peer_name);
- return_unref:
- if (p) {
- peer_unref(p);
- }
- return iaxs[callno] ? send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_REGAUTH, 0, ied.buf, ied.pos, -1) : -1;
- }
- static int registry_rerequest(struct iax_ies *ies, int callno, struct sockaddr_in *sin)
- {
- struct iax2_registry *reg;
- /* Start pessimistic */
- struct iax_ie_data ied;
- char peer[256] = "";
- char challenge[256] = "";
- int res;
- int authmethods = 0;
- if (ies->authmethods)
- authmethods = ies->authmethods;
- if (ies->username)
- ast_copy_string(peer, ies->username, sizeof(peer));
- if (ies->challenge)
- ast_copy_string(challenge, ies->challenge, sizeof(challenge));
- memset(&ied, 0, sizeof(ied));
- reg = iaxs[callno]->reg;
- if (reg) {
- struct sockaddr_in reg_addr;
- ast_sockaddr_to_sin(®->addr, ®_addr);
- if (inaddrcmp(®_addr, sin)) {
- ast_log(LOG_WARNING, "Received unsolicited registry authenticate request from '%s'\n", ast_inet_ntoa(sin->sin_addr));
- return -1;
- }
- if (ast_strlen_zero(reg->secret)) {
- ast_log(LOG_NOTICE, "No secret associated with peer '%s'\n", reg->username);
- reg->regstate = REG_STATE_NOAUTH;
- return -1;
- }
- iax_ie_append_str(&ied, IAX_IE_USERNAME, reg->username);
- iax_ie_append_short(&ied, IAX_IE_REFRESH, reg->refresh);
- if (reg->secret[0] == '[') {
- char tmpkey[256];
- ast_copy_string(tmpkey, reg->secret + 1, sizeof(tmpkey));
- tmpkey[strlen(tmpkey) - 1] = '\0';
- res = authenticate(challenge, NULL, tmpkey, authmethods, &ied, sin, NULL);
- } else
- res = authenticate(challenge, reg->secret, NULL, authmethods, &ied, sin, NULL);
- if (!res) {
- reg->regstate = REG_STATE_AUTHSENT;
- add_empty_calltoken_ie(iaxs[callno], &ied); /* this _MUST_ be the last ie added */
- return send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_REGREQ, 0, ied.buf, ied.pos, -1);
- } else
- return -1;
- ast_log(LOG_WARNING, "Registry acknowledge on unknown registery '%s'\n", peer);
- } else
- ast_log(LOG_NOTICE, "Can't reregister without a reg\n");
- return -1;
- }
- static void stop_stuff(int callno)
- {
- iax2_destroy_helper(iaxs[callno]);
- }
- static void __auth_reject(const void *nothing)
- {
- /* Called from IAX thread only, without iaxs lock */
- int callno = (int)(long)(nothing);
- struct iax_ie_data ied;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- memset(&ied, 0, sizeof(ied));
- if (iaxs[callno]->authfail == IAX_COMMAND_REGREJ) {
- iax_ie_append_str(&ied, IAX_IE_CAUSE, "Registration Refused");
- iax_ie_append_byte(&ied, IAX_IE_CAUSECODE, AST_CAUSE_FACILITY_REJECTED);
- } else if (iaxs[callno]->authfail == IAX_COMMAND_REJECT) {
- iax_ie_append_str(&ied, IAX_IE_CAUSE, "No authority found");
- iax_ie_append_byte(&ied, IAX_IE_CAUSECODE, AST_CAUSE_FACILITY_NOT_SUBSCRIBED);
- }
- send_command_final(iaxs[callno], AST_FRAME_IAX, iaxs[callno]->authfail, 0, ied.buf, ied.pos, -1);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int auth_reject(const void *data)
- {
- int callno = (int)(long)(data);
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno])
- iaxs[callno]->authid = -1;
- ast_mutex_unlock(&iaxsl[callno]);
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__auth_reject, data))
- #endif
- __auth_reject(data);
- return 0;
- }
- static int auth_fail(int callno, int failcode)
- {
- /* Schedule sending the authentication failure in one second, to prevent
- guessing */
- if (iaxs[callno]) {
- iaxs[callno]->authfail = failcode;
- if (delayreject) {
- iaxs[callno]->authid = iax2_sched_replace(iaxs[callno]->authid,
- sched, 1000, auth_reject, (void *)(long)callno);
- } else
- auth_reject((void *)(long)callno);
- }
- return 0;
- }
- static void __auto_hangup(const void *nothing)
- {
- /* Called from IAX thread only, without iaxs lock */
- int callno = (int)(long)(nothing);
- struct iax_ie_data ied;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- memset(&ied, 0, sizeof(ied));
- iax_ie_append_str(&ied, IAX_IE_CAUSE, "Timeout");
- iax_ie_append_byte(&ied, IAX_IE_CAUSECODE, AST_CAUSE_NO_USER_RESPONSE);
- send_command_final(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_HANGUP, 0, ied.buf, ied.pos, -1);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int auto_hangup(const void *data)
- {
- int callno = (int)(long)(data);
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- iaxs[callno]->autoid = -1;
- }
- ast_mutex_unlock(&iaxsl[callno]);
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__auto_hangup, data))
- #endif
- __auto_hangup(data);
- return 0;
- }
- static void iax2_dprequest(struct iax2_dpcache *dp, int callno)
- {
- struct iax_ie_data ied;
- /* Auto-hangup with 30 seconds of inactivity */
- iaxs[callno]->autoid = iax2_sched_replace(iaxs[callno]->autoid,
- sched, 30000, auto_hangup, (void *)(long)callno);
- memset(&ied, 0, sizeof(ied));
- iax_ie_append_str(&ied, IAX_IE_CALLED_NUMBER, dp->exten);
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_DPREQ, 0, ied.buf, ied.pos, -1);
- dp->flags |= CACHE_FLAG_TRANSMITTED;
- }
- static int iax2_vnak(int callno)
- {
- return send_command_immediate(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_VNAK, 0, NULL, 0, iaxs[callno]->iseqno);
- }
- static void vnak_retransmit(int callno, int last)
- {
- struct iax_frame *f;
- AST_LIST_TRAVERSE(&frame_queue[callno], f, list) {
- /* Send a copy immediately */
- if (((unsigned char) (f->oseqno - last) < 128) &&
- (f->retries >= 0)) {
- send_packet(f);
- }
- }
- }
- static void __iax2_poke_peer_s(const void *data)
- {
- struct iax2_peer *peer = (struct iax2_peer *)data;
- iax2_poke_peer(peer, 0);
- peer_unref(peer);
- }
- static int iax2_poke_peer_s(const void *data)
- {
- struct iax2_peer *peer = (struct iax2_peer *)data;
- peer->pokeexpire = -1;
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__iax2_poke_peer_s, data))
- #endif
- __iax2_poke_peer_s(data);
- return 0;
- }
- static int send_trunk(struct iax2_trunk_peer *tpeer, struct timeval *now)
- {
- int res = 0;
- struct iax_frame *fr;
- struct ast_iax2_meta_hdr *meta;
- struct ast_iax2_meta_trunk_hdr *mth;
- int calls = 0;
-
- /* Point to frame */
- fr = (struct iax_frame *)tpeer->trunkdata;
- /* Point to meta data */
- meta = (struct ast_iax2_meta_hdr *)fr->afdata;
- mth = (struct ast_iax2_meta_trunk_hdr *)meta->data;
- if (tpeer->trunkdatalen) {
- /* We're actually sending a frame, so fill the meta trunk header and meta header */
- meta->zeros = 0;
- meta->metacmd = IAX_META_TRUNK;
- if (ast_test_flag64(&globalflags, IAX_TRUNKTIMESTAMPS))
- meta->cmddata = IAX_META_TRUNK_MINI;
- else
- meta->cmddata = IAX_META_TRUNK_SUPERMINI;
- mth->ts = htonl(calc_txpeerstamp(tpeer, trunkfreq, now));
- /* And the rest of the ast_iax2 header */
- fr->direction = DIRECTION_OUTGRESS;
- fr->retrans = -1;
- fr->transfer = 0;
- /* Any appropriate call will do */
- fr->data = fr->afdata;
- fr->datalen = tpeer->trunkdatalen + sizeof(struct ast_iax2_meta_hdr) + sizeof(struct ast_iax2_meta_trunk_hdr);
- res = transmit_trunk(fr, &tpeer->addr, tpeer->sockfd);
- calls = tpeer->calls;
- #if 0
- ast_debug(1, "Trunking %d call chunks in %d bytes to %s:%d, ts=%d\n", calls, fr->datalen, ast_inet_ntoa(tpeer->addr.sin_addr), ntohs(tpeer->addr.sin_port), ntohl(mth->ts));
- #endif
- /* Reset transmit trunk side data */
- tpeer->trunkdatalen = 0;
- tpeer->calls = 0;
- }
- if (res < 0)
- return res;
- return calls;
- }
- static inline int iax2_trunk_expired(struct iax2_trunk_peer *tpeer, struct timeval *now)
- {
- /* Drop when trunk is about 5 seconds idle */
- if (now->tv_sec > tpeer->trunkact.tv_sec + 5)
- return 1;
- return 0;
- }
- static int timing_read(int *id, int fd, short events, void *cbdata)
- {
- int res, processed = 0, totalcalls = 0;
- struct iax2_trunk_peer *tpeer = NULL, *drop = NULL;
- struct timeval now = ast_tvnow();
- if (iaxtrunkdebug)
- ast_verbose("Beginning trunk processing. Trunk queue ceiling is %d bytes per host\n", trunkmaxsize);
- if (timer) {
- if (ast_timer_ack(timer, 1) < 0) {
- ast_log(LOG_ERROR, "Timer failed acknowledge\n");
- return 0;
- }
- }
- /* For each peer that supports trunking... */
- AST_LIST_LOCK(&tpeers);
- AST_LIST_TRAVERSE_SAFE_BEGIN(&tpeers, tpeer, list) {
- processed++;
- res = 0;
- ast_mutex_lock(&tpeer->lock);
- /* We can drop a single tpeer per pass. That makes all this logic
- substantially easier */
- if (!drop && iax2_trunk_expired(tpeer, &now)) {
- /* Take it out of the list, but don't free it yet, because it
- could be in use */
- AST_LIST_REMOVE_CURRENT(list);
- drop = tpeer;
- } else {
- res = send_trunk(tpeer, &now);
- trunk_timed++;
- if (iaxtrunkdebug)
- ast_verbose(" - Trunk peer (%s:%d) has %d call chunk%s in transit, %u bytes backloged and has hit a high water mark of %u bytes\n", ast_inet_ntoa(tpeer->addr.sin_addr), ntohs(tpeer->addr.sin_port), res, (res != 1) ? "s" : "", tpeer->trunkdatalen, tpeer->trunkdataalloc);
- }
- totalcalls += res;
- res = 0;
- ast_mutex_unlock(&tpeer->lock);
- }
- AST_LIST_TRAVERSE_SAFE_END;
- AST_LIST_UNLOCK(&tpeers);
- if (drop) {
- ast_mutex_lock(&drop->lock);
- /* Once we have this lock, we're sure nobody else is using it or could use it once we release it,
- because by the time they could get tpeerlock, we've already grabbed it */
- ast_debug(1, "Dropping unused iax2 trunk peer '%s:%d'\n", ast_inet_ntoa(drop->addr.sin_addr), ntohs(drop->addr.sin_port));
- if (drop->trunkdata) {
- ast_free(drop->trunkdata);
- drop->trunkdata = NULL;
- }
- ast_mutex_unlock(&drop->lock);
- ast_mutex_destroy(&drop->lock);
- ast_free(drop);
- }
- if (iaxtrunkdebug)
- ast_verbose("Ending trunk processing with %d peers and %d call chunks processed\n", processed, totalcalls);
- iaxtrunkdebug = 0;
- return 1;
- }
- struct dpreq_data {
- int callno;
- char context[AST_MAX_EXTENSION];
- char callednum[AST_MAX_EXTENSION];
- char *callerid;
- };
- static void dp_lookup(int callno, const char *context, const char *callednum, const char *callerid, int skiplock)
- {
- unsigned short dpstatus = 0;
- struct iax_ie_data ied1;
- int mm;
- memset(&ied1, 0, sizeof(ied1));
- mm = ast_matchmore_extension(NULL, context, callednum, 1, callerid);
- /* Must be started */
- if (ast_parking_ext_valid(callednum, NULL, context) || ast_exists_extension(NULL, context, callednum, 1, callerid)) {
- dpstatus = IAX_DPSTATUS_EXISTS;
- } else if (ast_canmatch_extension(NULL, context, callednum, 1, callerid)) {
- dpstatus = IAX_DPSTATUS_CANEXIST;
- } else {
- dpstatus = IAX_DPSTATUS_NONEXISTENT;
- }
- if (ast_ignore_pattern(context, callednum))
- dpstatus |= IAX_DPSTATUS_IGNOREPAT;
- if (mm)
- dpstatus |= IAX_DPSTATUS_MATCHMORE;
- if (!skiplock)
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- iax_ie_append_str(&ied1, IAX_IE_CALLED_NUMBER, callednum);
- iax_ie_append_short(&ied1, IAX_IE_DPSTATUS, dpstatus);
- iax_ie_append_short(&ied1, IAX_IE_REFRESH, iaxdefaultdpcache);
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_DPREP, 0, ied1.buf, ied1.pos, -1);
- }
- if (!skiplock)
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static void *dp_lookup_thread(void *data)
- {
- /* Look up for dpreq */
- struct dpreq_data *dpr = data;
- dp_lookup(dpr->callno, dpr->context, dpr->callednum, dpr->callerid, 0);
- if (dpr->callerid)
- ast_free(dpr->callerid);
- ast_free(dpr);
- return NULL;
- }
- static void spawn_dp_lookup(int callno, const char *context, const char *callednum, const char *callerid)
- {
- pthread_t newthread;
- struct dpreq_data *dpr;
-
- if (!(dpr = ast_calloc(1, sizeof(*dpr))))
- return;
- dpr->callno = callno;
- ast_copy_string(dpr->context, context, sizeof(dpr->context));
- ast_copy_string(dpr->callednum, callednum, sizeof(dpr->callednum));
- if (callerid)
- dpr->callerid = ast_strdup(callerid);
- if (ast_pthread_create_detached(&newthread, NULL, dp_lookup_thread, dpr)) {
- ast_log(LOG_WARNING, "Unable to start lookup thread!\n");
- }
- }
- struct iax_dual {
- struct ast_channel *chan1;
- struct ast_channel *chan2;
- char *park_exten;
- char *park_context;
- };
- static void *iax_park_thread(void *stuff)
- {
- struct iax_dual *d;
- int res;
- int ext = 0;
- d = stuff;
- ast_debug(4, "IAX Park: Transferer channel %s, Transferee %s\n",
- ast_channel_name(d->chan2), ast_channel_name(d->chan1));
- res = ast_park_call_exten(d->chan1, d->chan2, d->park_exten, d->park_context, 0, &ext);
- if (res) {
- /* Parking failed. */
- ast_hangup(d->chan1);
- } else {
- ast_log(LOG_NOTICE, "Parked on extension '%d'\n", ext);
- }
- ast_hangup(d->chan2);
- ast_free(d->park_exten);
- ast_free(d->park_context);
- ast_free(d);
- return NULL;
- }
- /*! DO NOT hold any locks while calling iax_park */
- static int iax_park(struct ast_channel *chan1, struct ast_channel *chan2, const char *park_exten, const char *park_context)
- {
- struct iax_dual *d;
- struct ast_channel *chan1m, *chan2m;/* Chan2m: The transferer, chan1m: The transferee */
- pthread_t th;
- chan1m = ast_channel_alloc(0, AST_STATE_DOWN, 0, 0, ast_channel_accountcode(chan2), ast_channel_exten(chan1), ast_channel_context(chan1), ast_channel_linkedid(chan1), ast_channel_amaflags(chan1), "Parking/%s", ast_channel_name(chan1));
- chan2m = ast_channel_alloc(0, AST_STATE_DOWN, 0, 0, ast_channel_accountcode(chan2), ast_channel_exten(chan2), ast_channel_context(chan2), ast_channel_linkedid(chan2), ast_channel_amaflags(chan2), "IAXPeer/%s", ast_channel_name(chan2));
- d = ast_calloc(1, sizeof(*d));
- if (!chan1m || !chan2m || !d) {
- if (chan1m) {
- ast_hangup(chan1m);
- }
- if (chan2m) {
- ast_hangup(chan2m);
- }
- ast_free(d);
- return -1;
- }
- d->park_exten = ast_strdup(park_exten);
- d->park_context = ast_strdup(park_context);
- if (!d->park_exten || !d->park_context) {
- ast_hangup(chan1m);
- ast_hangup(chan2m);
- ast_free(d->park_exten);
- ast_free(d->park_context);
- ast_free(d);
- return -1;
- }
- /* Make formats okay */
- ast_format_copy(ast_channel_readformat(chan1m), ast_channel_readformat(chan1));
- ast_format_copy(ast_channel_writeformat(chan1m), ast_channel_writeformat(chan1));
- /* Prepare for taking over the channel */
- if (ast_channel_masquerade(chan1m, chan1)) {
- ast_hangup(chan1m);
- ast_hangup(chan2m);
- ast_free(d->park_exten);
- ast_free(d->park_context);
- ast_free(d);
- return -1;
- }
- /* Setup the extensions and such */
- ast_channel_context_set(chan1m, ast_channel_context(chan1));
- ast_channel_exten_set(chan1m, ast_channel_exten(chan1));
- ast_channel_priority_set(chan1m, ast_channel_priority(chan1));
- ast_do_masquerade(chan1m);
- /* We make a clone of the peer channel too, so we can play
- back the announcement */
- /* Make formats okay */
- ast_format_copy(ast_channel_readformat(chan2m), ast_channel_readformat(chan2));
- ast_format_copy(ast_channel_writeformat(chan2m), ast_channel_writeformat(chan2));
- ast_channel_parkinglot_set(chan2m, ast_channel_parkinglot(chan2));
- /* Prepare for taking over the channel */
- if (ast_channel_masquerade(chan2m, chan2)) {
- ast_hangup(chan1m);
- ast_hangup(chan2m);
- ast_free(d->park_exten);
- ast_free(d->park_context);
- ast_free(d);
- return -1;
- }
- /* Setup the extensions and such */
- ast_channel_context_set(chan2m, ast_channel_context(chan2));
- ast_channel_exten_set(chan2m, ast_channel_exten(chan2));
- ast_channel_priority_set(chan2m, ast_channel_priority(chan2));
- ast_do_masquerade(chan2m);
- d->chan1 = chan1m; /* Transferee */
- d->chan2 = chan2m; /* Transferer */
- if (ast_pthread_create_detached_background(&th, NULL, iax_park_thread, d) < 0) {
- /* Could not start thread */
- ast_hangup(chan1m);
- ast_hangup(chan2m);
- ast_free(d->park_exten);
- ast_free(d->park_context);
- ast_free(d);
- return -1;
- }
- return 0;
- }
- static int check_provisioning(struct sockaddr_in *sin, int sockfd, char *si, unsigned int ver)
- {
- unsigned int ourver;
- char rsi[80];
- snprintf(rsi, sizeof(rsi), "si-%s", si);
- if (iax_provision_version(&ourver, rsi, 1))
- return 0;
- ast_debug(1, "Service identifier '%s', we think '%08x', they think '%08x'\n", si, ourver, ver);
- if (ourver != ver)
- iax2_provision(sin, sockfd, NULL, rsi, 1);
- return 0;
- }
- static void construct_rr(struct chan_iax2_pvt *pvt, struct iax_ie_data *iep)
- {
- jb_info stats;
- jb_getinfo(pvt->jb, &stats);
- memset(iep, 0, sizeof(*iep));
- iax_ie_append_int(iep,IAX_IE_RR_JITTER, stats.jitter);
- if(stats.frames_in == 0) stats.frames_in = 1;
- iax_ie_append_int(iep,IAX_IE_RR_LOSS, ((0xff & (stats.losspct/1000)) << 24 | (stats.frames_lost & 0x00ffffff)));
- iax_ie_append_int(iep,IAX_IE_RR_PKTS, stats.frames_in);
- iax_ie_append_short(iep,IAX_IE_RR_DELAY, stats.current - stats.min);
- iax_ie_append_int(iep,IAX_IE_RR_DROPPED, stats.frames_dropped);
- iax_ie_append_int(iep,IAX_IE_RR_OOO, stats.frames_ooo);
- }
- static void save_rr(struct iax_frame *fr, struct iax_ies *ies)
- {
- iaxs[fr->callno]->remote_rr.jitter = ies->rr_jitter;
- iaxs[fr->callno]->remote_rr.losspct = ies->rr_loss >> 24;
- iaxs[fr->callno]->remote_rr.losscnt = ies->rr_loss & 0xffffff;
- iaxs[fr->callno]->remote_rr.packets = ies->rr_pkts;
- iaxs[fr->callno]->remote_rr.delay = ies->rr_delay;
- iaxs[fr->callno]->remote_rr.dropped = ies->rr_dropped;
- iaxs[fr->callno]->remote_rr.ooo = ies->rr_ooo;
- }
- static void save_osptoken(struct iax_frame *fr, struct iax_ies *ies)
- {
- int i;
- unsigned int length, offset = 0;
- char full_osptoken[IAX_MAX_OSPBUFF_SIZE];
- for (i = 0; i < IAX_MAX_OSPBLOCK_NUM; i++) {
- length = ies->ospblocklength[i];
- if (length != 0) {
- if (length > IAX_MAX_OSPBLOCK_SIZE) {
- /* OSP token block length wrong, clear buffer */
- offset = 0;
- break;
- } else {
- memcpy(full_osptoken + offset, ies->osptokenblock[i], length);
- offset += length;
- }
- } else {
- break;
- }
- }
- *(full_osptoken + offset) = '\0';
- if (strlen(full_osptoken) != offset) {
- /* OSP token length wrong, clear buffer */
- *full_osptoken = '\0';
- }
- ast_string_field_set(iaxs[fr->callno], osptoken, full_osptoken);
- }
- static void log_jitterstats(unsigned short callno)
- {
- int localjitter = -1, localdelay = 0, locallost = -1, locallosspct = -1, localdropped = 0, localooo = -1, localpackets = -1;
- jb_info jbinfo;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno] && iaxs[callno]->owner && ast_channel_name(iaxs[callno]->owner)) {
- if(ast_test_flag64(iaxs[callno], IAX_USEJITTERBUF)) {
- jb_getinfo(iaxs[callno]->jb, &jbinfo);
- localjitter = jbinfo.jitter;
- localdelay = jbinfo.current - jbinfo.min;
- locallost = jbinfo.frames_lost;
- locallosspct = jbinfo.losspct/1000;
- localdropped = jbinfo.frames_dropped;
- localooo = jbinfo.frames_ooo;
- localpackets = jbinfo.frames_in;
- }
- ast_debug(3, "JB STATS:%s ping=%u ljitterms=%d ljbdelayms=%d ltotlost=%d lrecentlosspct=%d ldropped=%d looo=%d lrecvd=%d rjitterms=%d rjbdelayms=%d rtotlost=%d rrecentlosspct=%d rdropped=%d rooo=%d rrecvd=%d\n",
- ast_channel_name(iaxs[callno]->owner),
- iaxs[callno]->pingtime,
- localjitter,
- localdelay,
- locallost,
- locallosspct,
- localdropped,
- localooo,
- localpackets,
- iaxs[callno]->remote_rr.jitter,
- iaxs[callno]->remote_rr.delay,
- iaxs[callno]->remote_rr.losscnt,
- iaxs[callno]->remote_rr.losspct/1000,
- iaxs[callno]->remote_rr.dropped,
- iaxs[callno]->remote_rr.ooo,
- iaxs[callno]->remote_rr.packets);
- manager_event(EVENT_FLAG_REPORTING, "JitterBufStats", "Owner: %s\r\nPing: %u\r\nLocalJitter: %d\r\nLocalJBDelay: %d\r\nLocalTotalLost: %d\r\nLocalLossPercent: %d\r\nLocalDropped: %d\r\nLocalooo: %d\r\nLocalReceived: %d\r\nRemoteJitter: %d\r\nRemoteJBDelay: %d\r\nRemoteTotalLost: %d\r\nRemoteLossPercent: %d\r\nRemoteDropped: %d\r\nRemoteooo: %d\r\nRemoteReceived: %d\r\n",
- ast_channel_name(iaxs[callno]->owner),
- iaxs[callno]->pingtime,
- localjitter,
- localdelay,
- locallost,
- locallosspct,
- localdropped,
- localooo,
- localpackets,
- iaxs[callno]->remote_rr.jitter,
- iaxs[callno]->remote_rr.delay,
- iaxs[callno]->remote_rr.losscnt,
- iaxs[callno]->remote_rr.losspct/1000,
- iaxs[callno]->remote_rr.dropped,
- iaxs[callno]->remote_rr.ooo,
- iaxs[callno]->remote_rr.packets);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- static int socket_process(struct iax2_thread *thread);
- /*!
- * \brief Handle any deferred full frames for this thread
- */
- static void handle_deferred_full_frames(struct iax2_thread *thread)
- {
- struct iax2_pkt_buf *pkt_buf;
- ast_mutex_lock(&thread->lock);
- while ((pkt_buf = AST_LIST_REMOVE_HEAD(&thread->full_frames, entry))) {
- ast_mutex_unlock(&thread->lock);
- thread->buf = pkt_buf->buf;
- thread->buf_len = pkt_buf->len;
- thread->buf_size = pkt_buf->len + 1;
-
- socket_process(thread);
- thread->buf = NULL;
- ast_free(pkt_buf);
- ast_mutex_lock(&thread->lock);
- }
- ast_mutex_unlock(&thread->lock);
- }
- /*!
- * \brief Queue the last read full frame for processing by a certain thread
- *
- * If there are already any full frames queued, they are sorted
- * by sequence number.
- */
- static void defer_full_frame(struct iax2_thread *from_here, struct iax2_thread *to_here)
- {
- struct iax2_pkt_buf *pkt_buf, *cur_pkt_buf;
- struct ast_iax2_full_hdr *fh, *cur_fh;
- if (!(pkt_buf = ast_calloc(1, sizeof(*pkt_buf) + from_here->buf_len)))
- return;
- pkt_buf->len = from_here->buf_len;
- memcpy(pkt_buf->buf, from_here->buf, pkt_buf->len);
- fh = (struct ast_iax2_full_hdr *) pkt_buf->buf;
- ast_mutex_lock(&to_here->lock);
- AST_LIST_TRAVERSE_SAFE_BEGIN(&to_here->full_frames, cur_pkt_buf, entry) {
- cur_fh = (struct ast_iax2_full_hdr *) cur_pkt_buf->buf;
- if (fh->oseqno < cur_fh->oseqno) {
- AST_LIST_INSERT_BEFORE_CURRENT(pkt_buf, entry);
- break;
- }
- }
- AST_LIST_TRAVERSE_SAFE_END
- if (!cur_pkt_buf)
- AST_LIST_INSERT_TAIL(&to_here->full_frames, pkt_buf, entry);
- to_here->iostate = IAX_IOSTATE_READY;
- ast_cond_signal(&to_here->cond);
- ast_mutex_unlock(&to_here->lock);
- }
- static int socket_read(int *id, int fd, short events, void *cbdata)
- {
- struct iax2_thread *thread;
- socklen_t len;
- time_t t;
- static time_t last_errtime = 0;
- struct ast_iax2_full_hdr *fh;
- if (!(thread = find_idle_thread())) {
- time(&t);
- if (t != last_errtime) {
- last_errtime = t;
- ast_debug(1, "Out of idle IAX2 threads for I/O, pausing!\n");
- }
- usleep(1);
- return 1;
- }
- len = sizeof(thread->iosin);
- thread->iofd = fd;
- thread->buf_len = recvfrom(fd, thread->readbuf, sizeof(thread->readbuf), 0, (struct sockaddr *) &thread->iosin, &len);
- thread->buf_size = sizeof(thread->readbuf);
- thread->buf = thread->readbuf;
- if (thread->buf_len < 0) {
- if (errno != ECONNREFUSED && errno != EAGAIN)
- ast_log(LOG_WARNING, "Error: %s\n", strerror(errno));
- handle_error();
- thread->iostate = IAX_IOSTATE_IDLE;
- signal_condition(&thread->lock, &thread->cond);
- return 1;
- }
- if (test_losspct && ((100.0 * ast_random() / (RAND_MAX + 1.0)) < test_losspct)) { /* simulate random loss condition */
- thread->iostate = IAX_IOSTATE_IDLE;
- signal_condition(&thread->lock, &thread->cond);
- return 1;
- }
-
- /* Determine if this frame is a full frame; if so, and any thread is currently
- processing a full frame for the same callno from this peer, then drop this
- frame (and the peer will retransmit it) */
- fh = (struct ast_iax2_full_hdr *) thread->buf;
- if (ntohs(fh->scallno) & IAX_FLAG_FULL) {
- struct iax2_thread *cur = NULL;
- uint16_t callno = ntohs(fh->scallno) & ~IAX_FLAG_FULL;
-
- AST_LIST_LOCK(&active_list);
- AST_LIST_TRAVERSE(&active_list, cur, list) {
- if ((cur->ffinfo.callno == callno) &&
- !inaddrcmp(&cur->ffinfo.sin, &thread->iosin))
- break;
- }
- if (cur) {
- /* we found another thread processing a full frame for this call,
- so queue it up for processing later. */
- defer_full_frame(thread, cur);
- AST_LIST_UNLOCK(&active_list);
- thread->iostate = IAX_IOSTATE_IDLE;
- signal_condition(&thread->lock, &thread->cond);
- return 1;
- } else {
- /* this thread is going to process this frame, so mark it */
- thread->ffinfo.callno = callno;
- memcpy(&thread->ffinfo.sin, &thread->iosin, sizeof(thread->ffinfo.sin));
- thread->ffinfo.type = fh->type;
- thread->ffinfo.csub = fh->csub;
- AST_LIST_INSERT_HEAD(&active_list, thread, list);
- }
- AST_LIST_UNLOCK(&active_list);
- }
-
- /* Mark as ready and send on its way */
- thread->iostate = IAX_IOSTATE_READY;
- #ifdef DEBUG_SCHED_MULTITHREAD
- ast_copy_string(thread->curfunc, "socket_process", sizeof(thread->curfunc));
- #endif
- signal_condition(&thread->lock, &thread->cond);
- return 1;
- }
- static int socket_process_meta(int packet_len, struct ast_iax2_meta_hdr *meta, struct sockaddr_in *sin, int sockfd,
- struct iax_frame *fr)
- {
- unsigned char metatype;
- struct ast_iax2_meta_trunk_mini *mtm;
- struct ast_iax2_meta_trunk_hdr *mth;
- struct ast_iax2_meta_trunk_entry *mte;
- struct iax2_trunk_peer *tpeer;
- unsigned int ts;
- void *ptr;
- struct timeval rxtrunktime;
- struct ast_frame f = { 0, };
- if (packet_len < sizeof(*meta)) {
- ast_log(LOG_WARNING, "Rejecting packet from '%s.%d' that is flagged as a meta frame but is too short\n",
- ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
- return 1;
- }
- if (meta->metacmd != IAX_META_TRUNK)
- return 1;
- if (packet_len < (sizeof(*meta) + sizeof(*mth))) {
- ast_log(LOG_WARNING, "midget meta trunk packet received (%d of %d min)\n", packet_len,
- (int) (sizeof(*meta) + sizeof(*mth)));
- return 1;
- }
- mth = (struct ast_iax2_meta_trunk_hdr *)(meta->data);
- ts = ntohl(mth->ts);
- metatype = meta->cmddata;
- packet_len -= (sizeof(*meta) + sizeof(*mth));
- ptr = mth->data;
- tpeer = find_tpeer(sin, sockfd);
- if (!tpeer) {
- ast_log(LOG_WARNING, "Unable to accept trunked packet from '%s:%d': No matching peer\n",
- ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
- return 1;
- }
- tpeer->trunkact = ast_tvnow();
- if (!ts || ast_tvzero(tpeer->rxtrunktime))
- tpeer->rxtrunktime = tpeer->trunkact;
- rxtrunktime = tpeer->rxtrunktime;
- ast_mutex_unlock(&tpeer->lock);
- while (packet_len >= sizeof(*mte)) {
- /* Process channels */
- unsigned short callno, trunked_ts, len;
- if (metatype == IAX_META_TRUNK_MINI) {
- mtm = (struct ast_iax2_meta_trunk_mini *) ptr;
- ptr += sizeof(*mtm);
- packet_len -= sizeof(*mtm);
- len = ntohs(mtm->len);
- callno = ntohs(mtm->mini.callno);
- trunked_ts = ntohs(mtm->mini.ts);
- } else if (metatype == IAX_META_TRUNK_SUPERMINI) {
- mte = (struct ast_iax2_meta_trunk_entry *)ptr;
- ptr += sizeof(*mte);
- packet_len -= sizeof(*mte);
- len = ntohs(mte->len);
- callno = ntohs(mte->callno);
- trunked_ts = 0;
- } else {
- ast_log(LOG_WARNING, "Unknown meta trunk cmd from '%s:%d': dropping\n", ast_inet_ntoa(sin->sin_addr), ntohs(sin->sin_port));
- break;
- }
- /* Stop if we don't have enough data */
- if (len > packet_len)
- break;
- fr->callno = find_callno_locked(callno & ~IAX_FLAG_FULL, 0, sin, NEW_PREVENT, sockfd, 0);
- if (!fr->callno)
- continue;
- /* If it's a valid call, deliver the contents. If not, we
- drop it, since we don't have a scallno to use for an INVAL */
- /* Process as a mini frame */
- memset(&f, 0, sizeof(f));
- f.frametype = AST_FRAME_VOICE;
- if (!iaxs[fr->callno]) {
- /* drop it */
- } else if (iaxs[fr->callno]->voiceformat == 0) {
- ast_log(LOG_WARNING, "Received trunked frame before first full voice frame\n");
- iax2_vnak(fr->callno);
- } else {
- ast_format_from_old_bitfield(&f.subclass.format, iaxs[fr->callno]->voiceformat);
- f.datalen = len;
- if (f.datalen >= 0) {
- if (f.datalen)
- f.data.ptr = ptr;
- else
- f.data.ptr = NULL;
- if (trunked_ts)
- fr->ts = (iaxs[fr->callno]->last & 0xFFFF0000L) | (trunked_ts & 0xffff);
- else
- fr->ts = fix_peerts(&rxtrunktime, fr->callno, ts);
- /* Don't pass any packets until we're started */
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED)) {
- struct iax_frame *duped_fr;
- /* Common things */
- f.src = "IAX2";
- f.mallocd = 0;
- f.offset = 0;
- if (f.datalen && (f.frametype == AST_FRAME_VOICE))
- f.samples = ast_codec_get_samples(&f);
- else
- f.samples = 0;
- fr->outoforder = 0;
- iax_frame_wrap(fr, &f);
- duped_fr = iaxfrdup2(fr);
- if (duped_fr)
- schedule_delivery(duped_fr, 1, 1, &fr->ts);
- if (iaxs[fr->callno] && iaxs[fr->callno]->last < fr->ts)
- iaxs[fr->callno]->last = fr->ts;
- }
- } else {
- ast_log(LOG_WARNING, "Datalen < 0?\n");
- }
- }
- ast_mutex_unlock(&iaxsl[fr->callno]);
- ptr += len;
- packet_len -= len;
- }
- return 1;
- }
- static int acf_iaxvar_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
- {
- struct ast_datastore *variablestore;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- struct ast_var_t *var;
- if (!chan) {
- ast_log(LOG_WARNING, "No channel was provided to %s function.\n", cmd);
- return -1;
- }
- variablestore = ast_channel_datastore_find(chan, &iax2_variable_datastore_info, NULL);
- if (!variablestore) {
- *buf = '\0';
- return 0;
- }
- varlist = variablestore->data;
- AST_LIST_LOCK(varlist);
- AST_LIST_TRAVERSE(varlist, var, entries) {
- if (strcmp(var->name, data) == 0) {
- ast_copy_string(buf, var->value, len);
- break;
- }
- }
- AST_LIST_UNLOCK(varlist);
- return 0;
- }
- static int acf_iaxvar_write(struct ast_channel *chan, const char *cmd, char *data, const char *value)
- {
- struct ast_datastore *variablestore;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- struct ast_var_t *var;
- if (!chan) {
- ast_log(LOG_WARNING, "No channel was provided to %s function.\n", cmd);
- return -1;
- }
- variablestore = ast_channel_datastore_find(chan, &iax2_variable_datastore_info, NULL);
- if (!variablestore) {
- variablestore = ast_datastore_alloc(&iax2_variable_datastore_info, NULL);
- if (!variablestore) {
- ast_log(LOG_ERROR, "Memory allocation error\n");
- return -1;
- }
- varlist = ast_calloc(1, sizeof(*varlist));
- if (!varlist) {
- ast_datastore_free(variablestore);
- ast_log(LOG_ERROR, "Unable to assign new variable '%s'\n", data);
- return -1;
- }
- AST_LIST_HEAD_INIT(varlist);
- variablestore->data = varlist;
- variablestore->inheritance = DATASTORE_INHERIT_FOREVER;
- ast_channel_datastore_add(chan, variablestore);
- } else
- varlist = variablestore->data;
- AST_LIST_LOCK(varlist);
- AST_LIST_TRAVERSE_SAFE_BEGIN(varlist, var, entries) {
- if (strcmp(var->name, data) == 0) {
- AST_LIST_REMOVE_CURRENT(entries);
- ast_var_delete(var);
- break;
- }
- }
- AST_LIST_TRAVERSE_SAFE_END;
- var = ast_var_assign(data, value);
- if (var)
- AST_LIST_INSERT_TAIL(varlist, var, entries);
- else
- ast_log(LOG_ERROR, "Unable to assign new variable '%s'\n", data);
- AST_LIST_UNLOCK(varlist);
- return 0;
- }
- static struct ast_custom_function iaxvar_function = {
- .name = "IAXVAR",
- .read = acf_iaxvar_read,
- .write = acf_iaxvar_write,
- };
- static void set_hangup_source_and_cause(int callno, unsigned char causecode)
- {
- iax2_lock_owner(callno);
- if (iaxs[callno] && iaxs[callno]->owner) {
- struct ast_channel *owner;
- const char *name;
- owner = iaxs[callno]->owner;
- if (causecode) {
- ast_channel_hangupcause_set(owner, causecode);
- }
- name = ast_strdupa(ast_channel_name(owner));
- ast_channel_ref(owner);
- ast_channel_unlock(owner);
- ast_mutex_unlock(&iaxsl[callno]);
- ast_set_hangupsource(owner, name, 0);
- ast_channel_unref(owner);
- ast_mutex_lock(&iaxsl[callno]);
- }
- }
- static int socket_process_helper(struct iax2_thread *thread)
- {
- struct sockaddr_in sin;
- int res;
- int updatehistory=1;
- int new = NEW_PREVENT;
- int dcallno = 0;
- char decrypted = 0;
- struct ast_iax2_full_hdr *fh = (struct ast_iax2_full_hdr *)thread->buf;
- struct ast_iax2_mini_hdr *mh = (struct ast_iax2_mini_hdr *)thread->buf;
- struct ast_iax2_meta_hdr *meta = (struct ast_iax2_meta_hdr *)thread->buf;
- struct ast_iax2_video_hdr *vh = (struct ast_iax2_video_hdr *)thread->buf;
- struct iax_frame *fr;
- struct iax_frame *cur;
- struct ast_frame f = { 0, };
- struct ast_channel *c = NULL;
- struct iax2_dpcache *dp;
- struct iax2_peer *peer;
- struct iax_ies ies;
- struct iax_ie_data ied0, ied1;
- iax2_format format;
- int fd;
- int exists;
- int minivid = 0;
- char empty[32]=""; /* Safety measure */
- struct iax_frame *duped_fr;
- char host_pref_buf[128];
- char caller_pref_buf[128];
- struct ast_codec_pref pref;
- char *using_prefs = "mine";
- /* allocate an iax_frame with 4096 bytes of data buffer */
- fr = ast_alloca(sizeof(*fr) + 4096);
- memset(fr, 0, sizeof(*fr));
- fr->afdatalen = 4096; /* From ast_alloca() above */
- /* Copy frequently used parameters to the stack */
- res = thread->buf_len;
- fd = thread->iofd;
- memcpy(&sin, &thread->iosin, sizeof(sin));
- if (res < sizeof(*mh)) {
- ast_log(LOG_WARNING, "midget packet received (%d of %d min)\n", res, (int) sizeof(*mh));
- return 1;
- }
- if ((vh->zeros == 0) && (ntohs(vh->callno) & 0x8000)) {
- if (res < sizeof(*vh)) {
- ast_log(LOG_WARNING, "Rejecting packet from '%s.%d' that is flagged as a video frame but is too short\n", ast_inet_ntoa(sin.sin_addr), ntohs(sin.sin_port));
- return 1;
- }
- /* This is a video frame, get call number */
- fr->callno = find_callno(ntohs(vh->callno) & ~0x8000, dcallno, &sin, new, fd, 0);
- minivid = 1;
- } else if ((meta->zeros == 0) && !(ntohs(meta->metacmd) & 0x8000))
- return socket_process_meta(res, meta, &sin, fd, fr);
- #ifdef DEBUG_SUPPORT
- if (res >= sizeof(*fh))
- iax_outputframe(NULL, fh, 1, &sin, res - sizeof(*fh));
- #endif
- if (ntohs(mh->callno) & IAX_FLAG_FULL) {
- if (res < sizeof(*fh)) {
- ast_log(LOG_WARNING, "Rejecting packet from '%s.%d' that is flagged as a full frame but is too short\n", ast_inet_ntoa(sin.sin_addr), ntohs(sin.sin_port));
- return 1;
- }
- /* Get the destination call number */
- dcallno = ntohs(fh->dcallno) & ~IAX_FLAG_RETRANS;
- /* check to make sure this full frame isn't encrypted before we attempt
- * to look inside of it. If it is encrypted, decrypt it first. Its ok if the
- * callno is not found here, that just means one hasn't been allocated for
- * this connection yet. */
- if ((dcallno != 1) && (fr->callno = find_callno(ntohs(mh->callno) & ~IAX_FLAG_FULL, dcallno, &sin, NEW_PREVENT, fd, 1))) {
- ast_mutex_lock(&iaxsl[fr->callno]);
- if (iaxs[fr->callno] && ast_test_flag64(iaxs[fr->callno], IAX_ENCRYPTED)) {
- if (decrypt_frame(fr->callno, fh, &f, &res)) {
- ast_log(LOG_NOTICE, "Packet Decrypt Failed!\n");
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- decrypted = 1;
- }
- ast_mutex_unlock(&iaxsl[fr->callno]);
- }
- /* Retrieve the type and subclass */
- f.frametype = fh->type;
- if (f.frametype == AST_FRAME_VIDEO) {
- ast_format_from_old_bitfield(&f.subclass.format, (uncompress_subclass(fh->csub & ~0x40)));
- if ((fh->csub >> 6) & 0x1) {
- ast_format_set_video_mark(&f.subclass.format);
- }
- } else if (f.frametype == AST_FRAME_VOICE) {
- ast_format_from_old_bitfield(&f.subclass.format, uncompress_subclass(fh->csub));
- } else {
- f.subclass.integer = uncompress_subclass(fh->csub);
- }
- /* Deal with POKE/PONG without allocating a callno */
- if (f.frametype == AST_FRAME_IAX && f.subclass.integer == IAX_COMMAND_POKE) {
- /* Reply back with a PONG, but don't care about the result. */
- send_apathetic_reply(1, ntohs(fh->scallno), &sin, IAX_COMMAND_PONG, ntohl(fh->ts), fh->iseqno + 1, fd, NULL);
- return 1;
- } else if (f.frametype == AST_FRAME_IAX && f.subclass.integer == IAX_COMMAND_ACK && dcallno == 1) {
- /* Ignore */
- return 1;
- }
- f.datalen = res - sizeof(*fh);
- if (f.datalen) {
- if (f.frametype == AST_FRAME_IAX) {
- if (iax_parse_ies(&ies, thread->buf + sizeof(struct ast_iax2_full_hdr), f.datalen)) {
- ast_log(LOG_WARNING, "Undecodable frame received from '%s'\n", ast_inet_ntoa(sin.sin_addr));
- ast_variables_destroy(ies.vars);
- return 1;
- }
- f.data.ptr = NULL;
- f.datalen = 0;
- } else {
- f.data.ptr = thread->buf + sizeof(struct ast_iax2_full_hdr);
- memset(&ies, 0, sizeof(ies));
- }
- } else {
- if (f.frametype == AST_FRAME_IAX)
- f.data.ptr = NULL;
- else
- f.data.ptr = empty;
- memset(&ies, 0, sizeof(ies));
- }
- if (!dcallno && iax2_allow_new(f.frametype, f.subclass.integer, 1)) {
- /* only set NEW_ALLOW if calltoken checks out */
- if (handle_call_token(fh, &ies, &sin, fd)) {
- ast_variables_destroy(ies.vars);
- return 1;
- }
- if (ies.calltoken && ies.calltokendata) {
- /* if we've gotten this far, and the calltoken ie data exists,
- * then calltoken validation _MUST_ have taken place. If calltoken
- * data is provided, it is always validated reguardless of any
- * calltokenoptional or requirecalltoken options */
- new = NEW_ALLOW_CALLTOKEN_VALIDATED;
- } else {
- new = NEW_ALLOW;
- }
- }
- } else {
- /* Don't know anything about it yet */
- f.frametype = AST_FRAME_NULL;
- f.subclass.integer = 0;
- memset(&ies, 0, sizeof(ies));
- }
- if (!fr->callno) {
- int check_dcallno = 0;
- /*
- * We enforce accurate destination call numbers for ACKs. This forces the other
- * end to know the destination call number before call setup can complete.
- *
- * Discussed in the following thread:
- * http://lists.digium.com/pipermail/asterisk-dev/2008-May/033217.html
- */
- if ((ntohs(mh->callno) & IAX_FLAG_FULL) && ((f.frametype == AST_FRAME_IAX) && (f.subclass.integer == IAX_COMMAND_ACK))) {
- check_dcallno = 1;
- }
- if (!(fr->callno = find_callno(ntohs(mh->callno) & ~IAX_FLAG_FULL, dcallno, &sin, new, fd, check_dcallno))) {
- if (f.frametype == AST_FRAME_IAX && f.subclass.integer == IAX_COMMAND_NEW) {
- send_apathetic_reply(1, ntohs(fh->scallno), &sin, IAX_COMMAND_REJECT, ntohl(fh->ts), fh->iseqno + 1, fd, NULL);
- } else if (f.frametype == AST_FRAME_IAX && (f.subclass.integer == IAX_COMMAND_REGREQ || f.subclass.integer == IAX_COMMAND_REGREL)) {
- send_apathetic_reply(1, ntohs(fh->scallno), &sin, IAX_COMMAND_REGREJ, ntohl(fh->ts), fh->iseqno + 1, fd, NULL);
- }
- ast_variables_destroy(ies.vars);
- return 1;
- }
- }
- if (fr->callno > 0) {
- struct ast_callid *mount_callid;
- ast_mutex_lock(&iaxsl[fr->callno]);
- if (iaxs[fr->callno] && ((mount_callid = iax_pvt_callid_get(fr->callno)))) {
- /* Bind to thread */
- ast_callid_threadassoc_add(mount_callid);
- ast_callid_unref(mount_callid);
- }
- }
- if (!fr->callno || !iaxs[fr->callno]) {
- /* A call arrived for a nonexistent destination. Unless it's an "inval"
- frame, reply with an inval */
- if (ntohs(mh->callno) & IAX_FLAG_FULL) {
- /* We can only raw hangup control frames */
- if (((f.subclass.integer != IAX_COMMAND_INVAL) &&
- (f.subclass.integer != IAX_COMMAND_TXCNT) &&
- (f.subclass.integer != IAX_COMMAND_TXACC) &&
- (f.subclass.integer != IAX_COMMAND_FWDOWNL))||
- (f.frametype != AST_FRAME_IAX))
- raw_hangup(&sin, ntohs(fh->dcallno) & ~IAX_FLAG_RETRANS, ntohs(mh->callno) & ~IAX_FLAG_FULL,
- fd);
- }
- if (fr->callno > 0)
- ast_mutex_unlock(&iaxsl[fr->callno]);
- ast_variables_destroy(ies.vars);
- return 1;
- }
- if (ast_test_flag64(iaxs[fr->callno], IAX_ENCRYPTED) && !decrypted) {
- if (decrypt_frame(fr->callno, fh, &f, &res)) {
- ast_log(LOG_NOTICE, "Packet Decrypt Failed!\n");
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- decrypted = 1;
- }
- #ifdef DEBUG_SUPPORT
- if (decrypted) {
- iax_outputframe(NULL, fh, 3, &sin, res - sizeof(*fh));
- }
- #endif
- if (iaxs[fr->callno]->owner && fh->type == AST_FRAME_IAX &&
- (fh->csub == IAX_COMMAND_HANGUP
- || fh->csub == IAX_COMMAND_REJECT
- || fh->csub == IAX_COMMAND_REGREJ
- || fh->csub == IAX_COMMAND_TXREJ)) {
- struct ast_control_pvt_cause_code *cause_code;
- int data_size = sizeof(*cause_code);
- char subclass[40] = "";
- /* get subclass text */
- iax_frame_subclass2str(fh->csub, subclass, sizeof(subclass));
- /* add length of "IAX2 " */
- data_size += 5;
- /* for IAX hangup frames, add length of () and number */
- data_size += 3;
- if (ies.causecode > 9) {
- data_size++;
- }
- if (ies.causecode > 99) {
- data_size++;
- }
- /* add length of subclass */
- data_size += strlen(subclass);
- cause_code = ast_alloca(data_size);
- memset(cause_code, 0, data_size);
- ast_copy_string(cause_code->chan_name, ast_channel_name(iaxs[fr->callno]->owner), AST_CHANNEL_NAME);
- cause_code->ast_cause = ies.causecode;
- snprintf(cause_code->code, data_size - sizeof(*cause_code) + 1, "IAX2 %s(%d)", subclass, ies.causecode);
- iax2_lock_owner(fr->callno);
- if (iaxs[fr->callno] && iaxs[fr->callno]->owner) {
- ast_queue_control_data(iaxs[fr->callno]->owner, AST_CONTROL_PVT_CAUSE_CODE, cause_code, data_size);
- ast_channel_hangupcause_hash_set(iaxs[fr->callno]->owner, cause_code, data_size);
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- if (!iaxs[fr->callno]) {
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- }
- /* count this frame */
- iaxs[fr->callno]->frames_received++;
- if (!inaddrcmp(&sin, &iaxs[fr->callno]->addr) && !minivid &&
- f.subclass.integer != IAX_COMMAND_TXCNT && /* for attended transfer */
- f.subclass.integer != IAX_COMMAND_TXACC) { /* for attended transfer */
- unsigned short new_peercallno;
- new_peercallno = (unsigned short) (ntohs(mh->callno) & ~IAX_FLAG_FULL);
- if (new_peercallno && new_peercallno != iaxs[fr->callno]->peercallno) {
- if (iaxs[fr->callno]->peercallno) {
- remove_by_peercallno(iaxs[fr->callno]);
- }
- iaxs[fr->callno]->peercallno = new_peercallno;
- store_by_peercallno(iaxs[fr->callno]);
- }
- }
- if (ntohs(mh->callno) & IAX_FLAG_FULL) {
- if (iaxdebug)
- ast_debug(1, "Received packet %d, (%u, %d)\n", fh->oseqno, f.frametype, f.subclass.integer);
- /* Check if it's out of order (and not an ACK or INVAL) */
- fr->oseqno = fh->oseqno;
- fr->iseqno = fh->iseqno;
- fr->ts = ntohl(fh->ts);
- #ifdef IAXTESTS
- if (test_resync) {
- ast_debug(1, "Simulating frame ts resync, was %u now %u\n", fr->ts, fr->ts + test_resync);
- fr->ts += test_resync;
- }
- #endif /* IAXTESTS */
- #if 0
- if ( (ntohs(fh->dcallno) & IAX_FLAG_RETRANS) ||
- ( (f.frametype != AST_FRAME_VOICE) && ! (f.frametype == AST_FRAME_IAX &&
- (f.subclass == IAX_COMMAND_NEW ||
- f.subclass == IAX_COMMAND_AUTHREQ ||
- f.subclass == IAX_COMMAND_ACCEPT ||
- f.subclass == IAX_COMMAND_REJECT)) ) )
- #endif
- if ((ntohs(fh->dcallno) & IAX_FLAG_RETRANS) || (f.frametype != AST_FRAME_VOICE))
- updatehistory = 0;
- if ((iaxs[fr->callno]->iseqno != fr->oseqno) &&
- (iaxs[fr->callno]->iseqno ||
- ((f.subclass.integer != IAX_COMMAND_TXCNT) &&
- (f.subclass.integer != IAX_COMMAND_TXREADY) && /* for attended transfer */
- (f.subclass.integer != IAX_COMMAND_TXREL) && /* for attended transfer */
- (f.subclass.integer != IAX_COMMAND_UNQUELCH ) && /* for attended transfer */
- (f.subclass.integer != IAX_COMMAND_TXACC)) ||
- (f.frametype != AST_FRAME_IAX))) {
- if (
- ((f.subclass.integer != IAX_COMMAND_ACK) &&
- (f.subclass.integer != IAX_COMMAND_INVAL) &&
- (f.subclass.integer != IAX_COMMAND_TXCNT) &&
- (f.subclass.integer != IAX_COMMAND_TXREADY) && /* for attended transfer */
- (f.subclass.integer != IAX_COMMAND_TXREL) && /* for attended transfer */
- (f.subclass.integer != IAX_COMMAND_UNQUELCH ) && /* for attended transfer */
- (f.subclass.integer != IAX_COMMAND_TXACC) &&
- (f.subclass.integer != IAX_COMMAND_VNAK)) ||
- (f.frametype != AST_FRAME_IAX)) {
- /* If it's not an ACK packet, it's out of order. */
- ast_debug(1, "Packet arrived out of order (expecting %d, got %d) (frametype = %u, subclass = %d)\n",
- iaxs[fr->callno]->iseqno, fr->oseqno, f.frametype, f.subclass.integer);
- /* Check to see if we need to request retransmission,
- * and take sequence number wraparound into account */
- if ((unsigned char) (iaxs[fr->callno]->iseqno - fr->oseqno) < 128) {
- /* If we've already seen it, ack it XXX There's a border condition here XXX */
- if ((f.frametype != AST_FRAME_IAX) ||
- ((f.subclass.integer != IAX_COMMAND_ACK) && (f.subclass.integer != IAX_COMMAND_INVAL))) {
- ast_debug(1, "Acking anyway\n");
- /* XXX Maybe we should handle its ack to us, but then again, it's probably outdated anyway, and if
- we have anything to send, we'll retransmit and get an ACK back anyway XXX */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- }
- } else {
- /* Send a VNAK requesting retransmission */
- iax2_vnak(fr->callno);
- }
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- } else {
- /* Increment unless it's an ACK or VNAK */
- if (((f.subclass.integer != IAX_COMMAND_ACK) &&
- (f.subclass.integer != IAX_COMMAND_INVAL) &&
- (f.subclass.integer != IAX_COMMAND_TXCNT) &&
- (f.subclass.integer != IAX_COMMAND_TXACC) &&
- (f.subclass.integer != IAX_COMMAND_VNAK)) ||
- (f.frametype != AST_FRAME_IAX))
- iaxs[fr->callno]->iseqno++;
- }
- /* Ensure text frames are NULL-terminated */
- if (f.frametype == AST_FRAME_TEXT && thread->buf[res - 1] != '\0') {
- if (res < thread->buf_size)
- thread->buf[res++] = '\0';
- else /* Trims one character from the text message, but that's better than overwriting the end of the buffer. */
- thread->buf[res - 1] = '\0';
- }
- /* Handle implicit ACKing unless this is an INVAL, and only if this is
- from the real peer, not the transfer peer */
- if (!inaddrcmp(&sin, &iaxs[fr->callno]->addr) &&
- ((f.subclass.integer != IAX_COMMAND_INVAL) ||
- (f.frametype != AST_FRAME_IAX))) {
- unsigned char x;
- int call_to_destroy;
- /* First we have to qualify that the ACKed value is within our window */
- if (iaxs[fr->callno]->rseqno >= iaxs[fr->callno]->oseqno || (fr->iseqno >= iaxs[fr->callno]->rseqno && fr->iseqno < iaxs[fr->callno]->oseqno))
- x = fr->iseqno;
- else
- x = iaxs[fr->callno]->oseqno;
- if ((x != iaxs[fr->callno]->oseqno) || (iaxs[fr->callno]->oseqno == fr->iseqno)) {
- /* The acknowledgement is within our window. Time to acknowledge everything
- that it says to */
- for (x=iaxs[fr->callno]->rseqno; x != fr->iseqno; x++) {
- /* Ack the packet with the given timestamp */
- if (iaxdebug)
- ast_debug(1, "Cancelling transmission of packet %d\n", x);
- call_to_destroy = 0;
- AST_LIST_TRAVERSE(&frame_queue[fr->callno], cur, list) {
- /* If it's our call, and our timestamp, mark -1 retries */
- if (x == cur->oseqno) {
- cur->retries = -1;
- /* Destroy call if this is the end */
- if (cur->final)
- call_to_destroy = fr->callno;
- }
- }
- if (call_to_destroy) {
- if (iaxdebug)
- ast_debug(1, "Really destroying %d, having been acked on final message\n", call_to_destroy);
- ast_mutex_lock(&iaxsl[call_to_destroy]);
- iax2_destroy(call_to_destroy);
- ast_mutex_unlock(&iaxsl[call_to_destroy]);
- }
- }
- /* Note how much we've received acknowledgement for */
- if (iaxs[fr->callno])
- iaxs[fr->callno]->rseqno = fr->iseqno;
- else {
- /* Stop processing now */
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- } else {
- ast_debug(1, "Received iseqno %d not within window %d->%d\n", fr->iseqno, iaxs[fr->callno]->rseqno, iaxs[fr->callno]->oseqno);
- }
- }
- if (inaddrcmp(&sin, &iaxs[fr->callno]->addr) &&
- ((f.frametype != AST_FRAME_IAX) ||
- ((f.subclass.integer != IAX_COMMAND_TXACC) &&
- (f.subclass.integer != IAX_COMMAND_TXCNT)))) {
- /* Only messages we accept from a transfer host are TXACC and TXCNT */
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- /* when we receive the first full frame for a new incoming channel,
- it is safe to start the PBX on the channel because we have now
- completed a 3-way handshake with the peer */
- if ((f.frametype == AST_FRAME_VOICE) ||
- (f.frametype == AST_FRAME_VIDEO) ||
- (f.frametype == AST_FRAME_IAX)) {
- if (ast_test_flag64(iaxs[fr->callno], IAX_DELAYPBXSTART)) {
- ast_clear_flag64(iaxs[fr->callno], IAX_DELAYPBXSTART);
- if (!ast_iax2_new(fr->callno, AST_STATE_RING, iaxs[fr->callno]->chosenformat, NULL,
- ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_AUTHENTICATED))) {
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- }
- if (ies.vars) {
- struct ast_datastore *variablestore = NULL;
- struct ast_variable *var, *prev = NULL;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- iax2_lock_owner(fr->callno);
- if (!iaxs[fr->callno]) {
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- if ((c = iaxs[fr->callno]->owner)) {
- varlist = ast_calloc(1, sizeof(*varlist));
- variablestore = ast_datastore_alloc(&iax2_variable_datastore_info, NULL);
- if (variablestore && varlist) {
- variablestore->data = varlist;
- variablestore->inheritance = DATASTORE_INHERIT_FOREVER;
- AST_LIST_HEAD_INIT(varlist);
- ast_debug(1, "I can haz IAX vars?\n");
- for (var = ies.vars; var; var = var->next) {
- struct ast_var_t *newvar = ast_var_assign(var->name, var->value);
- if (prev) {
- ast_free(prev);
- }
- prev = var;
- if (!newvar) {
- /* Don't abort list traversal, as this would leave ies.vars in an inconsistent state. */
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- } else {
- AST_LIST_INSERT_TAIL(varlist, newvar, entries);
- }
- }
- if (prev) {
- ast_free(prev);
- }
- ies.vars = NULL;
- ast_channel_datastore_add(c, variablestore);
- } else {
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- if (variablestore) {
- ast_datastore_free(variablestore);
- }
- if (varlist) {
- ast_free(varlist);
- }
- }
- ast_channel_unlock(c);
- } else {
- /* No channel yet, so transfer the variables directly over to the pvt,
- * for later inheritance. */
- ast_debug(1, "No channel, so populating IAXVARs to the pvt, as an intermediate step.\n");
- for (var = ies.vars; var && var->next; var = var->next);
- if (var) {
- var->next = iaxs[fr->callno]->iaxvars;
- iaxs[fr->callno]->iaxvars = ies.vars;
- ies.vars = NULL;
- }
- }
- }
- if (ies.vars) {
- ast_debug(1, "I have IAX variables, but they were not processed\n");
- }
- }
- /* once we receive our first IAX Full Frame that is not CallToken related, send all
- * queued signaling frames that were being held. */
- if ((f.frametype == AST_FRAME_IAX) && (f.subclass.integer != IAX_COMMAND_CALLTOKEN) && iaxs[fr->callno]->hold_signaling) {
- send_signaling(iaxs[fr->callno]);
- }
- if (f.frametype == AST_FRAME_VOICE) {
- if (ast_format_to_old_bitfield(&f.subclass.format) != iaxs[fr->callno]->voiceformat) {
- iaxs[fr->callno]->voiceformat = ast_format_to_old_bitfield(&f.subclass.format);
- ast_debug(1, "Ooh, voice format changed to '%s'\n", ast_getformatname(&f.subclass.format));
- if (iaxs[fr->callno]->owner) {
- iax2_lock_owner(fr->callno);
- if (iaxs[fr->callno]) {
- if (iaxs[fr->callno]->owner) {
- struct ast_format_cap *orignative = ast_format_cap_dup(ast_channel_nativeformats(iaxs[fr->callno]->owner));
- struct ast_format_cap *native = ast_channel_nativeformats(iaxs[fr->callno]->owner);
- if (orignative) {
- ast_format_cap_set(native, &f.subclass.format);
- if (ast_channel_readformat(iaxs[fr->callno]->owner)->id) {
- ast_set_read_format(iaxs[fr->callno]->owner, ast_channel_readformat(iaxs[fr->callno]->owner));
- }
- ast_format_cap_copy(native, orignative);
- orignative = ast_format_cap_destroy(orignative);
- }
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- } else {
- ast_debug(1, "Neat, somebody took away the channel at a magical time but i found it!\n");
- /* Free remote variables (if any) */
- if (ies.vars) {
- ast_variables_destroy(ies.vars);
- ast_debug(1, "I can haz iaxvars, but they is no good. :-(\n");
- ies.vars = NULL;
- }
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- }
- }
- }
- if (f.frametype == AST_FRAME_VIDEO) {
- if (f.subclass.format.id != ast_format_id_from_old_bitfield(iaxs[fr->callno]->videoformat)) {
- ast_debug(1, "Ooh, video format changed to %s\n", ast_getformatname(&f.subclass.format));
- iaxs[fr->callno]->videoformat = ast_format_to_old_bitfield(&f.subclass.format);
- }
- }
- if (f.frametype == AST_FRAME_IAX) {
- AST_SCHED_DEL(sched, iaxs[fr->callno]->initid);
- /* Handle the IAX pseudo frame itself */
- if (iaxdebug)
- ast_debug(1, "IAX subclass %d received\n", f.subclass.integer);
- /* Update last ts unless the frame's timestamp originated with us. */
- if (iaxs[fr->callno]->last < fr->ts &&
- f.subclass.integer != IAX_COMMAND_ACK &&
- f.subclass.integer != IAX_COMMAND_PONG &&
- f.subclass.integer != IAX_COMMAND_LAGRP) {
- iaxs[fr->callno]->last = fr->ts;
- if (iaxdebug)
- ast_debug(1, "For call=%d, set last=%u\n", fr->callno, fr->ts);
- }
- iaxs[fr->callno]->last_iax_message = f.subclass.integer;
- if (!iaxs[fr->callno]->first_iax_message) {
- iaxs[fr->callno]->first_iax_message = f.subclass.integer;
- }
- switch(f.subclass.integer) {
- case IAX_COMMAND_ACK:
- /* Do nothing */
- break;
- case IAX_COMMAND_QUELCH:
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED)) {
- /* Generate Manager Hold event, if necessary*/
- if (iaxs[fr->callno]->owner) {
- ast_manager_event(iaxs[fr->callno]->owner, EVENT_FLAG_CALL, "Hold",
- "Status: On\r\n"
- "Channel: %s\r\n"
- "Uniqueid: %s\r\n",
- ast_channel_name(iaxs[fr->callno]->owner),
- ast_channel_uniqueid(iaxs[fr->callno]->owner));
- }
- ast_set_flag64(iaxs[fr->callno], IAX_QUELCH);
- if (ies.musiconhold) {
- iax2_lock_owner(fr->callno);
- if (!iaxs[fr->callno] || !iaxs[fr->callno]->owner) {
- break;
- }
- if (ast_bridged_channel(iaxs[fr->callno]->owner)) {
- const char *moh_suggest = iaxs[fr->callno]->mohsuggest;
- /*
- * We already hold the owner lock so we do not
- * need to check iaxs[fr->callno] after it returns.
- */
- iax2_queue_control_data(fr->callno, AST_CONTROL_HOLD,
- S_OR(moh_suggest, NULL),
- !ast_strlen_zero(moh_suggest) ? strlen(moh_suggest) + 1 : 0);
- }
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- }
- break;
- case IAX_COMMAND_UNQUELCH:
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED)) {
- iax2_lock_owner(fr->callno);
- if (!iaxs[fr->callno]) {
- break;
- }
- /* Generate Manager Unhold event, if necessary */
- if (iaxs[fr->callno]->owner && ast_test_flag64(iaxs[fr->callno], IAX_QUELCH)) {
- ast_manager_event(iaxs[fr->callno]->owner, EVENT_FLAG_CALL, "Hold",
- "Status: Off\r\n"
- "Channel: %s\r\n"
- "Uniqueid: %s\r\n",
- ast_channel_name(iaxs[fr->callno]->owner),
- ast_channel_uniqueid(iaxs[fr->callno]->owner));
- }
- ast_clear_flag64(iaxs[fr->callno], IAX_QUELCH);
- if (!iaxs[fr->callno]->owner) {
- break;
- }
- if (ast_bridged_channel(iaxs[fr->callno]->owner)) {
- /*
- * We already hold the owner lock so we do not
- * need to check iaxs[fr->callno] after it returns.
- */
- iax2_queue_control_data(fr->callno, AST_CONTROL_UNHOLD, NULL, 0);
- }
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- break;
- case IAX_COMMAND_TXACC:
- if (iaxs[fr->callno]->transferring == TRANSFER_BEGIN) {
- /* Ack the packet with the given timestamp */
- AST_LIST_TRAVERSE(&frame_queue[fr->callno], cur, list) {
- /* Cancel any outstanding txcnt's */
- if (cur->transfer) {
- cur->retries = -1;
- }
- }
- memset(&ied1, 0, sizeof(ied1));
- iax_ie_append_short(&ied1, IAX_IE_CALLNO, iaxs[fr->callno]->callno);
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_TXREADY, 0, ied1.buf, ied1.pos, -1);
- iaxs[fr->callno]->transferring = TRANSFER_READY;
- }
- break;
- case IAX_COMMAND_NEW:
- /* Ignore if it's already up */
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED | IAX_STATE_TBD))
- break;
- if (ies.provverpres && ies.serviceident && sin.sin_addr.s_addr) {
- ast_mutex_unlock(&iaxsl[fr->callno]);
- check_provisioning(&sin, fd, ies.serviceident, ies.provver);
- ast_mutex_lock(&iaxsl[fr->callno]);
- if (!iaxs[fr->callno]) {
- break;
- }
- }
- /* If we're in trunk mode, do it now, and update the trunk number in our frame before continuing */
- if (ast_test_flag64(iaxs[fr->callno], IAX_TRUNK)) {
- int new_callno;
- if ((new_callno = make_trunk(fr->callno, 1)) != -1)
- fr->callno = new_callno;
- }
- /* For security, always ack immediately */
- if (delayreject)
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- if (check_access(fr->callno, &sin, &ies)) {
- /* They're not allowed on */
- auth_fail(fr->callno, IAX_COMMAND_REJECT);
- if (authdebug)
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, who was trying to reach '%s@%s'\n", ast_inet_ntoa(sin.sin_addr), iaxs[fr->callno]->exten, iaxs[fr->callno]->context);
- break;
- }
- if (ast_strlen_zero(iaxs[fr->callno]->secret) && ast_test_flag64(iaxs[fr->callno], IAX_FORCE_ENCRYPT)) {
- auth_fail(fr->callno, IAX_COMMAND_REJECT);
- ast_log(LOG_WARNING, "Rejected connect attempt. No secret present while force encrypt enabled.\n");
- break;
- }
- if (strcasecmp(iaxs[fr->callno]->exten, "TBD")) {
- const char *context, *exten, *cid_num;
- context = ast_strdupa(iaxs[fr->callno]->context);
- exten = ast_strdupa(iaxs[fr->callno]->exten);
- cid_num = ast_strdupa(iaxs[fr->callno]->cid_num);
- /* This might re-enter the IAX code and need the lock */
- ast_mutex_unlock(&iaxsl[fr->callno]);
- exists = ast_exists_extension(NULL, context, exten, 1, cid_num);
- ast_mutex_lock(&iaxsl[fr->callno]);
- if (!iaxs[fr->callno]) {
- break;
- }
- } else
- exists = 0;
- /* Get OSP token if it does exist */
- save_osptoken(fr, &ies);
- if (ast_strlen_zero(iaxs[fr->callno]->secret) && ast_strlen_zero(iaxs[fr->callno]->inkeys)) {
- if (strcmp(iaxs[fr->callno]->exten, "TBD") && !exists) {
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "No such context/extension");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_NO_ROUTE_DESTINATION);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- if (authdebug)
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, request '%s@%s' does not exist\n", ast_inet_ntoa(sin.sin_addr), iaxs[fr->callno]->exten, iaxs[fr->callno]->context);
- } else {
- /* Select an appropriate format */
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOPREFS)) {
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- using_prefs = "reqonly";
- } else {
- using_prefs = "disabled";
- }
- format = iaxs[fr->callno]->peerformat & iaxs[fr->callno]->capability;
- memset(&pref, 0, sizeof(pref));
- strcpy(caller_pref_buf, "disabled");
- strcpy(host_pref_buf, "disabled");
- } else {
- struct ast_format tmpfmt;
- using_prefs = "mine";
- /* If the information elements are in here... use them */
- if (ies.codec_prefs)
- ast_codec_pref_convert(&iaxs[fr->callno]->rprefs, ies.codec_prefs, 32, 0);
- if (ast_codec_pref_index(&iaxs[fr->callno]->rprefs, 0, &tmpfmt)) {
- /* If we are codec_first_choice we let the caller have the 1st shot at picking the codec.*/
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_USER_FIRST)) {
- pref = iaxs[fr->callno]->rprefs;
- using_prefs = "caller";
- } else {
- pref = iaxs[fr->callno]->prefs;
- }
- } else
- pref = iaxs[fr->callno]->prefs;
- format = iax2_codec_choose(&pref, iaxs[fr->callno]->capability & iaxs[fr->callno]->peercapability, 0);
- ast_codec_pref_string(&iaxs[fr->callno]->rprefs, caller_pref_buf, sizeof(caller_pref_buf) - 1);
- ast_codec_pref_string(&iaxs[fr->callno]->prefs, host_pref_buf, sizeof(host_pref_buf) - 1);
- }
- if (!format) {
- if(!ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP))
- format = iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability;
- if (!format) {
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "Unable to negotiate codec");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_BEARERCAPABILITY_NOTAVAIL);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- if (authdebug) {
- char tmp[256], tmp2[256], tmp3[256];
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested '%s' incompatible with our capability '%s'.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp, sizeof(tmp), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->capability));
- } else {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested/capability '%s'/'%s' incompatible with our capability '%s'.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp, sizeof(tmp), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->peercapability),
- iax2_getformatname_multiple(tmp3, sizeof(tmp3), iaxs[fr->callno]->capability));
- }
- }
- } else {
- /* Pick one... */
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- if(!(iaxs[fr->callno]->peerformat & iaxs[fr->callno]->capability))
- format = 0;
- } else {
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOPREFS)) {
- using_prefs = ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP) ? "reqonly" : "disabled";
- memset(&pref, 0, sizeof(pref));
- format = iax2_best_codec(iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability);
- strcpy(caller_pref_buf,"disabled");
- strcpy(host_pref_buf,"disabled");
- } else {
- struct ast_format tmpfmt;
- using_prefs = "mine";
- if (ast_codec_pref_index(&iaxs[fr->callno]->rprefs, 0, &tmpfmt)) {
- /* Do the opposite of what we tried above. */
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_USER_FIRST)) {
- pref = iaxs[fr->callno]->prefs;
- } else {
- pref = iaxs[fr->callno]->rprefs;
- using_prefs = "caller";
- }
- format = iax2_codec_choose(&pref, iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability, 1);
- } else /* if no codec_prefs IE do it the old way */
- format = iax2_best_codec(iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability);
- }
- }
- if (!format) {
- char tmp[256], tmp2[256], tmp3[256];
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "Unable to negotiate codec");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_BEARERCAPABILITY_NOTAVAIL);
- ast_log(LOG_ERROR, "No best format in '%s'???\n", iax2_getformatname_multiple(tmp, sizeof(tmp), iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability));
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- if (authdebug) {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested/capability '%s'/'%s' incompatible with our capability '%s'.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp, sizeof(tmp), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->peercapability),
- iax2_getformatname_multiple(tmp3, sizeof(tmp3), iaxs[fr->callno]->capability));
- }
- ast_set_flag64(iaxs[fr->callno], IAX_ALREADYGONE);
- break;
- }
- }
- }
- if (format) {
- /* No authentication required, let them in */
- memset(&ied1, 0, sizeof(ied1));
- iax_ie_append_int(&ied1, IAX_IE_FORMAT, format);
- iax_ie_append_versioned_uint64(&ied1, IAX_IE_FORMAT2, 0, format);
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACCEPT, 0, ied1.buf, ied1.pos, -1);
- if (strcmp(iaxs[fr->callno]->exten, "TBD")) {
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED);
- ast_verb(3, "Accepting UNAUTHENTICATED call from %s:\n"
- "%srequested format = %s,\n"
- "%srequested prefs = %s,\n"
- "%sactual format = %s,\n"
- "%shost prefs = %s,\n"
- "%spriority = %s\n",
- ast_inet_ntoa(sin.sin_addr),
- VERBOSE_PREFIX_4,
- iax2_getformatname(iaxs[fr->callno]->peerformat),
- VERBOSE_PREFIX_4,
- caller_pref_buf,
- VERBOSE_PREFIX_4,
- iax2_getformatname(format),
- VERBOSE_PREFIX_4,
- host_pref_buf,
- VERBOSE_PREFIX_4,
- using_prefs);
- iaxs[fr->callno]->chosenformat = format;
- /* Since this is a new call, we should go ahead and set the callid for it. */
- iax_pvt_callid_new(fr->callno);
- ast_set_flag64(iaxs[fr->callno], IAX_DELAYPBXSTART);
- } else {
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_TBD);
- /* If this is a TBD call, we're ready but now what... */
- ast_verb(3, "Accepted unauthenticated TBD call from %s\n", ast_inet_ntoa(sin.sin_addr));
- }
- }
- }
- break;
- }
- if (iaxs[fr->callno]->authmethods & IAX_AUTH_MD5)
- merge_encryption(iaxs[fr->callno],ies.encmethods);
- else
- iaxs[fr->callno]->encmethods = 0;
- if (!authenticate_request(fr->callno) && iaxs[fr->callno])
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_AUTHENTICATED);
- break;
- case IAX_COMMAND_DPREQ:
- /* Request status in the dialplan */
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_TBD) &&
- !ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED) && ies.called_number) {
- if (iaxcompat) {
- /* Spawn a thread for the lookup */
- spawn_dp_lookup(fr->callno, iaxs[fr->callno]->context, ies.called_number, iaxs[fr->callno]->cid_num);
- } else {
- /* Just look it up */
- dp_lookup(fr->callno, iaxs[fr->callno]->context, ies.called_number, iaxs[fr->callno]->cid_num, 1);
- }
- }
- break;
- case IAX_COMMAND_HANGUP:
- ast_set_flag64(iaxs[fr->callno], IAX_ALREADYGONE);
- ast_debug(1, "Immediately destroying %d, having received hangup\n", fr->callno);
- /* Set hangup cause according to remote and hangupsource */
- if (iaxs[fr->callno]->owner) {
- set_hangup_source_and_cause(fr->callno, ies.causecode);
- if (!iaxs[fr->callno]) {
- break;
- }
- }
- /* Send ack immediately, before we destroy */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- iax2_destroy(fr->callno);
- break;
- case IAX_COMMAND_REJECT:
- /* Set hangup cause according to remote and hangup source */
- if (iaxs[fr->callno]->owner) {
- set_hangup_source_and_cause(fr->callno, ies.causecode);
- if (!iaxs[fr->callno]) {
- break;
- }
- }
- if (!ast_test_flag64(iaxs[fr->callno], IAX_PROVISION)) {
- if (iaxs[fr->callno]->owner && authdebug)
- ast_log(LOG_WARNING, "Call rejected by %s: %s\n",
- ast_inet_ntoa(iaxs[fr->callno]->addr.sin_addr),
- ies.cause ? ies.cause : "<Unknown>");
- ast_debug(1, "Immediately destroying %d, having received reject\n",
- fr->callno);
- }
- /* Send ack immediately, before we destroy */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK,
- fr->ts, NULL, 0, fr->iseqno);
- if (!ast_test_flag64(iaxs[fr->callno], IAX_PROVISION))
- iaxs[fr->callno]->error = EPERM;
- iax2_destroy(fr->callno);
- break;
- case IAX_COMMAND_TRANSFER:
- {
- struct ast_channel *bridged_chan;
- struct ast_channel *owner;
- iax2_lock_owner(fr->callno);
- if (!iaxs[fr->callno]) {
- /* Initiating call went away before we could transfer. */
- break;
- }
- owner = iaxs[fr->callno]->owner;
- bridged_chan = owner ? ast_bridged_channel(owner) : NULL;
- if (bridged_chan && ies.called_number) {
- const char *context;
- context = ast_strdupa(iaxs[fr->callno]->context);
- ast_channel_ref(owner);
- ast_channel_ref(bridged_chan);
- ast_channel_unlock(owner);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- /* Set BLINDTRANSFER channel variables */
- pbx_builtin_setvar_helper(owner, "BLINDTRANSFER", ast_channel_name(bridged_chan));
- pbx_builtin_setvar_helper(bridged_chan, "BLINDTRANSFER", ast_channel_name(owner));
- /* DO NOT hold any locks while calling ast_parking_ext_valid() */
- if (ast_parking_ext_valid(ies.called_number, owner, context)) {
- ast_debug(1, "Parking call '%s'\n", ast_channel_name(bridged_chan));
- if (iax_park(bridged_chan, owner, ies.called_number, context)) {
- ast_log(LOG_WARNING, "Failed to park call '%s'\n",
- ast_channel_name(bridged_chan));
- }
- } else {
- if (ast_async_goto(bridged_chan, context, ies.called_number, 1)) {
- ast_log(LOG_WARNING,
- "Async goto of '%s' to '%s@%s' failed\n",
- ast_channel_name(bridged_chan), ies.called_number, context);
- } else {
- ast_debug(1, "Async goto of '%s' to '%s@%s' started\n",
- ast_channel_name(bridged_chan), ies.called_number, context);
- }
- }
- ast_channel_unref(owner);
- ast_channel_unref(bridged_chan);
- ast_mutex_lock(&iaxsl[fr->callno]);
- } else {
- ast_debug(1, "Async goto not applicable on call %d\n", fr->callno);
- if (owner) {
- ast_channel_unlock(owner);
- }
- }
- break;
- }
- case IAX_COMMAND_ACCEPT:
- /* Ignore if call is already up or needs authentication or is a TBD */
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED | IAX_STATE_TBD | IAX_STATE_AUTHENTICATED))
- break;
- if (ast_test_flag64(iaxs[fr->callno], IAX_PROVISION)) {
- /* Send ack immediately, before we destroy */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- iax2_destroy(fr->callno);
- break;
- }
- if (ies.format) {
- iaxs[fr->callno]->peerformat = ies.format;
- } else {
- if (iaxs[fr->callno]->owner)
- iaxs[fr->callno]->peerformat = ast_format_cap_to_old_bitfield(ast_channel_nativeformats(iaxs[fr->callno]->owner));
- else
- iaxs[fr->callno]->peerformat = iaxs[fr->callno]->capability;
- }
- ast_verb(3, "Call accepted by %s (format %s)\n", ast_inet_ntoa(iaxs[fr->callno]->addr.sin_addr), iax2_getformatname(iaxs[fr->callno]->peerformat));
- if (!(iaxs[fr->callno]->peerformat & iaxs[fr->callno]->capability)) {
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "Unable to negotiate codec");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_BEARERCAPABILITY_NOTAVAIL);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- if (authdebug) {
- char tmp1[256], tmp2[256];
- ast_log(LOG_NOTICE, "Rejected call to %s, format %s incompatible with our capability %s.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->capability));
- }
- } else {
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED);
- iax2_lock_owner(fr->callno);
- if (iaxs[fr->callno] && iaxs[fr->callno]->owner) {
- char tmp[256];
- /* Switch us to use a compatible format */
- ast_format_cap_from_old_bitfield(ast_channel_nativeformats(iaxs[fr->callno]->owner), iaxs[fr->callno]->peerformat);
- ast_verb(3, "Format for call is %s\n", ast_getformatname_multiple(tmp, sizeof(tmp), ast_channel_nativeformats(iaxs[fr->callno]->owner)));
- /* Setup read/write formats properly. */
- if (ast_channel_writeformat(iaxs[fr->callno]->owner)->id)
- ast_set_write_format(iaxs[fr->callno]->owner, ast_channel_writeformat(iaxs[fr->callno]->owner));
- if (ast_channel_readformat(iaxs[fr->callno]->owner)->id)
- ast_set_read_format(iaxs[fr->callno]->owner, ast_channel_readformat(iaxs[fr->callno]->owner));
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- }
- if (iaxs[fr->callno]) {
- AST_LIST_LOCK(&dpcache);
- AST_LIST_TRAVERSE(&iaxs[fr->callno]->dpentries, dp, peer_list)
- if (!(dp->flags & CACHE_FLAG_TRANSMITTED))
- iax2_dprequest(dp, fr->callno);
- AST_LIST_UNLOCK(&dpcache);
- }
- break;
- case IAX_COMMAND_POKE:
- /* Send back a pong packet with the original timestamp */
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_PONG, fr->ts, NULL, 0, -1);
- break;
- case IAX_COMMAND_PING:
- {
- struct iax_ie_data pingied;
- construct_rr(iaxs[fr->callno], &pingied);
- /* Send back a pong packet with the original timestamp */
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_PONG, fr->ts, pingied.buf, pingied.pos, -1);
- }
- break;
- case IAX_COMMAND_PONG:
- /* Calculate ping time */
- iaxs[fr->callno]->pingtime = calc_timestamp(iaxs[fr->callno], 0, &f) - fr->ts;
- /* save RR info */
- save_rr(fr, &ies);
- /* Good time to write jb stats for this call */
- log_jitterstats(fr->callno);
- if (iaxs[fr->callno]->peerpoke) {
- peer = iaxs[fr->callno]->peerpoke;
- if ((peer->lastms < 0) || (peer->historicms > peer->maxms)) {
- if (iaxs[fr->callno]->pingtime <= peer->maxms) {
- ast_log(LOG_NOTICE, "Peer '%s' is now REACHABLE! Time: %u\n", peer->name, iaxs[fr->callno]->pingtime);
- manager_event(EVENT_FLAG_SYSTEM, "PeerStatus", "ChannelType: IAX2\r\nPeer: IAX2/%s\r\nPeerStatus: Reachable\r\nTime: %u\r\n", peer->name, iaxs[fr->callno]->pingtime);
- ast_devstate_changed(AST_DEVICE_NOT_INUSE, AST_DEVSTATE_CACHABLE, "IAX2/%s", peer->name); /* Activate notification */
- }
- } else if ((peer->historicms > 0) && (peer->historicms <= peer->maxms)) {
- if (iaxs[fr->callno]->pingtime > peer->maxms) {
- ast_log(LOG_NOTICE, "Peer '%s' is now TOO LAGGED (%u ms)!\n", peer->name, iaxs[fr->callno]->pingtime);
- manager_event(EVENT_FLAG_SYSTEM, "PeerStatus", "ChannelType: IAX2\r\nPeer: IAX2/%s\r\nPeerStatus: Lagged\r\nTime: %u\r\n", peer->name, iaxs[fr->callno]->pingtime);
- ast_devstate_changed(AST_DEVICE_UNAVAILABLE, AST_DEVSTATE_CACHABLE, "IAX2/%s", peer->name); /* Activate notification */
- }
- }
- peer->lastms = iaxs[fr->callno]->pingtime;
- if (peer->smoothing && (peer->lastms > -1))
- peer->historicms = (iaxs[fr->callno]->pingtime + peer->historicms) / 2;
- else if (peer->smoothing && peer->lastms < 0)
- peer->historicms = (0 + peer->historicms) / 2;
- else
- peer->historicms = iaxs[fr->callno]->pingtime;
- /* Remove scheduled iax2_poke_noanswer */
- if (peer->pokeexpire > -1) {
- if (!AST_SCHED_DEL(sched, peer->pokeexpire)) {
- peer_unref(peer);
- peer->pokeexpire = -1;
- }
- }
- /* Schedule the next cycle */
- if ((peer->lastms < 0) || (peer->historicms > peer->maxms))
- peer->pokeexpire = iax2_sched_add(sched, peer->pokefreqnotok, iax2_poke_peer_s, peer_ref(peer));
- else
- peer->pokeexpire = iax2_sched_add(sched, peer->pokefreqok, iax2_poke_peer_s, peer_ref(peer));
- if (peer->pokeexpire == -1)
- peer_unref(peer);
- /* and finally send the ack */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- /* And wrap up the qualify call */
- iax2_destroy(fr->callno);
- peer->callno = 0;
- ast_debug(1, "Peer %s: got pong, lastms %d, historicms %d, maxms %d\n", peer->name, peer->lastms, peer->historicms, peer->maxms);
- }
- break;
- case IAX_COMMAND_LAGRQ:
- case IAX_COMMAND_LAGRP:
- f.src = "LAGRQ";
- f.mallocd = 0;
- f.offset = 0;
- f.samples = 0;
- iax_frame_wrap(fr, &f);
- if (f.subclass.integer == IAX_COMMAND_LAGRQ) {
- /* Received a LAGRQ - echo back a LAGRP */
- fr->af.subclass.integer = IAX_COMMAND_LAGRP;
- iax2_send(iaxs[fr->callno], &fr->af, fr->ts, -1, 0, 0, 0);
- } else {
- /* Received LAGRP in response to our LAGRQ */
- unsigned int ts;
- /* This is a reply we've been given, actually measure the difference */
- ts = calc_timestamp(iaxs[fr->callno], 0, &fr->af);
- iaxs[fr->callno]->lag = ts - fr->ts;
- if (iaxdebug)
- ast_debug(1, "Peer %s lag measured as %dms\n",
- ast_inet_ntoa(iaxs[fr->callno]->addr.sin_addr), iaxs[fr->callno]->lag);
- }
- break;
- case IAX_COMMAND_AUTHREQ:
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED | IAX_STATE_TBD)) {
- ast_log(LOG_WARNING, "Call on %s is already up, can't start on it\n", iaxs[fr->callno]->owner ? ast_channel_name(iaxs[fr->callno]->owner) : "<Unknown>");
- break;
- }
- if (authenticate_reply(iaxs[fr->callno], &iaxs[fr->callno]->addr, &ies, iaxs[fr->callno]->secret, iaxs[fr->callno]->outkey)) {
- struct ast_frame hangup_fr = { .frametype = AST_FRAME_CONTROL,
- .subclass.integer = AST_CONTROL_HANGUP,
- };
- ast_log(LOG_WARNING,
- "I don't know how to authenticate %s to %s\n",
- ies.username ? ies.username : "<unknown>", ast_inet_ntoa(iaxs[fr->callno]->addr.sin_addr));
- iax2_queue_frame(fr->callno, &hangup_fr);
- }
- break;
- case IAX_COMMAND_AUTHREP:
- /* For security, always ack immediately */
- if (delayreject)
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- /* Ignore once we've started */
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED | IAX_STATE_TBD)) {
- ast_log(LOG_WARNING, "Call on %s is already up, can't start on it\n", iaxs[fr->callno]->owner ? ast_channel_name(iaxs[fr->callno]->owner) : "<Unknown>");
- break;
- }
- if (authenticate_verify(iaxs[fr->callno], &ies)) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Host %s failed to authenticate as %s\n", ast_inet_ntoa(iaxs[fr->callno]->addr.sin_addr), iaxs[fr->callno]->username);
- memset(&ied0, 0, sizeof(ied0));
- auth_fail(fr->callno, IAX_COMMAND_REJECT);
- break;
- }
- if (strcasecmp(iaxs[fr->callno]->exten, "TBD")) {
- /* This might re-enter the IAX code and need the lock */
- exists = ast_exists_extension(NULL, iaxs[fr->callno]->context, iaxs[fr->callno]->exten, 1, iaxs[fr->callno]->cid_num);
- } else
- exists = 0;
- if (strcmp(iaxs[fr->callno]->exten, "TBD") && !exists) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, request '%s@%s' does not exist\n", ast_inet_ntoa(sin.sin_addr), iaxs[fr->callno]->exten, iaxs[fr->callno]->context);
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "No such context/extension");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_NO_ROUTE_DESTINATION);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- } else {
- /* Select an appropriate format */
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOPREFS)) {
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- using_prefs = "reqonly";
- } else {
- using_prefs = "disabled";
- }
- format = iaxs[fr->callno]->peerformat & iaxs[fr->callno]->capability;
- memset(&pref, 0, sizeof(pref));
- strcpy(caller_pref_buf, "disabled");
- strcpy(host_pref_buf, "disabled");
- } else {
- struct ast_format tmpfmt;
- using_prefs = "mine";
- if (ies.codec_prefs)
- ast_codec_pref_convert(&iaxs[fr->callno]->rprefs, ies.codec_prefs, 32, 0);
- if (ast_codec_pref_index(&iaxs[fr->callno]->rprefs, 0, &tmpfmt)) {
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_USER_FIRST)) {
- pref = iaxs[fr->callno]->rprefs;
- using_prefs = "caller";
- } else {
- pref = iaxs[fr->callno]->prefs;
- }
- } else /* if no codec_prefs IE do it the old way */
- pref = iaxs[fr->callno]->prefs;
- format = iax2_codec_choose(&pref, iaxs[fr->callno]->capability & iaxs[fr->callno]->peercapability, 0);
- ast_codec_pref_string(&iaxs[fr->callno]->rprefs, caller_pref_buf, sizeof(caller_pref_buf) - 1);
- ast_codec_pref_string(&iaxs[fr->callno]->prefs, host_pref_buf, sizeof(host_pref_buf) - 1);
- }
- if (!format) {
- char tmp1[256], tmp2[256], tmp3[256];
- if(!ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- ast_debug(1, "We don't do requested format %s, falling back to peer capability '%s'\n",
- iax2_getformatname(iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peercapability));
- format = iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability;
- }
- if (!format) {
- if (authdebug) {
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested '%s' incompatible with our capability '%s'.\n", ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->capability));
- } else {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested/capability '%s'/'%s' incompatible with our capability '%s'.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->peercapability),
- iax2_getformatname_multiple(tmp3, sizeof(tmp3), iaxs[fr->callno]->capability));
- }
- }
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "Unable to negotiate codec");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_BEARERCAPABILITY_NOTAVAIL);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- } else {
- /* Pick one... */
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- if(!(iaxs[fr->callno]->peerformat & iaxs[fr->callno]->capability))
- format = 0;
- } else {
- if(ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOPREFS)) {
- using_prefs = ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP) ? "reqonly" : "disabled";
- memset(&pref, 0, sizeof(pref));
- format = ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP) ?
- iaxs[fr->callno]->peerformat : iax2_best_codec(iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability);
- strcpy(caller_pref_buf,"disabled");
- strcpy(host_pref_buf,"disabled");
- } else {
- struct ast_format tmpfmt;
- using_prefs = "mine";
- if (ast_codec_pref_index(&iaxs[fr->callno]->rprefs, 0, &tmpfmt)) {
- /* Do the opposite of what we tried above. */
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_USER_FIRST)) {
- pref = iaxs[fr->callno]->prefs;
- } else {
- pref = iaxs[fr->callno]->rprefs;
- using_prefs = "caller";
- }
- format = iax2_codec_choose(&pref, iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability, 1);
- } else /* if no codec_prefs IE do it the old way */
- format = iax2_best_codec(iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability);
- }
- }
- if (!format) {
- char tmp1[256], tmp2[256], tmp3[256];
- ast_log(LOG_ERROR, "No best format in %s???\n",
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peercapability & iaxs[fr->callno]->capability));
- if (authdebug) {
- if (ast_test_flag64(iaxs[fr->callno], IAX_CODEC_NOCAP)) {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested '%s' incompatible with our capability '%s'.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->capability));
- } else {
- ast_log(LOG_NOTICE, "Rejected connect attempt from %s, requested/capability '%s'/'%s' incompatible with our capability '%s'.\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp1, sizeof(tmp1), iaxs[fr->callno]->peerformat),
- iax2_getformatname_multiple(tmp2, sizeof(tmp2), iaxs[fr->callno]->peercapability),
- iax2_getformatname_multiple(tmp3, sizeof(tmp3), iaxs[fr->callno]->capability));
- }
- }
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "Unable to negotiate codec");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_BEARERCAPABILITY_NOTAVAIL);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- }
- }
- }
- if (format) {
- /* Authentication received */
- memset(&ied1, 0, sizeof(ied1));
- iax_ie_append_int(&ied1, IAX_IE_FORMAT, format);
- iax_ie_append_versioned_uint64(&ied1, IAX_IE_FORMAT2, 0, format);
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACCEPT, 0, ied1.buf, ied1.pos, -1);
- if (strcmp(iaxs[fr->callno]->exten, "TBD")) {
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED);
- ast_verb(3, "Accepting AUTHENTICATED call from %s:\n"
- "%srequested format = %s,\n"
- "%srequested prefs = %s,\n"
- "%sactual format = %s,\n"
- "%shost prefs = %s,\n"
- "%spriority = %s\n",
- ast_inet_ntoa(sin.sin_addr),
- VERBOSE_PREFIX_4,
- iax2_getformatname(iaxs[fr->callno]->peerformat),
- VERBOSE_PREFIX_4,
- caller_pref_buf,
- VERBOSE_PREFIX_4,
- iax2_getformatname(format),
- VERBOSE_PREFIX_4,
- host_pref_buf,
- VERBOSE_PREFIX_4,
- using_prefs);
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED);
- if (!(c = ast_iax2_new(fr->callno, AST_STATE_RING, format, NULL, 1)))
- iax2_destroy(fr->callno);
- else if (ies.vars) {
- struct ast_datastore *variablestore;
- struct ast_variable *var, *prev = NULL;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- varlist = ast_calloc(1, sizeof(*varlist));
- variablestore = ast_datastore_alloc(&iax2_variable_datastore_info, NULL);
- if (variablestore && varlist) {
- variablestore->data = varlist;
- variablestore->inheritance = DATASTORE_INHERIT_FOREVER;
- AST_LIST_HEAD_INIT(varlist);
- ast_debug(1, "I can haz IAX vars? w00t\n");
- for (var = ies.vars; var; var = var->next) {
- struct ast_var_t *newvar = ast_var_assign(var->name, var->value);
- if (prev)
- ast_free(prev);
- prev = var;
- if (!newvar) {
- /* Don't abort list traversal, as this would leave ies.vars in an inconsistent state. */
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- } else {
- AST_LIST_INSERT_TAIL(varlist, newvar, entries);
- }
- }
- if (prev)
- ast_free(prev);
- ies.vars = NULL;
- ast_channel_datastore_add(c, variablestore);
- } else {
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- if (variablestore)
- ast_datastore_free(variablestore);
- if (varlist)
- ast_free(varlist);
- }
- }
- } else {
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_TBD);
- /* If this is a TBD call, we're ready but now what... */
- ast_verb(3, "Accepted AUTHENTICATED TBD call from %s\n", ast_inet_ntoa(sin.sin_addr));
- if (ast_test_flag64(iaxs[fr->callno], IAX_IMMEDIATE)) {
- goto immediatedial;
- }
- }
- }
- }
- break;
- case IAX_COMMAND_DIAL:
- immediatedial:
- if (ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_TBD)) {
- ast_clear_flag(&iaxs[fr->callno]->state, IAX_STATE_TBD);
- ast_string_field_set(iaxs[fr->callno], exten, ies.called_number ? ies.called_number : "s");
- if (!ast_exists_extension(NULL, iaxs[fr->callno]->context, iaxs[fr->callno]->exten, 1, iaxs[fr->callno]->cid_num)) {
- if (authdebug)
- ast_log(LOG_NOTICE, "Rejected dial attempt from %s, request '%s@%s' does not exist\n", ast_inet_ntoa(sin.sin_addr), iaxs[fr->callno]->exten, iaxs[fr->callno]->context);
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "No such context/extension");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_NO_ROUTE_DESTINATION);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- if (!iaxs[fr->callno]) {
- break;
- }
- } else {
- char tmp[256];
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED);
- ast_verb(3, "Accepting DIAL from %s, formats = %s\n",
- ast_inet_ntoa(sin.sin_addr),
- iax2_getformatname_multiple(tmp, sizeof(tmp), iaxs[fr->callno]->peerformat));
- ast_set_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED);
- send_command(iaxs[fr->callno], AST_FRAME_CONTROL, AST_CONTROL_PROGRESS, 0, NULL, 0, -1);
- if (!(c = ast_iax2_new(fr->callno, AST_STATE_RING, iaxs[fr->callno]->peerformat, NULL, 1)))
- iax2_destroy(fr->callno);
- else if (ies.vars) {
- struct ast_datastore *variablestore;
- struct ast_variable *var, *prev = NULL;
- AST_LIST_HEAD(, ast_var_t) *varlist;
- varlist = ast_calloc(1, sizeof(*varlist));
- variablestore = ast_datastore_alloc(&iax2_variable_datastore_info, NULL);
- ast_debug(1, "I can haz IAX vars? w00t\n");
- if (variablestore && varlist) {
- variablestore->data = varlist;
- variablestore->inheritance = DATASTORE_INHERIT_FOREVER;
- AST_LIST_HEAD_INIT(varlist);
- for (var = ies.vars; var; var = var->next) {
- struct ast_var_t *newvar = ast_var_assign(var->name, var->value);
- if (prev)
- ast_free(prev);
- prev = var;
- if (!newvar) {
- /* Don't abort list traversal, as this would leave ies.vars in an inconsistent state. */
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- } else {
- AST_LIST_INSERT_TAIL(varlist, newvar, entries);
- }
- }
- if (prev)
- ast_free(prev);
- ies.vars = NULL;
- ast_channel_datastore_add(c, variablestore);
- } else {
- ast_log(LOG_ERROR, "Memory allocation error while processing IAX2 variables\n");
- if (variablestore)
- ast_datastore_free(variablestore);
- if (varlist)
- ast_free(varlist);
- }
- }
- }
- }
- break;
- case IAX_COMMAND_INVAL:
- iaxs[fr->callno]->error = ENOTCONN;
- ast_debug(1, "Immediately destroying %d, having received INVAL\n", fr->callno);
- iax2_destroy(fr->callno);
- ast_debug(1, "Destroying call %d\n", fr->callno);
- break;
- case IAX_COMMAND_VNAK:
- ast_debug(1, "Received VNAK: resending outstanding frames\n");
- /* Force retransmission */
- vnak_retransmit(fr->callno, fr->iseqno);
- break;
- case IAX_COMMAND_REGREQ:
- case IAX_COMMAND_REGREL:
- /* For security, always ack immediately */
- if (delayreject)
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- if (register_verify(fr->callno, &sin, &ies)) {
- if (!iaxs[fr->callno]) {
- break;
- }
- /* Send delayed failure */
- auth_fail(fr->callno, IAX_COMMAND_REGREJ);
- break;
- }
- if (!iaxs[fr->callno]) {
- break;
- }
- if ((ast_strlen_zero(iaxs[fr->callno]->secret) && ast_strlen_zero(iaxs[fr->callno]->inkeys)) ||
- ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_AUTHENTICATED)) {
- if (f.subclass.integer == IAX_COMMAND_REGREL) {
- memset(&sin, 0, sizeof(sin));
- sin.sin_family = AF_INET;
- }
- if (update_registry(&sin, fr->callno, ies.devicetype, fd, ies.refresh)) {
- ast_log(LOG_WARNING, "Registry error\n");
- }
- if (!iaxs[fr->callno]) {
- break;
- }
- if (ies.provverpres && ies.serviceident && sin.sin_addr.s_addr) {
- ast_mutex_unlock(&iaxsl[fr->callno]);
- check_provisioning(&sin, fd, ies.serviceident, ies.provver);
- ast_mutex_lock(&iaxsl[fr->callno]);
- }
- break;
- }
- registry_authrequest(fr->callno);
- break;
- case IAX_COMMAND_REGACK:
- if (iax2_ack_registry(&ies, &sin, fr->callno))
- ast_log(LOG_WARNING, "Registration failure\n");
- /* Send ack immediately, before we destroy */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- iax2_destroy(fr->callno);
- break;
- case IAX_COMMAND_REGREJ:
- if (iaxs[fr->callno]->reg) {
- if (authdebug) {
- ast_log(LOG_NOTICE, "Registration of '%s' rejected: '%s' from: '%s'\n", iaxs[fr->callno]->reg->username, ies.cause ? ies.cause : "<unknown>", ast_inet_ntoa(sin.sin_addr));
- manager_event(EVENT_FLAG_SYSTEM, "Registry", "ChannelType: IAX2\r\nUsername: %s\r\nStatus: Rejected\r\nCause: %s\r\n", iaxs[fr->callno]->reg->username, ies.cause ? ies.cause : "<unknown>");
- }
- iaxs[fr->callno]->reg->regstate = REG_STATE_REJECTED;
- }
- /* Send ack immediately, before we destroy */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- iax2_destroy(fr->callno);
- break;
- case IAX_COMMAND_REGAUTH:
- /* Authentication request */
- if (registry_rerequest(&ies, fr->callno, &sin)) {
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_str(&ied0, IAX_IE_CAUSE, "No authority found");
- iax_ie_append_byte(&ied0, IAX_IE_CAUSECODE, AST_CAUSE_FACILITY_NOT_SUBSCRIBED);
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- }
- break;
- case IAX_COMMAND_TXREJ:
- while (iaxs[fr->callno]
- && iaxs[fr->callno]->bridgecallno
- && ast_mutex_trylock(&iaxsl[iaxs[fr->callno]->bridgecallno])) {
- DEADLOCK_AVOIDANCE(&iaxsl[fr->callno]);
- }
- if (!iaxs[fr->callno]) {
- break;
- }
- iaxs[fr->callno]->transferring = TRANSFER_NONE;
- ast_verb(3, "Channel '%s' unable to transfer\n", iaxs[fr->callno]->owner ? ast_channel_name(iaxs[fr->callno]->owner) : "<Unknown>");
- memset(&iaxs[fr->callno]->transfer, 0, sizeof(iaxs[fr->callno]->transfer));
- if (!iaxs[fr->callno]->bridgecallno) {
- break;
- }
- if (iaxs[iaxs[fr->callno]->bridgecallno]
- && iaxs[iaxs[fr->callno]->bridgecallno]->transferring) {
- iaxs[iaxs[fr->callno]->bridgecallno]->transferring = TRANSFER_NONE;
- send_command(iaxs[iaxs[fr->callno]->bridgecallno], AST_FRAME_IAX, IAX_COMMAND_TXREJ, 0, NULL, 0, -1);
- }
- ast_mutex_unlock(&iaxsl[iaxs[fr->callno]->bridgecallno]);
- break;
- case IAX_COMMAND_TXREADY:
- while (iaxs[fr->callno]
- && iaxs[fr->callno]->bridgecallno
- && ast_mutex_trylock(&iaxsl[iaxs[fr->callno]->bridgecallno])) {
- DEADLOCK_AVOIDANCE(&iaxsl[fr->callno]);
- }
- if (!iaxs[fr->callno]) {
- break;
- }
- if (iaxs[fr->callno]->transferring == TRANSFER_BEGIN) {
- iaxs[fr->callno]->transferring = TRANSFER_READY;
- } else if (iaxs[fr->callno]->transferring == TRANSFER_MBEGIN) {
- iaxs[fr->callno]->transferring = TRANSFER_MREADY;
- } else {
- if (iaxs[fr->callno]->bridgecallno) {
- ast_mutex_unlock(&iaxsl[iaxs[fr->callno]->bridgecallno]);
- }
- break;
- }
- ast_verb(3, "Channel '%s' ready to transfer\n", iaxs[fr->callno]->owner ? ast_channel_name(iaxs[fr->callno]->owner) : "<Unknown>");
- if (!iaxs[fr->callno]->bridgecallno) {
- break;
- }
- if (!iaxs[iaxs[fr->callno]->bridgecallno]
- || (iaxs[iaxs[fr->callno]->bridgecallno]->transferring != TRANSFER_READY
- && iaxs[iaxs[fr->callno]->bridgecallno]->transferring != TRANSFER_MREADY)) {
- ast_mutex_unlock(&iaxsl[iaxs[fr->callno]->bridgecallno]);
- break;
- }
- /* Both sides are ready */
- /* XXX what isn't checked here is that both sides match transfer types. */
- if (iaxs[fr->callno]->transferring == TRANSFER_MREADY) {
- ast_verb(3, "Attempting media bridge of %s and %s\n", iaxs[fr->callno]->owner ? ast_channel_name(iaxs[fr->callno]->owner) : "<Unknown>",
- iaxs[iaxs[fr->callno]->bridgecallno]->owner ? ast_channel_name(iaxs[iaxs[fr->callno]->bridgecallno]->owner) : "<Unknown>");
- iaxs[iaxs[fr->callno]->bridgecallno]->transferring = TRANSFER_MEDIA;
- iaxs[fr->callno]->transferring = TRANSFER_MEDIA;
- memset(&ied0, 0, sizeof(ied0));
- memset(&ied1, 0, sizeof(ied1));
- iax_ie_append_short(&ied0, IAX_IE_CALLNO, iaxs[iaxs[fr->callno]->bridgecallno]->peercallno);
- iax_ie_append_short(&ied1, IAX_IE_CALLNO, iaxs[fr->callno]->peercallno);
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_TXMEDIA, 0, ied0.buf, ied0.pos, -1);
- send_command(iaxs[iaxs[fr->callno]->bridgecallno], AST_FRAME_IAX, IAX_COMMAND_TXMEDIA, 0, ied1.buf, ied1.pos, -1);
- } else {
- ast_verb(3, "Releasing %s and %s\n", iaxs[fr->callno]->owner ? ast_channel_name(iaxs[fr->callno]->owner) : "<Unknown>",
- iaxs[iaxs[fr->callno]->bridgecallno]->owner ? ast_channel_name(iaxs[iaxs[fr->callno]->bridgecallno]->owner) : "<Unknown>");
- iaxs[iaxs[fr->callno]->bridgecallno]->transferring = TRANSFER_RELEASED;
- iaxs[fr->callno]->transferring = TRANSFER_RELEASED;
- ast_set_flag64(iaxs[iaxs[fr->callno]->bridgecallno], IAX_ALREADYGONE);
- ast_set_flag64(iaxs[fr->callno], IAX_ALREADYGONE);
- /* Stop doing lag & ping requests */
- stop_stuff(fr->callno);
- stop_stuff(iaxs[fr->callno]->bridgecallno);
- memset(&ied0, 0, sizeof(ied0));
- memset(&ied1, 0, sizeof(ied1));
- iax_ie_append_short(&ied0, IAX_IE_CALLNO, iaxs[iaxs[fr->callno]->bridgecallno]->peercallno);
- iax_ie_append_short(&ied1, IAX_IE_CALLNO, iaxs[fr->callno]->peercallno);
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_TXREL, 0, ied0.buf, ied0.pos, -1);
- send_command(iaxs[iaxs[fr->callno]->bridgecallno], AST_FRAME_IAX, IAX_COMMAND_TXREL, 0, ied1.buf, ied1.pos, -1);
- }
- ast_mutex_unlock(&iaxsl[iaxs[fr->callno]->bridgecallno]);
- break;
- case IAX_COMMAND_TXREQ:
- try_transfer(iaxs[fr->callno], &ies);
- break;
- case IAX_COMMAND_TXCNT:
- if (iaxs[fr->callno]->transferring)
- send_command_transfer(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_TXACC, 0, NULL, 0);
- break;
- case IAX_COMMAND_TXREL:
- /* Send ack immediately, rather than waiting until we've changed addresses */
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- complete_transfer(fr->callno, &ies);
- stop_stuff(fr->callno); /* for attended transfer to work with libiax */
- break;
- case IAX_COMMAND_TXMEDIA:
- if (iaxs[fr->callno]->transferring == TRANSFER_READY) {
- AST_LIST_TRAVERSE(&frame_queue[fr->callno], cur, list) {
- /* Cancel any outstanding frames and start anew */
- if (cur->transfer) {
- cur->retries = -1;
- }
- }
- /* Start sending our media to the transfer address, but otherwise leave the call as-is */
- iaxs[fr->callno]->transferring = TRANSFER_MEDIAPASS;
- }
- break;
- case IAX_COMMAND_RTKEY:
- if (!IAX_CALLENCRYPTED(iaxs[fr->callno])) {
- ast_log(LOG_WARNING,
- "we've been told to rotate our encryption key, "
- "but this isn't an encrypted call. bad things will happen.\n"
- );
- break;
- }
- IAX_DEBUGDIGEST("Receiving", ies.challenge);
- ast_aes_set_decrypt_key((unsigned char *) ies.challenge, &iaxs[fr->callno]->dcx);
- break;
- case IAX_COMMAND_DPREP:
- complete_dpreply(iaxs[fr->callno], &ies);
- break;
- case IAX_COMMAND_UNSUPPORT:
- ast_log(LOG_NOTICE, "Peer did not understand our iax command '%d'\n", ies.iax_unknown);
- break;
- case IAX_COMMAND_FWDOWNL:
- /* Firmware download */
- if (!ast_test_flag64(&globalflags, IAX_ALLOWFWDOWNLOAD)) {
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_UNSUPPORT, 0, NULL, 0, -1);
- break;
- }
- memset(&ied0, 0, sizeof(ied0));
- res = iax_firmware_append(&ied0, (unsigned char *)ies.devicetype, ies.fwdesc);
- if (res < 0)
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_REJECT, 0, ied0.buf, ied0.pos, -1);
- else if (res > 0)
- send_command_final(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_FWDATA, 0, ied0.buf, ied0.pos, -1);
- else
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_FWDATA, 0, ied0.buf, ied0.pos, -1);
- break;
- case IAX_COMMAND_CALLTOKEN:
- {
- struct iax_frame *cur;
- /* find last sent frame */
- if ((cur = AST_LIST_LAST(&frame_queue[fr->callno])) && ies.calltoken && ies.calltokendata) {
- resend_with_token(fr->callno, cur, (char *) ies.calltokendata);
- }
- break;
- }
- default:
- ast_debug(1, "Unknown IAX command %d on %d/%d\n", f.subclass.integer, fr->callno, iaxs[fr->callno]->peercallno);
- memset(&ied0, 0, sizeof(ied0));
- iax_ie_append_byte(&ied0, IAX_IE_IAX_UNKNOWN, f.subclass.integer);
- send_command(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_UNSUPPORT, 0, ied0.buf, ied0.pos, -1);
- }
- /* Free remote variables (if any) */
- if (ies.vars) {
- ast_variables_destroy(ies.vars);
- ast_debug(1, "I can haz IAX vars, but they is no good :-(\n");
- ies.vars = NULL;
- }
- /* Don't actually pass these frames along */
- if ((f.subclass.integer != IAX_COMMAND_ACK) &&
- (f.subclass.integer != IAX_COMMAND_TXCNT) &&
- (f.subclass.integer != IAX_COMMAND_TXACC) &&
- (f.subclass.integer != IAX_COMMAND_INVAL) &&
- (f.subclass.integer != IAX_COMMAND_VNAK)) {
- if (iaxs[fr->callno] && iaxs[fr->callno]->aseqno != iaxs[fr->callno]->iseqno) {
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- }
- }
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- /* Unless this is an ACK or INVAL frame, ack it */
- if (iaxs[fr->callno] && iaxs[fr->callno]->aseqno != iaxs[fr->callno]->iseqno)
- send_command_immediate(iaxs[fr->callno], AST_FRAME_IAX, IAX_COMMAND_ACK, fr->ts, NULL, 0,fr->iseqno);
- } else if (minivid) {
- f.frametype = AST_FRAME_VIDEO;
- if (iaxs[fr->callno]->videoformat > 0) {
- if (ntohs(vh->ts) & 0x8000LL) {
- ast_format_set_video_mark(&f.subclass.format);
- }
- ast_format_from_old_bitfield(&f.subclass.format, iaxs[fr->callno]->videoformat);
- } else {
- ast_log(LOG_WARNING, "Received mini frame before first full video frame\n");
- iax2_vnak(fr->callno);
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- f.datalen = res - sizeof(*vh);
- if (f.datalen)
- f.data.ptr = thread->buf + sizeof(*vh);
- else
- f.data.ptr = NULL;
- #ifdef IAXTESTS
- if (test_resync) {
- fr->ts = (iaxs[fr->callno]->last & 0xFFFF8000L) | ((ntohs(vh->ts) + test_resync) & 0x7fff);
- } else
- #endif /* IAXTESTS */
- fr->ts = (iaxs[fr->callno]->last & 0xFFFF8000L) | (ntohs(vh->ts) & 0x7fff);
- } else {
- /* A mini frame */
- f.frametype = AST_FRAME_VOICE;
- if (iaxs[fr->callno]->voiceformat > 0)
- ast_format_from_old_bitfield(&f.subclass.format, iaxs[fr->callno]->voiceformat);
- else {
- ast_debug(1, "Received mini frame before first full voice frame\n");
- iax2_vnak(fr->callno);
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- f.datalen = res - sizeof(struct ast_iax2_mini_hdr);
- if (f.datalen < 0) {
- ast_log(LOG_WARNING, "Datalen < 0?\n");
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- if (f.datalen)
- f.data.ptr = thread->buf + sizeof(*mh);
- else
- f.data.ptr = NULL;
- #ifdef IAXTESTS
- if (test_resync) {
- fr->ts = (iaxs[fr->callno]->last & 0xFFFF0000L) | ((ntohs(mh->ts) + test_resync) & 0xffff);
- } else
- #endif /* IAXTESTS */
- fr->ts = (iaxs[fr->callno]->last & 0xFFFF0000L) | ntohs(mh->ts);
- /* FIXME? Surely right here would be the right place to undo timestamp wraparound? */
- }
- /* Don't pass any packets until we're started */
- if (!iaxs[fr->callno]
- || !ast_test_flag(&iaxs[fr->callno]->state, IAX_STATE_STARTED)) {
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- if (f.frametype == AST_FRAME_CONTROL) {
- if (!iax2_is_control_frame_allowed(f.subclass.integer)) {
- /* Control frame not allowed to come from the wire. */
- ast_debug(2, "Callno %d: Blocked receiving control frame %d.\n",
- fr->callno, f.subclass.integer);
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- if (f.subclass.integer == AST_CONTROL_CONNECTED_LINE
- || f.subclass.integer == AST_CONTROL_REDIRECTING) {
- if (iaxs[fr->callno]
- && !ast_test_flag64(iaxs[fr->callno], IAX_RECVCONNECTEDLINE)) {
- /* We are not configured to allow receiving these updates. */
- ast_debug(2, "Callno %d: Config blocked receiving control frame %d.\n",
- fr->callno, f.subclass.integer);
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- }
- iax2_lock_owner(fr->callno);
- if (iaxs[fr->callno] && iaxs[fr->callno]->owner) {
- if (f.subclass.integer == AST_CONTROL_BUSY) {
- ast_channel_hangupcause_set(iaxs[fr->callno]->owner, AST_CAUSE_BUSY);
- } else if (f.subclass.integer == AST_CONTROL_CONGESTION) {
- ast_channel_hangupcause_set(iaxs[fr->callno]->owner, AST_CAUSE_CONGESTION);
- }
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- }
- if (f.frametype == AST_FRAME_CONTROL
- && f.subclass.integer == AST_CONTROL_CONNECTED_LINE
- && iaxs[fr->callno]) {
- struct ast_party_connected_line connected;
- /*
- * Process a received connected line update.
- *
- * Initialize defaults.
- */
- ast_party_connected_line_init(&connected);
- connected.id.number.presentation = iaxs[fr->callno]->calling_pres;
- connected.id.name.presentation = iaxs[fr->callno]->calling_pres;
- if (!ast_connected_line_parse_data(f.data.ptr, f.datalen, &connected)) {
- ast_string_field_set(iaxs[fr->callno], cid_num, connected.id.number.str);
- ast_string_field_set(iaxs[fr->callno], cid_name, connected.id.name.str);
- iaxs[fr->callno]->calling_pres = ast_party_id_presentation(&connected.id);
- iax2_lock_owner(fr->callno);
- if (iaxs[fr->callno] && iaxs[fr->callno]->owner) {
- ast_set_callerid(iaxs[fr->callno]->owner,
- S_COR(connected.id.number.valid, connected.id.number.str, ""),
- S_COR(connected.id.name.valid, connected.id.name.str, ""),
- NULL);
- ast_channel_caller(iaxs[fr->callno]->owner)->id.number.presentation = connected.id.number.presentation;
- ast_channel_caller(iaxs[fr->callno]->owner)->id.name.presentation = connected.id.name.presentation;
- ast_channel_unlock(iaxs[fr->callno]->owner);
- }
- }
- ast_party_connected_line_free(&connected);
- }
- /* Common things */
- f.src = "IAX2";
- f.mallocd = 0;
- f.offset = 0;
- f.len = 0;
- if (f.datalen && (f.frametype == AST_FRAME_VOICE)) {
- f.samples = ast_codec_get_samples(&f);
- /* We need to byteswap incoming slinear samples from network byte order */
- if (f.subclass.format.id == AST_FORMAT_SLINEAR)
- ast_frame_byteswap_be(&f);
- } else
- f.samples = 0;
- iax_frame_wrap(fr, &f);
- /* If this is our most recent packet, use it as our basis for timestamping */
- if (iaxs[fr->callno] && iaxs[fr->callno]->last < fr->ts) {
- /*iaxs[fr->callno]->last = fr->ts; (do it afterwards cos schedule/forward_delivery needs the last ts too)*/
- fr->outoforder = 0;
- } else {
- if (iaxdebug && iaxs[fr->callno]) {
- ast_debug(1, "Received out of order packet... (type=%u, subclass %d, ts = %u, last = %u)\n", f.frametype, f.subclass.integer, fr->ts, iaxs[fr->callno]->last);
- }
- fr->outoforder = 1;
- }
- fr->cacheable = ((f.frametype == AST_FRAME_VOICE) || (f.frametype == AST_FRAME_VIDEO));
- if (iaxs[fr->callno]) {
- duped_fr = iaxfrdup2(fr);
- if (duped_fr) {
- schedule_delivery(duped_fr, updatehistory, 0, &fr->ts);
- }
- }
- if (iaxs[fr->callno] && iaxs[fr->callno]->last < fr->ts) {
- iaxs[fr->callno]->last = fr->ts;
- #if 1
- if (iaxdebug)
- ast_debug(1, "For call=%d, set last=%u\n", fr->callno, fr->ts);
- #endif
- }
- /* Always run again */
- ast_variables_destroy(ies.vars);
- ast_mutex_unlock(&iaxsl[fr->callno]);
- return 1;
- }
- static int socket_process(struct iax2_thread *thread)
- {
- struct ast_callid *callid;
- int res = socket_process_helper(thread);
- if ((callid = ast_read_threadstorage_callid())) {
- ast_callid_threadassoc_remove();
- callid = ast_callid_unref(callid);
- }
- return res;
- }
- /* Function to clean up process thread if it is cancelled */
- static void iax2_process_thread_cleanup(void *data)
- {
- struct iax2_thread *thread = data;
- ast_mutex_destroy(&thread->lock);
- ast_cond_destroy(&thread->cond);
- ast_mutex_destroy(&thread->init_lock);
- ast_cond_destroy(&thread->init_cond);
- ast_free(thread);
- /* Ignore check_return warning from Coverity for ast_atomic_dec_and_test below */
- ast_atomic_dec_and_test(&iaxactivethreadcount);
- }
- static void *iax2_process_thread(void *data)
- {
- struct iax2_thread *thread = data;
- struct timeval wait;
- struct timespec ts;
- int put_into_idle = 0;
- int first_time = 1;
- int old_state;
- ast_atomic_fetchadd_int(&iaxactivethreadcount, 1);
- pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &old_state);
- pthread_cleanup_push(iax2_process_thread_cleanup, data);
- for (;;) {
- /* Wait for something to signal us to be awake */
- ast_mutex_lock(&thread->lock);
- if (thread->stop) {
- ast_mutex_unlock(&thread->lock);
- break;
- }
- /* Flag that we're ready to accept signals */
- if (first_time) {
- signal_condition(&thread->init_lock, &thread->init_cond);
- first_time = 0;
- }
- /* Put into idle list if applicable */
- if (put_into_idle) {
- insert_idle_thread(thread);
- }
- if (thread->type == IAX_THREAD_TYPE_DYNAMIC) {
- struct iax2_thread *t = NULL;
- /* Wait to be signalled or time out */
- wait = ast_tvadd(ast_tvnow(), ast_samp2tv(30000, 1000));
- ts.tv_sec = wait.tv_sec;
- ts.tv_nsec = wait.tv_usec * 1000;
- if (ast_cond_timedwait(&thread->cond, &thread->lock, &ts) == ETIMEDOUT) {
- /* This thread was never put back into the available dynamic
- * thread list, so just go away. */
- if (!put_into_idle || thread->stop) {
- ast_mutex_unlock(&thread->lock);
- break;
- }
- AST_LIST_LOCK(&dynamic_list);
- /* Account for the case where this thread is acquired *right* after a timeout */
- if ((t = AST_LIST_REMOVE(&dynamic_list, thread, list)))
- ast_atomic_fetchadd_int(&iaxdynamicthreadcount, -1);
- AST_LIST_UNLOCK(&dynamic_list);
- if (t) {
- /* This dynamic thread timed out waiting for a task and was
- * not acquired immediately after the timeout,
- * so it's time to go away. */
- ast_mutex_unlock(&thread->lock);
- break;
- }
- /* Someone grabbed our thread *right* after we timed out.
- * Wait for them to set us up with something to do and signal
- * us to continue. */
- wait = ast_tvadd(ast_tvnow(), ast_samp2tv(30000, 1000));
- ts.tv_sec = wait.tv_sec;
- ts.tv_nsec = wait.tv_usec * 1000;
- if (ast_cond_timedwait(&thread->cond, &thread->lock, &ts) == ETIMEDOUT) {
- ast_mutex_unlock(&thread->lock);
- break;
- }
- }
- } else {
- ast_cond_wait(&thread->cond, &thread->lock);
- }
- /* Go back into our respective list */
- put_into_idle = 1;
- ast_mutex_unlock(&thread->lock);
- if (thread->stop) {
- break;
- }
- /* See what we need to do */
- switch (thread->iostate) {
- case IAX_IOSTATE_IDLE:
- continue;
- case IAX_IOSTATE_READY:
- thread->actions++;
- thread->iostate = IAX_IOSTATE_PROCESSING;
- socket_process(thread);
- handle_deferred_full_frames(thread);
- break;
- case IAX_IOSTATE_SCHEDREADY:
- thread->actions++;
- thread->iostate = IAX_IOSTATE_PROCESSING;
- #ifdef SCHED_MULTITHREADED
- thread->schedfunc(thread->scheddata);
- #endif
- break;
- default:
- break;
- }
- /* The network thread added us to the active_thread list when we were given
- * frames to process, Now that we are done, we must remove ourselves from
- * the active list, and return to the idle list */
- AST_LIST_LOCK(&active_list);
- AST_LIST_REMOVE(&active_list, thread, list);
- AST_LIST_UNLOCK(&active_list);
- /* Make sure another frame didn't sneak in there after we thought we were done. */
- handle_deferred_full_frames(thread);
- time(&thread->checktime);
- thread->iostate = IAX_IOSTATE_IDLE;
- #ifdef DEBUG_SCHED_MULTITHREAD
- thread->curfunc[0]='\0';
- #endif
- }
- /*!
- * \note For some reason, idle threads are exiting without being
- * removed from an idle list, which is causing memory
- * corruption. Forcibly remove it from the list, if it's there.
- */
- AST_LIST_LOCK(&idle_list);
- AST_LIST_REMOVE(&idle_list, thread, list);
- AST_LIST_UNLOCK(&idle_list);
- AST_LIST_LOCK(&dynamic_list);
- AST_LIST_REMOVE(&dynamic_list, thread, list);
- AST_LIST_UNLOCK(&dynamic_list);
- if (!thread->stop) {
- /* Nobody asked me to stop so nobody is waiting to join me. */
- pthread_detach(pthread_self());
- }
- /* I am exiting here on my own volition, I need to clean up my own data structures
- * Assume that I am no longer in any of the lists (idle, active, or dynamic)
- */
- pthread_cleanup_pop(1);
- return NULL;
- }
- static int iax2_do_register(struct iax2_registry *reg)
- {
- struct iax_ie_data ied;
- if (iaxdebug)
- ast_debug(1, "Sending registration request for '%s'\n", reg->username);
- if (reg->dnsmgr &&
- ((reg->regstate == REG_STATE_TIMEOUT) || !ast_sockaddr_ipv4(®->addr))) {
- /* Maybe the IP has changed, force DNS refresh */
- ast_dnsmgr_refresh(reg->dnsmgr);
- }
- /*
- * if IP has Changed, free allocated call to create a new one with new IP
- * call has the pointer to IP and must be updated to the new one
- */
- if (reg->dnsmgr && ast_dnsmgr_changed(reg->dnsmgr) && (reg->callno > 0)) {
- int callno = reg->callno;
- ast_mutex_lock(&iaxsl[callno]);
- iax2_destroy(callno);
- ast_mutex_unlock(&iaxsl[callno]);
- reg->callno = 0;
- }
- if (!ast_sockaddr_ipv4(®->addr)) {
- if (iaxdebug)
- ast_debug(1, "Unable to send registration request for '%s' without IP address\n", reg->username);
- /* Setup the next registration attempt */
- reg->expire = iax2_sched_replace(reg->expire, sched,
- (5 * reg->refresh / 6) * 1000, iax2_do_register_s, reg);
- return -1;
- }
- if (!ast_sockaddr_port(®->addr) && reg->port) {
- ast_sockaddr_set_port(®->addr, reg->port);
- }
- if (!reg->callno) {
- struct sockaddr_in reg_addr;
- ast_debug(3, "Allocate call number\n");
- ast_sockaddr_to_sin(®->addr, ®_addr);
- reg->callno = find_callno_locked(0, 0, ®_addr, NEW_FORCE, defaultsockfd, 0);
- if (reg->callno < 1) {
- ast_log(LOG_WARNING, "Unable to create call for registration\n");
- return -1;
- } else
- ast_debug(3, "Registration created on call %d\n", reg->callno);
- iaxs[reg->callno]->reg = reg;
- ast_mutex_unlock(&iaxsl[reg->callno]);
- }
- /* Setup the next registration a little early */
- reg->expire = iax2_sched_replace(reg->expire, sched,
- (5 * reg->refresh / 6) * 1000, iax2_do_register_s, reg);
- /* Send the request */
- memset(&ied, 0, sizeof(ied));
- iax_ie_append_str(&ied, IAX_IE_USERNAME, reg->username);
- iax_ie_append_short(&ied, IAX_IE_REFRESH, reg->refresh);
- add_empty_calltoken_ie(iaxs[reg->callno], &ied); /* this _MUST_ be the last ie added */
- send_command(iaxs[reg->callno],AST_FRAME_IAX, IAX_COMMAND_REGREQ, 0, ied.buf, ied.pos, -1);
- reg->regstate = REG_STATE_REGSENT;
- return 0;
- }
- static int iax2_provision(struct sockaddr_in *end, int sockfd, const char *dest, const char *template, int force)
- {
- /* Returns 1 if provisioned, -1 if not able to find destination, or 0 if no provisioning
- is found for template */
- struct iax_ie_data provdata;
- struct iax_ie_data ied;
- unsigned int sig;
- struct sockaddr_in sin;
- int callno;
- struct create_addr_info cai;
- memset(&cai, 0, sizeof(cai));
- ast_debug(1, "Provisioning '%s' from template '%s'\n", dest, template);
- if (iax_provision_build(&provdata, &sig, template, force)) {
- ast_debug(1, "No provisioning found for template '%s'\n", template);
- return 0;
- }
- if (end) {
- memcpy(&sin, end, sizeof(sin));
- cai.sockfd = sockfd;
- } else if (create_addr(dest, NULL, &sin, &cai))
- return -1;
- /* Build the rest of the message */
- memset(&ied, 0, sizeof(ied));
- iax_ie_append_raw(&ied, IAX_IE_PROVISIONING, provdata.buf, provdata.pos);
- callno = find_callno_locked(0, 0, &sin, NEW_FORCE, cai.sockfd, 0);
- if (!callno)
- return -1;
- if (iaxs[callno]) {
- /* Schedule autodestruct in case they don't ever give us anything back */
- iaxs[callno]->autoid = iax2_sched_replace(iaxs[callno]->autoid,
- sched, 15000, auto_hangup, (void *)(long)callno);
- ast_set_flag64(iaxs[callno], IAX_PROVISION);
- /* Got a call number now, so go ahead and send the provisioning information */
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_PROVISION, 0, ied.buf, ied.pos, -1);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- return 1;
- }
- static char *papp = "IAX2Provision";
- /*! iax2provision
- \ingroup applications
- */
- static int iax2_prov_app(struct ast_channel *chan, const char *data)
- {
- int res;
- char *sdata;
- char *opts;
- int force =0;
- unsigned short callno = PTR_TO_CALLNO(ast_channel_tech_pvt(chan));
- if (ast_strlen_zero(data))
- data = "default";
- sdata = ast_strdupa(data);
- opts = strchr(sdata, '|');
- if (opts)
- *opts='\0';
- if (ast_channel_tech(chan) != &iax2_tech) {
- ast_log(LOG_NOTICE, "Can't provision a non-IAX device!\n");
- return -1;
- }
- if (!callno || !iaxs[callno] || !iaxs[callno]->addr.sin_addr.s_addr) {
- ast_log(LOG_NOTICE, "Can't provision something with no IP?\n");
- return -1;
- }
- res = iax2_provision(&iaxs[callno]->addr, iaxs[callno]->sockfd, NULL, sdata, force);
- ast_verb(3, "Provisioned IAXY at '%s' with '%s'= %d\n",
- ast_inet_ntoa(iaxs[callno]->addr.sin_addr),
- sdata, res);
- return res;
- }
- static char *handle_cli_iax2_provision(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- int force = 0;
- int res;
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 provision";
- e->usage =
- "Usage: iax2 provision <host> <template> [forced]\n"
- " Provisions the given peer or IP address using a template\n"
- " matching either 'template' or '*' if the template is not\n"
- " found. If 'forced' is specified, even empty provisioning\n"
- " fields will be provisioned as empty fields.\n";
- return NULL;
- case CLI_GENERATE:
- if (a->pos == 3)
- return iax_prov_complete_template(a->line, a->word, a->pos, a->n);
- return NULL;
- }
- if (a->argc < 4)
- return CLI_SHOWUSAGE;
- if (a->argc > 4) {
- if (!strcasecmp(a->argv[4], "forced"))
- force = 1;
- else
- return CLI_SHOWUSAGE;
- }
- res = iax2_provision(NULL, -1, a->argv[2], a->argv[3], force);
- if (res < 0)
- ast_cli(a->fd, "Unable to find peer/address '%s'\n", a->argv[2]);
- else if (res < 1)
- ast_cli(a->fd, "No template (including wildcard) matching '%s'\n", a->argv[3]);
- else
- ast_cli(a->fd, "Provisioning '%s' with template '%s'%s\n", a->argv[2], a->argv[3], force ? ", forced" : "");
- return CLI_SUCCESS;
- }
- static void __iax2_poke_noanswer(const void *data)
- {
- struct iax2_peer *peer = (struct iax2_peer *)data;
- int callno;
- if (peer->lastms > -1) {
- ast_log(LOG_NOTICE, "Peer '%s' is now UNREACHABLE! Time: %d\n", peer->name, peer->lastms);
- manager_event(EVENT_FLAG_SYSTEM, "PeerStatus", "ChannelType: IAX2\r\nPeer: IAX2/%s\r\nPeerStatus: Unreachable\r\nTime: %d\r\n", peer->name, peer->lastms);
- ast_devstate_changed(AST_DEVICE_UNAVAILABLE, AST_DEVSTATE_CACHABLE, "IAX2/%s", peer->name); /* Activate notification */
- }
- if ((callno = peer->callno) > 0) {
- ast_mutex_lock(&iaxsl[callno]);
- iax2_destroy(callno);
- ast_mutex_unlock(&iaxsl[callno]);
- }
- peer->callno = 0;
- peer->lastms = -1;
- /* Try again quickly */
- peer->pokeexpire = iax2_sched_add(sched, peer->pokefreqnotok, iax2_poke_peer_s, peer_ref(peer));
- if (peer->pokeexpire == -1)
- peer_unref(peer);
- }
- static int iax2_poke_noanswer(const void *data)
- {
- struct iax2_peer *peer = (struct iax2_peer *)data;
- peer->pokeexpire = -1;
- #ifdef SCHED_MULTITHREADED
- if (schedule_action(__iax2_poke_noanswer, data))
- #endif
- __iax2_poke_noanswer(data);
- peer_unref(peer);
- return 0;
- }
- static int iax2_poke_peer_cb(void *obj, void *arg, int flags)
- {
- struct iax2_peer *peer = obj;
- iax2_poke_peer(peer, 0);
- return 0;
- }
- static int iax2_poke_peer(struct iax2_peer *peer, int heldcall)
- {
- int callno;
- int poke_timeout;
- struct sockaddr_in peer_addr;
- if (!peer->maxms || (!ast_sockaddr_ipv4(&peer->addr) && !peer->dnsmgr)) {
- /* IF we have no IP without dnsmgr, or this isn't to be monitored, return
- immediately after clearing things out */
- peer->lastms = 0;
- peer->historicms = 0;
- peer->pokeexpire = -1;
- peer->callno = 0;
- return 0;
- }
- ast_sockaddr_to_sin(&peer->addr, &peer_addr);
- /* The peer could change the callno inside iax2_destroy, since we do deadlock avoidance */
- if ((callno = peer->callno) > 0) {
- ast_log(LOG_NOTICE, "Still have a callno...\n");
- ast_mutex_lock(&iaxsl[callno]);
- iax2_destroy(callno);
- ast_mutex_unlock(&iaxsl[callno]);
- }
- if (heldcall)
- ast_mutex_unlock(&iaxsl[heldcall]);
- callno = peer->callno = find_callno(0, 0, &peer_addr, NEW_FORCE, peer->sockfd, 0);
- if (heldcall)
- ast_mutex_lock(&iaxsl[heldcall]);
- if (callno < 1) {
- ast_log(LOG_WARNING, "Unable to allocate call for poking peer '%s'\n", peer->name);
- return -1;
- }
- if (peer->pokeexpire > -1) {
- if (!AST_SCHED_DEL(sched, peer->pokeexpire)) {
- peer->pokeexpire = -1;
- peer_unref(peer);
- }
- }
- if (peer->lastms < 0){
- /* If the host is already unreachable then use time less than the unreachable
- * interval. 5/6 is arbitrary multiplier to get value less than
- * peer->pokefreqnotok. Value less than peer->pokefreqnotok is used to expire
- * current POKE before starting new POKE (which is scheduled after
- * peer->pokefreqnotok). */
- poke_timeout = peer->pokefreqnotok * 5 / 6;
- } else {
- /* If the host is reachable, use timeout large enough to allow for multiple
- * POKE retries. Limit this value to less than peer->pokefreqok. 5/6 is arbitrary
- * multiplier to get value less than peer->pokefreqok. Value less than
- * peer->pokefreqok is used to expire current POKE before starting new POKE
- * (which is scheduled after peer->pokefreqok). */
- poke_timeout = MIN(MAX_RETRY_TIME * 2 + peer->maxms, peer->pokefreqok * 5 / 6);
- }
- /* Queue up a new task to handle no reply */
- peer->pokeexpire = iax2_sched_add(sched, poke_timeout, iax2_poke_noanswer, peer_ref(peer));
- if (peer->pokeexpire == -1)
- peer_unref(peer);
- /* And send the poke */
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- struct iax_ie_data ied = {
- .buf = { 0 },
- .pos = 0,
- };
- /* Speed up retransmission times for this qualify call */
- iaxs[callno]->pingtime = peer->maxms / 8;
- iaxs[callno]->peerpoke = peer;
- add_empty_calltoken_ie(iaxs[callno], &ied); /* this _MUST_ be the last ie added */
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_POKE, 0, ied.buf, ied.pos, -1);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- return 0;
- }
- static void free_context(struct iax2_context *con)
- {
- struct iax2_context *conl;
- while(con) {
- conl = con;
- con = con->next;
- ast_free(conl);
- }
- }
- static struct ast_channel *iax2_request(const char *type, struct ast_format_cap *cap, const struct ast_channel *requestor, const char *data, int *cause)
- {
- int callno;
- int res;
- struct sockaddr_in sin;
- struct ast_channel *c;
- struct parsed_dial_string pds;
- struct create_addr_info cai;
- char *tmpstr;
- struct ast_callid *callid;
- memset(&pds, 0, sizeof(pds));
- tmpstr = ast_strdupa(data);
- parse_dial_string(tmpstr, &pds);
- callid = ast_read_threadstorage_callid();
- if (ast_strlen_zero(pds.peer)) {
- ast_log(LOG_WARNING, "No peer provided in the IAX2 dial string '%s'\n", data);
- return NULL;
- }
- memset(&cai, 0, sizeof(cai));
- cai.capability = iax2_capability;
- ast_copy_flags64(&cai, &globalflags, IAX_NOTRANSFER | IAX_TRANSFERMEDIA | IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- /* Populate our address from the given */
- if (create_addr(pds.peer, NULL, &sin, &cai)) {
- *cause = AST_CAUSE_UNREGISTERED;
- return NULL;
- }
- if (pds.port)
- sin.sin_port = htons(atoi(pds.port));
- callno = find_callno_locked(0, 0, &sin, NEW_FORCE, cai.sockfd, 0);
- if (callno < 1) {
- ast_log(LOG_WARNING, "Unable to create call\n");
- *cause = AST_CAUSE_CONGESTION;
- return NULL;
- }
- /* If this is a trunk, update it now */
- ast_copy_flags64(iaxs[callno], &cai, IAX_TRUNK | IAX_SENDANI | IAX_NOTRANSFER | IAX_TRANSFERMEDIA | IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- if (ast_test_flag64(&cai, IAX_TRUNK)) {
- int new_callno;
- if ((new_callno = make_trunk(callno, 1)) != -1)
- callno = new_callno;
- }
- iaxs[callno]->maxtime = cai.maxtime;
- if (callid) {
- iax_pvt_callid_set(callno, callid);
- }
- if (cai.found) {
- ast_string_field_set(iaxs[callno], host, pds.peer);
- }
- c = ast_iax2_new(callno, AST_STATE_DOWN, cai.capability, requestor ? ast_channel_linkedid(requestor) : NULL, cai.found);
- ast_mutex_unlock(&iaxsl[callno]);
- if (c) {
- struct ast_format_cap *joint;
- if (callid) {
- ast_channel_callid_set(c, callid);
- }
- /* Choose a format we can live with */
- if ((joint = ast_format_cap_joint(ast_channel_nativeformats(c), cap))) {
- ast_format_cap_copy(ast_channel_nativeformats(c), joint);
- joint = ast_format_cap_destroy(joint);
- } else {
- struct ast_format best_fmt_cap;
- struct ast_format best_fmt_native;
- res = ast_translator_best_choice(cap, ast_channel_nativeformats(c), &best_fmt_cap, &best_fmt_native);
- if (res < 0) {
- char tmp[256];
- char tmp2[256];
- ast_log(LOG_WARNING, "Unable to create translator path for %s to %s on %s\n",
- ast_getformatname_multiple(tmp, sizeof(tmp), ast_channel_nativeformats(c)), ast_getformatname_multiple(tmp2, sizeof(tmp2), cap), ast_channel_name(c));
- ast_hangup(c);
- return NULL;
- }
- ast_format_cap_set(ast_channel_nativeformats(c), &best_fmt_native);
- }
- ast_best_codec(ast_channel_nativeformats(c), ast_channel_readformat(c));
- ast_format_copy(ast_channel_writeformat(c), ast_channel_readformat(c));
- }
- if (callid) {
- ast_callid_unref(callid);
- }
- return c;
- }
- static void *network_thread(void *ignore)
- {
- if (timer) {
- ast_io_add(io, ast_timer_fd(timer), timing_read, AST_IO_IN | AST_IO_PRI, NULL);
- }
- for (;;) {
- pthread_testcancel();
- /* Wake up once a second just in case SIGURG was sent while
- * we weren't in poll(), to make sure we don't hang when trying
- * to unload. */
- if (ast_io_wait(io, 1000) <= 0) {
- break;
- }
- }
- return NULL;
- }
- static int start_network_thread(void)
- {
- struct iax2_thread *thread;
- int threadcount = 0;
- int x;
- for (x = 0; x < iaxthreadcount; x++) {
- thread = ast_calloc(1, sizeof(*thread));
- if (thread) {
- thread->type = IAX_THREAD_TYPE_POOL;
- thread->threadnum = ++threadcount;
- ast_mutex_init(&thread->lock);
- ast_cond_init(&thread->cond, NULL);
- ast_mutex_init(&thread->init_lock);
- ast_cond_init(&thread->init_cond, NULL);
- ast_mutex_lock(&thread->init_lock);
- if (ast_pthread_create_background(&thread->threadid, NULL, iax2_process_thread, thread)) {
- ast_log(LOG_WARNING, "Failed to create new thread!\n");
- ast_mutex_destroy(&thread->lock);
- ast_cond_destroy(&thread->cond);
- ast_mutex_unlock(&thread->init_lock);
- ast_mutex_destroy(&thread->init_lock);
- ast_cond_destroy(&thread->init_cond);
- ast_free(thread);
- thread = NULL;
- continue;
- }
- /* Wait for the thread to be ready */
- ast_cond_wait(&thread->init_cond, &thread->init_lock);
- /* Done with init_lock */
- ast_mutex_unlock(&thread->init_lock);
- AST_LIST_LOCK(&idle_list);
- AST_LIST_INSERT_TAIL(&idle_list, thread, list);
- AST_LIST_UNLOCK(&idle_list);
- }
- }
- if (ast_pthread_create_background(&netthreadid, NULL, network_thread, NULL)) {
- ast_log(LOG_ERROR, "Failed to create new thread!\n");
- return -1;
- }
- ast_verb(2, "%d helper threads started\n", threadcount);
- return 0;
- }
- static struct iax2_context *build_context(const char *context)
- {
- struct iax2_context *con;
- if ((con = ast_calloc(1, sizeof(*con))))
- ast_copy_string(con->context, context, sizeof(con->context));
-
- return con;
- }
- static int get_auth_methods(const char *value)
- {
- int methods = 0;
- if (strstr(value, "rsa"))
- methods |= IAX_AUTH_RSA;
- if (strstr(value, "md5"))
- methods |= IAX_AUTH_MD5;
- if (strstr(value, "plaintext"))
- methods |= IAX_AUTH_PLAINTEXT;
- return methods;
- }
- /*! \brief Check if address can be used as packet source.
- \return 0 address available, 1 address unavailable, -1 error
- */
- static int check_srcaddr(struct ast_sockaddr *addr)
- {
- int sd;
- sd = socket(addr->ss.ss_family, SOCK_DGRAM, 0);
- if (sd < 0) {
- ast_log(LOG_ERROR, "Socket: %s\n", strerror(errno));
- return -1;
- }
- if (ast_bind(sd, addr) < 0) {
- ast_debug(1, "Can't bind: %s\n", strerror(errno));
- close(sd);
- return 1;
- }
- close(sd);
- return 0;
- }
- /*! \brief Parse the "sourceaddress" value,
- lookup in netsock list and set peer's sockfd. Defaults to defaultsockfd if
- not found. */
- static int peer_set_srcaddr(struct iax2_peer *peer, const char *srcaddr)
- {
- struct ast_sockaddr addr;
- int nonlocal = 1;
- int port = IAX_DEFAULT_PORTNO;
- int sockfd = defaultsockfd;
- char *tmp;
- char *host;
- char *portstr;
- tmp = ast_strdupa(srcaddr);
- ast_sockaddr_split_hostport(tmp, &host, &portstr, 0);
- if (portstr) {
- port = atoi(portstr);
- if (port < 1)
- port = IAX_DEFAULT_PORTNO;
- }
- addr.ss.ss_family = AF_INET;
- if (!ast_get_ip(&addr, host)) {
- struct ast_netsock *sock;
- if (check_srcaddr(&addr) == 0) {
- /* ip address valid. */
- ast_sockaddr_set_port(&addr, port);
- if (!(sock = ast_netsock_find(netsock, &addr)))
- sock = ast_netsock_find(outsock, &addr);
- if (sock) {
- sockfd = ast_netsock_sockfd(sock);
- nonlocal = 0;
- } else {
- /* INADDR_ANY matches anyway! */
- ast_sockaddr_parse(&addr, "0.0.0.0", 0);
- ast_sockaddr_set_port(&addr, port);
- if (ast_netsock_find(netsock, &addr)) {
- sock = ast_netsock_bind(outsock, io, srcaddr, port, qos.tos, qos.cos, socket_read, NULL);
- if (sock) {
- sockfd = ast_netsock_sockfd(sock);
- ast_netsock_unref(sock);
- nonlocal = 0;
- } else {
- nonlocal = 2;
- }
- }
- }
- }
- }
- peer->sockfd = sockfd;
- if (nonlocal == 1) {
- ast_log(LOG_WARNING,
- "Non-local or unbound address specified (%s) in sourceaddress for '%s', reverting to default\n",
- srcaddr,
- peer->name);
- return -1;
- } else if (nonlocal == 2) {
- ast_log(LOG_WARNING,
- "Unable to bind to sourceaddress '%s' for '%s', reverting to default\n",
- srcaddr,
- peer->name);
- return -1;
- } else {
- ast_debug(1, "Using sourceaddress %s for '%s'\n", srcaddr, peer->name);
- return 0;
- }
- }
- static void peer_destructor(void *obj)
- {
- struct iax2_peer *peer = obj;
- int callno = peer->callno;
- ast_free_acl_list(peer->acl);
- if (callno > 0) {
- ast_mutex_lock(&iaxsl[callno]);
- iax2_destroy(callno);
- ast_mutex_unlock(&iaxsl[callno]);
- }
- register_peer_exten(peer, 0);
- if (peer->dnsmgr)
- ast_dnsmgr_release(peer->dnsmgr);
- if (peer->mwi_event_sub)
- ast_event_unsubscribe(peer->mwi_event_sub);
- ast_string_field_free_memory(peer);
- }
- /*! \brief Create peer structure based on configuration */
- static struct iax2_peer *build_peer(const char *name, struct ast_variable *v, struct ast_variable *alt, int temponly)
- {
- struct iax2_peer *peer = NULL;
- struct ast_acl_list *oldacl = NULL;
- int maskfound = 0;
- int found = 0;
- int firstpass = 1;
- int subscribe_acl_change = 0;
- if (!temponly) {
- peer = ao2_find(peers, name, OBJ_KEY);
- if (peer && !ast_test_flag64(peer, IAX_DELME))
- firstpass = 0;
- }
- if (peer) {
- found++;
- if (firstpass) {
- oldacl = peer->acl;
- peer->acl = NULL;
- }
- unlink_peer(peer);
- } else if ((peer = ao2_alloc(sizeof(*peer), peer_destructor))) {
- peer->expire = -1;
- peer->pokeexpire = -1;
- peer->sockfd = defaultsockfd;
- peer->addr.ss.ss_family = AF_INET;
- peer->addr.len = sizeof(struct sockaddr_in);
- if (ast_string_field_init(peer, 32))
- peer = peer_unref(peer);
- }
- if (peer) {
- if (firstpass) {
- ast_copy_flags64(peer, &globalflags, IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE | IAX_FORCE_ENCRYPT);
- peer->encmethods = iax2_encryption;
- peer->adsi = adsi;
- ast_string_field_set(peer,secret,"");
- if (!found) {
- ast_string_field_set(peer, name, name);
- ast_sockaddr_set_port(&peer->addr, IAX_DEFAULT_PORTNO);
- peer->expiry = min_reg_expire;
- }
- peer->prefs = prefs;
- peer->capability = iax2_capability;
- peer->smoothing = 0;
- peer->pokefreqok = DEFAULT_FREQ_OK;
- peer->pokefreqnotok = DEFAULT_FREQ_NOTOK;
- peer->maxcallno = 0;
- peercnt_modify((unsigned char) 0, 0, &peer->addr);
- peer->calltoken_required = CALLTOKEN_DEFAULT;
- ast_string_field_set(peer,context,"");
- ast_string_field_set(peer,peercontext,"");
- ast_clear_flag64(peer, IAX_HASCALLERID);
- ast_string_field_set(peer, cid_name, "");
- ast_string_field_set(peer, cid_num, "");
- ast_string_field_set(peer, mohinterpret, mohinterpret);
- ast_string_field_set(peer, mohsuggest, mohsuggest);
- }
- if (!v) {
- v = alt;
- alt = NULL;
- }
- while(v) {
- if (!strcasecmp(v->name, "secret")) {
- ast_string_field_set(peer, secret, v->value);
- } else if (!strcasecmp(v->name, "mailbox")) {
- ast_string_field_set(peer, mailbox, v->value);
- } else if (!strcasecmp(v->name, "hasvoicemail")) {
- if (ast_true(v->value) && ast_strlen_zero(peer->mailbox)) {
- ast_string_field_set(peer, mailbox, name);
- }
- } else if (!strcasecmp(v->name, "mohinterpret")) {
- ast_string_field_set(peer, mohinterpret, v->value);
- } else if (!strcasecmp(v->name, "mohsuggest")) {
- ast_string_field_set(peer, mohsuggest, v->value);
- } else if (!strcasecmp(v->name, "dbsecret")) {
- ast_string_field_set(peer, dbsecret, v->value);
- } else if (!strcasecmp(v->name, "description")) {
- ast_string_field_set(peer, description, v->value);
- } else if (!strcasecmp(v->name, "trunk")) {
- ast_set2_flag64(peer, ast_true(v->value), IAX_TRUNK);
- if (ast_test_flag64(peer, IAX_TRUNK) && !timer) {
- ast_log(LOG_WARNING, "Unable to support trunking on peer '%s' without a timing interface\n", peer->name);
- ast_clear_flag64(peer, IAX_TRUNK);
- }
- } else if (!strcasecmp(v->name, "auth")) {
- peer->authmethods = get_auth_methods(v->value);
- } else if (!strcasecmp(v->name, "encryption")) {
- peer->encmethods |= get_encrypt_methods(v->value);
- if (!peer->encmethods) {
- ast_clear_flag64(peer, IAX_FORCE_ENCRYPT);
- }
- } else if (!strcasecmp(v->name, "forceencryption")) {
- if (ast_false(v->value)) {
- ast_clear_flag64(peer, IAX_FORCE_ENCRYPT);
- } else {
- peer->encmethods |= get_encrypt_methods(v->value);
- if (peer->encmethods) {
- ast_set_flag64(peer, IAX_FORCE_ENCRYPT);
- }
- }
- } else if (!strcasecmp(v->name, "transfer")) {
- if (!strcasecmp(v->value, "mediaonly")) {
- ast_set_flags_to64(peer, IAX_NOTRANSFER|IAX_TRANSFERMEDIA, IAX_TRANSFERMEDIA);
- } else if (ast_true(v->value)) {
- ast_set_flags_to64(peer, IAX_NOTRANSFER|IAX_TRANSFERMEDIA, 0);
- } else
- ast_set_flags_to64(peer, IAX_NOTRANSFER|IAX_TRANSFERMEDIA, IAX_NOTRANSFER);
- } else if (!strcasecmp(v->name, "jitterbuffer")) {
- ast_set2_flag64(peer, ast_true(v->value), IAX_USEJITTERBUF);
- } else if (!strcasecmp(v->name, "forcejitterbuffer")) {
- ast_set2_flag64(peer, ast_true(v->value), IAX_FORCEJITTERBUF);
- } else if (!strcasecmp(v->name, "host")) {
- if (!strcasecmp(v->value, "dynamic")) {
- /* They'll register with us */
- ast_set_flag64(peer, IAX_DYNAMIC);
- if (!found) {
- /* Initialize stuff iff we're not found, otherwise
- we keep going with what we had */
- if (ast_sockaddr_port(&peer->addr)) {
- peer->defaddr.sin_port = htons(ast_sockaddr_port(&peer->addr));
- }
- ast_sockaddr_setnull(&peer->addr);
- }
- } else {
- /* Non-dynamic. Make sure we become that way if we're not */
- AST_SCHED_DEL(sched, peer->expire);
- ast_clear_flag64(peer, IAX_DYNAMIC);
- if (ast_dnsmgr_lookup(v->value, &peer->addr, &peer->dnsmgr, srvlookup ? "_iax._udp" : NULL))
- return peer_unref(peer);
- if (!ast_sockaddr_port(&peer->addr)) {
- ast_sockaddr_set_port(&peer->addr, IAX_DEFAULT_PORTNO);
- }
- }
- if (!maskfound)
- inet_aton("255.255.255.255", &peer->mask);
- } else if (!strcasecmp(v->name, "defaultip")) {
- struct ast_sockaddr peer_defaddr_tmp;
- peer_defaddr_tmp.ss.ss_family = AF_INET;
- if (ast_get_ip(&peer_defaddr_tmp, v->value)) {
- return peer_unref(peer);
- }
- ast_sockaddr_to_sin(&peer_defaddr_tmp,
- &peer->defaddr);
- } else if (!strcasecmp(v->name, "sourceaddress")) {
- peer_set_srcaddr(peer, v->value);
- } else if (!strcasecmp(v->name, "permit") ||
- !strcasecmp(v->name, "deny") ||
- !strcasecmp(v->name, "acl")) {
- ast_append_acl(v->name, v->value, &peer->acl, NULL, &subscribe_acl_change);
- } else if (!strcasecmp(v->name, "mask")) {
- maskfound++;
- inet_aton(v->value, &peer->mask);
- } else if (!strcasecmp(v->name, "context")) {
- ast_string_field_set(peer, context, v->value);
- } else if (!strcasecmp(v->name, "regexten")) {
- ast_string_field_set(peer, regexten, v->value);
- } else if (!strcasecmp(v->name, "peercontext")) {
- ast_string_field_set(peer, peercontext, v->value);
- } else if (!strcasecmp(v->name, "port")) {
- if (ast_test_flag64(peer, IAX_DYNAMIC)) {
- peer->defaddr.sin_port = htons(atoi(v->value));
- } else {
- ast_sockaddr_set_port(&peer->addr, atoi(v->value));
- }
- } else if (!strcasecmp(v->name, "username")) {
- ast_string_field_set(peer, username, v->value);
- } else if (!strcasecmp(v->name, "allow")) {
- iax2_parse_allow_disallow(&peer->prefs, &peer->capability, v->value, 1);
- } else if (!strcasecmp(v->name, "disallow")) {
- iax2_parse_allow_disallow(&peer->prefs, &peer->capability, v->value, 0);
- } else if (!strcasecmp(v->name, "callerid")) {
- if (!ast_strlen_zero(v->value)) {
- char name2[80];
- char num2[80];
- ast_callerid_split(v->value, name2, sizeof(name2), num2, sizeof(num2));
- ast_string_field_set(peer, cid_name, name2);
- ast_string_field_set(peer, cid_num, num2);
- } else {
- ast_string_field_set(peer, cid_name, "");
- ast_string_field_set(peer, cid_num, "");
- }
- ast_set_flag64(peer, IAX_HASCALLERID);
- } else if (!strcasecmp(v->name, "fullname")) {
- ast_string_field_set(peer, cid_name, S_OR(v->value, ""));
- ast_set_flag64(peer, IAX_HASCALLERID);
- } else if (!strcasecmp(v->name, "cid_number")) {
- ast_string_field_set(peer, cid_num, S_OR(v->value, ""));
- ast_set_flag64(peer, IAX_HASCALLERID);
- } else if (!strcasecmp(v->name, "sendani")) {
- ast_set2_flag64(peer, ast_true(v->value), IAX_SENDANI);
- } else if (!strcasecmp(v->name, "inkeys")) {
- ast_string_field_set(peer, inkeys, v->value);
- } else if (!strcasecmp(v->name, "outkey")) {
- ast_string_field_set(peer, outkey, v->value);
- } else if (!strcasecmp(v->name, "qualify")) {
- if (!strcasecmp(v->value, "no")) {
- peer->maxms = 0;
- } else if (!strcasecmp(v->value, "yes")) {
- peer->maxms = DEFAULT_MAXMS;
- } else if (sscanf(v->value, "%30d", &peer->maxms) != 1) {
- ast_log(LOG_WARNING, "Qualification of peer '%s' should be 'yes', 'no', or a number of milliseconds at line %d of iax.conf\n", peer->name, v->lineno);
- peer->maxms = 0;
- }
- } else if (!strcasecmp(v->name, "qualifysmoothing")) {
- peer->smoothing = ast_true(v->value);
- } else if (!strcasecmp(v->name, "qualifyfreqok")) {
- if (sscanf(v->value, "%30d", &peer->pokefreqok) != 1) {
- ast_log(LOG_WARNING, "Qualification testing frequency of peer '%s' when OK should a number of milliseconds at line %d of iax.conf\n", peer->name, v->lineno);
- }
- } else if (!strcasecmp(v->name, "qualifyfreqnotok")) {
- if (sscanf(v->value, "%30d", &peer->pokefreqnotok) != 1) {
- ast_log(LOG_WARNING, "Qualification testing frequency of peer '%s' when NOT OK should be a number of milliseconds at line %d of iax.conf\n", peer->name, v->lineno);
- }
- } else if (!strcasecmp(v->name, "timezone")) {
- ast_string_field_set(peer, zonetag, v->value);
- } else if (!strcasecmp(v->name, "adsi")) {
- peer->adsi = ast_true(v->value);
- } else if (!strcasecmp(v->name, "connectedline")) {
- if (ast_true(v->value)) {
- ast_set_flag64(peer, IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- } else if (!strcasecmp(v->value, "send")) {
- ast_clear_flag64(peer, IAX_RECVCONNECTEDLINE);
- ast_set_flag64(peer, IAX_SENDCONNECTEDLINE);
- } else if (!strcasecmp(v->value, "receive")) {
- ast_clear_flag64(peer, IAX_SENDCONNECTEDLINE);
- ast_set_flag64(peer, IAX_RECVCONNECTEDLINE);
- } else {
- ast_clear_flag64(peer, IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- }
- } else if (!strcasecmp(v->name, "maxcallnumbers")) {
- if (sscanf(v->value, "%10hu", &peer->maxcallno) != 1) {
- ast_log(LOG_WARNING, "maxcallnumbers must be set to a valid number. %s is not valid at line %d.\n", v->value, v->lineno);
- } else {
- peercnt_modify((unsigned char) 1, peer->maxcallno, &peer->addr);
- }
- } else if (!strcasecmp(v->name, "requirecalltoken")) {
- /* default is required unless in optional ip list */
- if (ast_false(v->value)) {
- peer->calltoken_required = CALLTOKEN_NO;
- } else if (!strcasecmp(v->value, "auto")) {
- peer->calltoken_required = CALLTOKEN_AUTO;
- } else if (ast_true(v->value)) {
- peer->calltoken_required = CALLTOKEN_YES;
- } else {
- ast_log(LOG_WARNING, "requirecalltoken must be set to a valid value. at line %d\n", v->lineno);
- }
- } /* else if (strcasecmp(v->name,"type")) */
- /* ast_log(LOG_WARNING, "Ignoring %s\n", v->name); */
- v = v->next;
- if (!v) {
- v = alt;
- alt = NULL;
- }
- }
- if (!peer->authmethods)
- peer->authmethods = IAX_AUTH_MD5 | IAX_AUTH_PLAINTEXT;
- ast_clear_flag64(peer, IAX_DELME);
- }
- if (oldacl)
- ast_free_acl_list(oldacl);
- if (!ast_strlen_zero(peer->mailbox)) {
- char *mailbox, *context;
- context = mailbox = ast_strdupa(peer->mailbox);
- strsep(&context, "@");
- if (ast_strlen_zero(context))
- context = "default";
- peer->mwi_event_sub = ast_event_subscribe(AST_EVENT_MWI, mwi_event_cb, "IAX MWI subscription", NULL,
- AST_EVENT_IE_MAILBOX, AST_EVENT_IE_PLTYPE_STR, mailbox,
- AST_EVENT_IE_CONTEXT, AST_EVENT_IE_PLTYPE_STR, context,
- AST_EVENT_IE_END);
- }
- if (subscribe_acl_change) {
- acl_change_event_subscribe();
- }
- return peer;
- }
- static void user_destructor(void *obj)
- {
- struct iax2_user *user = obj;
- ast_free_acl_list(user->acl);
- free_context(user->contexts);
- if(user->vars) {
- ast_variables_destroy(user->vars);
- user->vars = NULL;
- }
- ast_string_field_free_memory(user);
- }
- /*! \brief Create in-memory user structure from configuration */
- static struct iax2_user *build_user(const char *name, struct ast_variable *v, struct ast_variable *alt, int temponly)
- {
- struct iax2_user *user = NULL;
- struct iax2_context *con, *conl = NULL;
- struct ast_acl_list *oldacl = NULL;
- struct iax2_context *oldcon = NULL;
- int format;
- int firstpass=1;
- int oldcurauthreq = 0;
- int subscribe_acl_change = 0;
- char *varname = NULL, *varval = NULL;
- struct ast_variable *tmpvar = NULL;
- if (!temponly) {
- user = ao2_find(users, name, OBJ_KEY);
- if (user && !ast_test_flag64(user, IAX_DELME))
- firstpass = 0;
- }
- if (user) {
- if (firstpass) {
- oldcurauthreq = user->curauthreq;
- oldacl = user->acl;
- oldcon = user->contexts;
- user->acl = NULL;
- user->contexts = NULL;
- }
- /* Already in the list, remove it and it will be added back (or FREE'd) */
- ao2_unlink(users, user);
- } else {
- user = ao2_alloc(sizeof(*user), user_destructor);
- }
-
- if (user) {
- if (firstpass) {
- ast_string_field_free_memory(user);
- memset(user, 0, sizeof(struct iax2_user));
- if (ast_string_field_init(user, 32)) {
- user = user_unref(user);
- goto cleanup;
- }
- user->maxauthreq = maxauthreq;
- user->curauthreq = oldcurauthreq;
- user->prefs = prefs;
- user->capability = iax2_capability;
- user->encmethods = iax2_encryption;
- user->adsi = adsi;
- user->calltoken_required = CALLTOKEN_DEFAULT;
- ast_string_field_set(user, name, name);
- ast_string_field_set(user, language, language);
- ast_copy_flags64(user, &globalflags, IAX_USEJITTERBUF | IAX_FORCEJITTERBUF | IAX_CODEC_USER_FIRST | IAX_CODEC_NOPREFS | IAX_CODEC_NOCAP | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE | IAX_FORCE_ENCRYPT);
- ast_clear_flag64(user, IAX_HASCALLERID);
- ast_string_field_set(user, cid_name, "");
- ast_string_field_set(user, cid_num, "");
- ast_string_field_set(user, accountcode, accountcode);
- ast_string_field_set(user, mohinterpret, mohinterpret);
- ast_string_field_set(user, mohsuggest, mohsuggest);
- }
- if (!v) {
- v = alt;
- alt = NULL;
- }
- while(v) {
- if (!strcasecmp(v->name, "context")) {
- con = build_context(v->value);
- if (con) {
- if (conl)
- conl->next = con;
- else
- user->contexts = con;
- conl = con;
- }
- } else if (!strcasecmp(v->name, "permit") ||
- !strcasecmp(v->name, "deny") ||
- !strcasecmp(v->name, "acl")) {
- ast_append_acl(v->name, v->value, &user->acl, NULL, &subscribe_acl_change);
- } else if (!strcasecmp(v->name, "setvar")) {
- varname = ast_strdupa(v->value);
- if ((varval = strchr(varname, '='))) {
- *varval = '\0';
- varval++;
- if((tmpvar = ast_variable_new(varname, varval, ""))) {
- tmpvar->next = user->vars;
- user->vars = tmpvar;
- }
- }
- } else if (!strcasecmp(v->name, "allow")) {
- iax2_parse_allow_disallow(&user->prefs, &user->capability, v->value, 1);
- } else if (!strcasecmp(v->name, "disallow")) {
- iax2_parse_allow_disallow(&user->prefs, &user->capability,v->value, 0);
- } else if (!strcasecmp(v->name, "trunk")) {
- ast_set2_flag64(user, ast_true(v->value), IAX_TRUNK);
- if (ast_test_flag64(user, IAX_TRUNK) && !timer) {
- ast_log(LOG_WARNING, "Unable to support trunking on user '%s' without a timing interface\n", user->name);
- ast_clear_flag64(user, IAX_TRUNK);
- }
- } else if (!strcasecmp(v->name, "auth")) {
- user->authmethods = get_auth_methods(v->value);
- } else if (!strcasecmp(v->name, "encryption")) {
- user->encmethods |= get_encrypt_methods(v->value);
- if (!user->encmethods) {
- ast_clear_flag64(user, IAX_FORCE_ENCRYPT);
- }
- } else if (!strcasecmp(v->name, "forceencryption")) {
- if (ast_false(v->value)) {
- ast_clear_flag64(user, IAX_FORCE_ENCRYPT);
- } else {
- user->encmethods |= get_encrypt_methods(v->value);
- if (user->encmethods) {
- ast_set_flag64(user, IAX_FORCE_ENCRYPT);
- }
- }
- } else if (!strcasecmp(v->name, "transfer")) {
- if (!strcasecmp(v->value, "mediaonly")) {
- ast_set_flags_to64(user, IAX_NOTRANSFER|IAX_TRANSFERMEDIA, IAX_TRANSFERMEDIA);
- } else if (ast_true(v->value)) {
- ast_set_flags_to64(user, IAX_NOTRANSFER|IAX_TRANSFERMEDIA, 0);
- } else
- ast_set_flags_to64(user, IAX_NOTRANSFER|IAX_TRANSFERMEDIA, IAX_NOTRANSFER);
- } else if (!strcasecmp(v->name, "codecpriority")) {
- if(!strcasecmp(v->value, "caller"))
- ast_set_flag64(user, IAX_CODEC_USER_FIRST);
- else if(!strcasecmp(v->value, "disabled"))
- ast_set_flag64(user, IAX_CODEC_NOPREFS);
- else if(!strcasecmp(v->value, "reqonly")) {
- ast_set_flag64(user, IAX_CODEC_NOCAP);
- ast_set_flag64(user, IAX_CODEC_NOPREFS);
- }
- } else if (!strcasecmp(v->name, "immediate")) {
- ast_set2_flag64(user, ast_true(v->value), IAX_IMMEDIATE);
- } else if (!strcasecmp(v->name, "jitterbuffer")) {
- ast_set2_flag64(user, ast_true(v->value), IAX_USEJITTERBUF);
- } else if (!strcasecmp(v->name, "forcejitterbuffer")) {
- ast_set2_flag64(user, ast_true(v->value), IAX_FORCEJITTERBUF);
- } else if (!strcasecmp(v->name, "dbsecret")) {
- ast_string_field_set(user, dbsecret, v->value);
- } else if (!strcasecmp(v->name, "secret")) {
- if (!ast_strlen_zero(user->secret)) {
- char *old = ast_strdupa(user->secret);
- ast_string_field_build(user, secret, "%s;%s", old, v->value);
- } else
- ast_string_field_set(user, secret, v->value);
- } else if (!strcasecmp(v->name, "callerid")) {
- if (!ast_strlen_zero(v->value) && strcasecmp(v->value, "asreceived")) {
- char name2[80];
- char num2[80];
- ast_callerid_split(v->value, name2, sizeof(name2), num2, sizeof(num2));
- ast_string_field_set(user, cid_name, name2);
- ast_string_field_set(user, cid_num, num2);
- ast_set_flag64(user, IAX_HASCALLERID);
- } else {
- ast_clear_flag64(user, IAX_HASCALLERID);
- ast_string_field_set(user, cid_name, "");
- ast_string_field_set(user, cid_num, "");
- }
- } else if (!strcasecmp(v->name, "fullname")) {
- if (!ast_strlen_zero(v->value)) {
- ast_string_field_set(user, cid_name, v->value);
- ast_set_flag64(user, IAX_HASCALLERID);
- } else {
- ast_string_field_set(user, cid_name, "");
- if (ast_strlen_zero(user->cid_num))
- ast_clear_flag64(user, IAX_HASCALLERID);
- }
- } else if (!strcasecmp(v->name, "cid_number")) {
- if (!ast_strlen_zero(v->value)) {
- ast_string_field_set(user, cid_num, v->value);
- ast_set_flag64(user, IAX_HASCALLERID);
- } else {
- ast_string_field_set(user, cid_num, "");
- if (ast_strlen_zero(user->cid_name))
- ast_clear_flag64(user, IAX_HASCALLERID);
- }
- } else if (!strcasecmp(v->name, "accountcode")) {
- ast_string_field_set(user, accountcode, v->value);
- } else if (!strcasecmp(v->name, "mohinterpret")) {
- ast_string_field_set(user, mohinterpret, v->value);
- } else if (!strcasecmp(v->name, "mohsuggest")) {
- ast_string_field_set(user, mohsuggest, v->value);
- } else if (!strcasecmp(v->name, "parkinglot")) {
- ast_string_field_set(user, parkinglot, v->value);
- } else if (!strcasecmp(v->name, "language")) {
- ast_string_field_set(user, language, v->value);
- } else if (!strcasecmp(v->name, "amaflags")) {
- format = ast_cdr_amaflags2int(v->value);
- if (format < 0) {
- ast_log(LOG_WARNING, "Invalid AMA Flags: %s at line %d\n", v->value, v->lineno);
- } else {
- user->amaflags = format;
- }
- } else if (!strcasecmp(v->name, "inkeys")) {
- ast_string_field_set(user, inkeys, v->value);
- } else if (!strcasecmp(v->name, "maxauthreq")) {
- user->maxauthreq = atoi(v->value);
- if (user->maxauthreq < 0)
- user->maxauthreq = 0;
- } else if (!strcasecmp(v->name, "adsi")) {
- user->adsi = ast_true(v->value);
- } else if (!strcasecmp(v->name, "connectedline")) {
- if (ast_true(v->value)) {
- ast_set_flag64(user, IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- } else if (!strcasecmp(v->value, "send")) {
- ast_clear_flag64(user, IAX_RECVCONNECTEDLINE);
- ast_set_flag64(user, IAX_SENDCONNECTEDLINE);
- } else if (!strcasecmp(v->value, "receive")) {
- ast_clear_flag64(user, IAX_SENDCONNECTEDLINE);
- ast_set_flag64(user, IAX_RECVCONNECTEDLINE);
- } else {
- ast_clear_flag64(user, IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- }
- } else if (!strcasecmp(v->name, "requirecalltoken")) {
- /* default is required unless in optional ip list */
- if (ast_false(v->value)) {
- user->calltoken_required = CALLTOKEN_NO;
- } else if (!strcasecmp(v->value, "auto")) {
- user->calltoken_required = CALLTOKEN_AUTO;
- } else if (ast_true(v->value)) {
- user->calltoken_required = CALLTOKEN_YES;
- } else {
- ast_log(LOG_WARNING, "requirecalltoken must be set to a valid value. at line %d\n", v->lineno);
- }
- } /* else if (strcasecmp(v->name,"type")) */
- /* ast_log(LOG_WARNING, "Ignoring %s\n", v->name); */
- v = v->next;
- if (!v) {
- v = alt;
- alt = NULL;
- }
- }
- if (!user->authmethods) {
- if (!ast_strlen_zero(user->secret)) {
- user->authmethods = IAX_AUTH_MD5 | IAX_AUTH_PLAINTEXT;
- if (!ast_strlen_zero(user->inkeys))
- user->authmethods |= IAX_AUTH_RSA;
- } else if (!ast_strlen_zero(user->inkeys)) {
- user->authmethods = IAX_AUTH_RSA;
- } else {
- user->authmethods = IAX_AUTH_MD5 | IAX_AUTH_PLAINTEXT;
- }
- }
- ast_clear_flag64(user, IAX_DELME);
- }
- cleanup:
- if (oldacl) {
- ast_free_acl_list(oldacl);
- }
- if (oldcon) {
- free_context(oldcon);
- }
- if (subscribe_acl_change) {
- acl_change_event_subscribe();
- }
- return user;
- }
- static int peer_delme_cb(void *obj, void *arg, int flags)
- {
- struct iax2_peer *peer = obj;
- ast_set_flag64(peer, IAX_DELME);
- return 0;
- }
- static int user_delme_cb(void *obj, void *arg, int flags)
- {
- struct iax2_user *user = obj;
- ast_set_flag64(user, IAX_DELME);
- return 0;
- }
- static void delete_users(void)
- {
- struct iax2_registry *reg;
- ao2_callback(users, OBJ_NODATA, user_delme_cb, NULL);
- AST_LIST_LOCK(®istrations);
- while ((reg = AST_LIST_REMOVE_HEAD(®istrations, entry))) {
- if (sched) {
- AST_SCHED_DEL(sched, reg->expire);
- }
- if (reg->callno) {
- int callno = reg->callno;
- ast_mutex_lock(&iaxsl[callno]);
- if (iaxs[callno]) {
- iaxs[callno]->reg = NULL;
- iax2_destroy(callno);
- }
- ast_mutex_unlock(&iaxsl[callno]);
- }
- if (reg->dnsmgr)
- ast_dnsmgr_release(reg->dnsmgr);
- ast_free(reg);
- }
- AST_LIST_UNLOCK(®istrations);
- ao2_callback(peers, OBJ_NODATA, peer_delme_cb, NULL);
- }
- static void prune_users(void)
- {
- struct iax2_user *user;
- struct ao2_iterator i;
- i = ao2_iterator_init(users, 0);
- while ((user = ao2_iterator_next(&i))) {
- if (ast_test_flag64(user, IAX_DELME) || ast_test_flag64(user, IAX_RTCACHEFRIENDS)) {
- ao2_unlink(users, user);
- }
- user_unref(user);
- }
- ao2_iterator_destroy(&i);
- }
- /* Prune peers who still are supposed to be deleted */
- static void prune_peers(void)
- {
- struct iax2_peer *peer;
- struct ao2_iterator i;
- i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- if (ast_test_flag64(peer, IAX_DELME) || ast_test_flag64(peer, IAX_RTCACHEFRIENDS)) {
- unlink_peer(peer);
- }
- peer_unref(peer);
- }
- ao2_iterator_destroy(&i);
- }
- static void set_config_destroy(void)
- {
- strcpy(accountcode, "");
- strcpy(language, "");
- strcpy(mohinterpret, "");
- strcpy(mohsuggest, "");
- trunkmaxsize = MAX_TRUNKDATA;
- amaflags = 0;
- delayreject = 0;
- ast_clear_flag64((&globalflags), IAX_NOTRANSFER | IAX_TRANSFERMEDIA | IAX_USEJITTERBUF |
- IAX_FORCEJITTERBUF | IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- delete_users();
- ao2_callback(callno_limits, OBJ_NODATA, addr_range_delme_cb, NULL);
- ao2_callback(calltoken_ignores, OBJ_NODATA, addr_range_delme_cb, NULL);
- }
- /*! \brief Load configuration */
- static int set_config(const char *config_file, int reload, int forced)
- {
- struct ast_config *cfg, *ucfg;
- iax2_format capability = iax2_capability;
- struct ast_variable *v;
- char *cat;
- const char *utype;
- const char *tosval;
- int format;
- int portno = IAX_DEFAULT_PORTNO;
- int x;
- int mtuv;
- int subscribe_network_change = 1;
- struct iax2_user *user;
- struct iax2_peer *peer;
- struct ast_netsock *ns;
- struct ast_flags config_flags = { (reload && !forced) ? CONFIG_FLAG_FILEUNCHANGED : 0 };
- #if 0
- static unsigned short int last_port=0;
- #endif
- cfg = ast_config_load(config_file, config_flags);
- if (!cfg) {
- ast_log(LOG_ERROR, "Unable to load config %s\n", config_file);
- return -1;
- } else if (cfg == CONFIG_STATUS_FILEUNCHANGED) {
- ucfg = ast_config_load("users.conf", config_flags);
- if (ucfg == CONFIG_STATUS_FILEUNCHANGED)
- return 0;
- /* Otherwise we need to reread both files */
- ast_clear_flag(&config_flags, CONFIG_FLAG_FILEUNCHANGED);
- if ((cfg = ast_config_load(config_file, config_flags)) == CONFIG_STATUS_FILEINVALID) {
- ast_log(LOG_ERROR, "Config file %s is in an invalid format. Aborting.\n", config_file);
- ast_config_destroy(ucfg);
- return 0;
- }
- if (!cfg) {
- /* should have been able to load the config here */
- ast_log(LOG_ERROR, "Unable to load config %s again\n", config_file);
- return -1;
- }
- } else if (cfg == CONFIG_STATUS_FILEINVALID) {
- ast_log(LOG_ERROR, "Config file %s is in an invalid format. Aborting.\n", config_file);
- return 0;
- } else { /* iax.conf changed, gotta reread users.conf, too */
- ast_clear_flag(&config_flags, CONFIG_FLAG_FILEUNCHANGED);
- if ((ucfg = ast_config_load("users.conf", config_flags)) == CONFIG_STATUS_FILEINVALID) {
- ast_log(LOG_ERROR, "Config file users.conf is in an invalid format. Aborting.\n");
- ast_config_destroy(cfg);
- return 0;
- }
- }
- if (reload) {
- set_config_destroy();
- }
- /* Reset global codec prefs */
- memset(&prefs, 0 , sizeof(struct ast_codec_pref));
- /* Reset Global Flags */
- memset(&globalflags, 0, sizeof(globalflags));
- ast_set_flag64(&globalflags, IAX_RTUPDATE);
- ast_set_flag64((&globalflags), IAX_SHRINKCALLERID);
- #ifdef SO_NO_CHECK
- nochecksums = 0;
- #endif
- /* Reset default parking lot */
- default_parkinglot[0] = '\0';
- min_reg_expire = IAX_DEFAULT_REG_EXPIRE;
- max_reg_expire = IAX_DEFAULT_REG_EXPIRE;
- global_max_trunk_mtu = MAX_TRUNK_MTU;
- global_maxcallno = DEFAULT_MAXCALLNO_LIMIT;
- global_maxcallno_nonval = DEFAULT_MAXCALLNO_LIMIT_NONVAL;
- maxauthreq = 3;
- srvlookup = 0;
- v = ast_variable_browse(cfg, "general");
- /* Seed initial tos value */
- tosval = ast_variable_retrieve(cfg, "general", "tos");
- if (tosval) {
- if (ast_str2tos(tosval, &qos.tos))
- ast_log(LOG_WARNING, "Invalid tos value, refer to QoS documentation\n");
- }
- /* Seed initial cos value */
- tosval = ast_variable_retrieve(cfg, "general", "cos");
- if (tosval) {
- if (ast_str2cos(tosval, &qos.cos))
- ast_log(LOG_WARNING, "Invalid cos value, refer to QoS documentation\n");
- }
- while(v) {
- if (!strcasecmp(v->name, "bindport")){
- if (reload)
- ast_log(LOG_NOTICE, "Ignoring bindport on reload\n");
- else
- portno = atoi(v->value);
- } else if (!strcasecmp(v->name, "pingtime"))
- ping_time = atoi(v->value);
- else if (!strcasecmp(v->name, "iaxthreadcount")) {
- if (reload) {
- if (atoi(v->value) != iaxthreadcount)
- ast_log(LOG_NOTICE, "Ignoring any changes to iaxthreadcount during reload\n");
- } else {
- iaxthreadcount = atoi(v->value);
- if (iaxthreadcount < 1) {
- ast_log(LOG_NOTICE, "iaxthreadcount must be at least 1.\n");
- iaxthreadcount = 1;
- } else if (iaxthreadcount > 256) {
- ast_log(LOG_NOTICE, "limiting iaxthreadcount to 256\n");
- iaxthreadcount = 256;
- }
- }
- } else if (!strcasecmp(v->name, "iaxmaxthreadcount")) {
- if (reload) {
- AST_LIST_LOCK(&dynamic_list);
- iaxmaxthreadcount = atoi(v->value);
- AST_LIST_UNLOCK(&dynamic_list);
- } else {
- iaxmaxthreadcount = atoi(v->value);
- if (iaxmaxthreadcount < 0) {
- ast_log(LOG_NOTICE, "iaxmaxthreadcount must be at least 0.\n");
- iaxmaxthreadcount = 0;
- } else if (iaxmaxthreadcount > 256) {
- ast_log(LOG_NOTICE, "Limiting iaxmaxthreadcount to 256\n");
- iaxmaxthreadcount = 256;
- }
- }
- } else if (!strcasecmp(v->name, "nochecksums")) {
- #ifdef SO_NO_CHECK
- if (ast_true(v->value))
- nochecksums = 1;
- else
- nochecksums = 0;
- #else
- if (ast_true(v->value))
- ast_log(LOG_WARNING, "Disabling RTP checksums is not supported on this operating system!\n");
- #endif
- }
- else if (!strcasecmp(v->name, "maxjitterbuffer"))
- maxjitterbuffer = atoi(v->value);
- else if (!strcasecmp(v->name, "resyncthreshold"))
- resyncthreshold = atoi(v->value);
- else if (!strcasecmp(v->name, "maxjitterinterps"))
- maxjitterinterps = atoi(v->value);
- else if (!strcasecmp(v->name, "jittertargetextra"))
- jittertargetextra = atoi(v->value);
- else if (!strcasecmp(v->name, "lagrqtime"))
- lagrq_time = atoi(v->value);
- else if (!strcasecmp(v->name, "maxregexpire"))
- max_reg_expire = atoi(v->value);
- else if (!strcasecmp(v->name, "minregexpire"))
- min_reg_expire = atoi(v->value);
- else if (!strcasecmp(v->name, "bindaddr")) {
- if (reload) {
- ast_log(LOG_NOTICE, "Ignoring bindaddr on reload\n");
- } else {
- if (!(ns = ast_netsock_bind(netsock, io, v->value, portno, qos.tos, qos.cos, socket_read, NULL))) {
- ast_log(LOG_WARNING, "Unable apply binding to '%s' at line %d\n", v->value, v->lineno);
- } else {
- if (strchr(v->value, ':'))
- ast_verb(2, "Binding IAX2 to '%s'\n", v->value);
- else
- ast_verb(2, "Binding IAX2 to '%s:%d'\n", v->value, portno);
- if (defaultsockfd < 0)
- defaultsockfd = ast_netsock_sockfd(ns);
- ast_netsock_unref(ns);
- }
- }
- } else if (!strcasecmp(v->name, "authdebug")) {
- authdebug = ast_true(v->value);
- } else if (!strcasecmp(v->name, "encryption")) {
- iax2_encryption |= get_encrypt_methods(v->value);
- if (!iax2_encryption) {
- ast_clear_flag64((&globalflags), IAX_FORCE_ENCRYPT);
- }
- } else if (!strcasecmp(v->name, "forceencryption")) {
- if (ast_false(v->value)) {
- ast_clear_flag64((&globalflags), IAX_FORCE_ENCRYPT);
- } else {
- iax2_encryption |= get_encrypt_methods(v->value);
- if (iax2_encryption) {
- ast_set_flag64((&globalflags), IAX_FORCE_ENCRYPT);
- }
- }
- } else if (!strcasecmp(v->name, "transfer")) {
- if (!strcasecmp(v->value, "mediaonly")) {
- ast_set_flags_to64((&globalflags), IAX_NOTRANSFER|IAX_TRANSFERMEDIA, IAX_TRANSFERMEDIA);
- } else if (ast_true(v->value)) {
- ast_set_flags_to64((&globalflags), IAX_NOTRANSFER|IAX_TRANSFERMEDIA, 0);
- } else
- ast_set_flags_to64((&globalflags), IAX_NOTRANSFER|IAX_TRANSFERMEDIA, IAX_NOTRANSFER);
- } else if (!strcasecmp(v->name, "codecpriority")) {
- if(!strcasecmp(v->value, "caller"))
- ast_set_flag64((&globalflags), IAX_CODEC_USER_FIRST);
- else if(!strcasecmp(v->value, "disabled"))
- ast_set_flag64((&globalflags), IAX_CODEC_NOPREFS);
- else if(!strcasecmp(v->value, "reqonly")) {
- ast_set_flag64((&globalflags), IAX_CODEC_NOCAP);
- ast_set_flag64((&globalflags), IAX_CODEC_NOPREFS);
- }
- } else if (!strcasecmp(v->name, "jitterbuffer"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_USEJITTERBUF);
- else if (!strcasecmp(v->name, "forcejitterbuffer"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_FORCEJITTERBUF);
- else if (!strcasecmp(v->name, "delayreject"))
- delayreject = ast_true(v->value);
- else if (!strcasecmp(v->name, "allowfwdownload"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_ALLOWFWDOWNLOAD);
- else if (!strcasecmp(v->name, "rtcachefriends"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_RTCACHEFRIENDS);
- else if (!strcasecmp(v->name, "rtignoreregexpire"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_RTIGNOREREGEXPIRE);
- else if (!strcasecmp(v->name, "rtupdate"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_RTUPDATE);
- else if (!strcasecmp(v->name, "rtsavesysname"))
- ast_set2_flag64((&globalflags), ast_true(v->value), IAX_RTSAVE_SYSNAME);
- else if (!strcasecmp(v->name, "trunktimestamps"))
- ast_set2_flag64(&globalflags, ast_true(v->value), IAX_TRUNKTIMESTAMPS);
- else if (!strcasecmp(v->name, "rtautoclear")) {
- int i = atoi(v->value);
- if(i > 0)
- global_rtautoclear = i;
- else
- i = 0;
- ast_set2_flag64((&globalflags), i || ast_true(v->value), IAX_RTAUTOCLEAR);
- } else if (!strcasecmp(v->name, "trunkfreq")) {
- trunkfreq = atoi(v->value);
- if (trunkfreq < 10) {
- ast_log(LOG_NOTICE, "trunkfreq must be between 10ms and 1000ms, using 10ms instead.\n");
- trunkfreq = 10;
- } else if (trunkfreq > 1000) {
- ast_log(LOG_NOTICE, "trunkfreq must be between 10ms and 1000ms, using 1000ms instead.\n");
- trunkfreq = 1000;
- }
- if (timer) {
- ast_timer_set_rate(timer, 1000 / trunkfreq);
- }
- } else if (!strcasecmp(v->name, "trunkmtu")) {
- mtuv = atoi(v->value);
- if (mtuv == 0 )
- global_max_trunk_mtu = 0;
- else if (mtuv >= 172 && mtuv < 4000)
- global_max_trunk_mtu = mtuv;
- else
- ast_log(LOG_NOTICE, "trunkmtu value out of bounds (%d) at line %d\n",
- mtuv, v->lineno);
- } else if (!strcasecmp(v->name, "trunkmaxsize")) {
- trunkmaxsize = atoi(v->value);
- if (trunkmaxsize == 0)
- trunkmaxsize = MAX_TRUNKDATA;
- } else if (!strcasecmp(v->name, "autokill")) {
- if (sscanf(v->value, "%30d", &x) == 1) {
- if (x >= 0)
- autokill = x;
- else
- ast_log(LOG_NOTICE, "Nice try, but autokill has to be >0 or 'yes' or 'no' at line %d\n", v->lineno);
- } else if (ast_true(v->value)) {
- autokill = DEFAULT_MAXMS;
- } else {
- autokill = 0;
- }
- } else if (!strcasecmp(v->name, "bandwidth")) {
- if (!strcasecmp(v->value, "low")) {
- capability = IAX_CAPABILITY_LOWBANDWIDTH;
- } else if (!strcasecmp(v->value, "medium")) {
- capability = IAX_CAPABILITY_MEDBANDWIDTH;
- } else if (!strcasecmp(v->value, "high")) {
- capability = IAX_CAPABILITY_FULLBANDWIDTH;
- } else
- ast_log(LOG_WARNING, "bandwidth must be either low, medium, or high\n");
- } else if (!strcasecmp(v->name, "allow")) {
- iax2_parse_allow_disallow(&prefs, &capability, v->value, 1);
- } else if (!strcasecmp(v->name, "disallow")) {
- iax2_parse_allow_disallow(&prefs, &capability, v->value, 0);
- } else if (!strcasecmp(v->name, "register")) {
- iax2_register(v->value, v->lineno);
- } else if (!strcasecmp(v->name, "iaxcompat")) {
- iaxcompat = ast_true(v->value);
- } else if (!strcasecmp(v->name, "regcontext")) {
- ast_copy_string(regcontext, v->value, sizeof(regcontext));
- /* Create context if it doesn't exist already */
- ast_context_find_or_create(NULL, NULL, regcontext, "IAX2");
- } else if (!strcasecmp(v->name, "tos")) {
- if (ast_str2tos(v->value, &qos.tos))
- ast_log(LOG_WARNING, "Invalid tos value at line %d, refer to QoS documentation\n", v->lineno);
- } else if (!strcasecmp(v->name, "cos")) {
- if (ast_str2cos(v->value, &qos.cos))
- ast_log(LOG_WARNING, "Invalid cos value at line %d, refer to QoS documentation\n", v->lineno);
- } else if (!strcasecmp(v->name, "parkinglot")) {
- ast_copy_string(default_parkinglot, v->value, sizeof(default_parkinglot));
- } else if (!strcasecmp(v->name, "accountcode")) {
- ast_copy_string(accountcode, v->value, sizeof(accountcode));
- } else if (!strcasecmp(v->name, "mohinterpret")) {
- ast_copy_string(mohinterpret, v->value, sizeof(mohinterpret));
- } else if (!strcasecmp(v->name, "mohsuggest")) {
- ast_copy_string(mohsuggest, v->value, sizeof(mohsuggest));
- } else if (!strcasecmp(v->name, "amaflags")) {
- format = ast_cdr_amaflags2int(v->value);
- if (format < 0) {
- ast_log(LOG_WARNING, "Invalid AMA Flags: %s at line %d\n", v->value, v->lineno);
- } else {
- amaflags = format;
- }
- } else if (!strcasecmp(v->name, "language")) {
- ast_copy_string(language, v->value, sizeof(language));
- } else if (!strcasecmp(v->name, "maxauthreq")) {
- maxauthreq = atoi(v->value);
- if (maxauthreq < 0)
- maxauthreq = 0;
- } else if (!strcasecmp(v->name, "adsi")) {
- adsi = ast_true(v->value);
- } else if (!strcasecmp(v->name, "srvlookup")) {
- srvlookup = ast_true(v->value);
- } else if (!strcasecmp(v->name, "connectedline")) {
- if (ast_true(v->value)) {
- ast_set_flag64((&globalflags), IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- } else if (!strcasecmp(v->value, "send")) {
- ast_clear_flag64((&globalflags), IAX_RECVCONNECTEDLINE);
- ast_set_flag64((&globalflags), IAX_SENDCONNECTEDLINE);
- } else if (!strcasecmp(v->value, "receive")) {
- ast_clear_flag64((&globalflags), IAX_SENDCONNECTEDLINE);
- ast_set_flag64((&globalflags), IAX_RECVCONNECTEDLINE);
- } else {
- ast_clear_flag64((&globalflags), IAX_SENDCONNECTEDLINE | IAX_RECVCONNECTEDLINE);
- }
- } else if (!strcasecmp(v->name, "maxcallnumbers")) {
- if (sscanf(v->value, "%10hu", &global_maxcallno) != 1) {
- ast_log(LOG_WARNING, "maxcallnumbers must be set to a valid number. %s is not valid at line %d\n", v->value, v->lineno);
- }
- } else if (!strcasecmp(v->name, "maxcallnumbers_nonvalidated")) {
- if (sscanf(v->value, "%10hu", &global_maxcallno_nonval) != 1) {
- ast_log(LOG_WARNING, "maxcallnumbers_nonvalidated must be set to a valid number. %s is not valid at line %d.\n", v->value, v->lineno);
- }
- } else if (!strcasecmp(v->name, "calltokenoptional")) {
- if (add_calltoken_ignore(v->value)) {
- ast_log(LOG_WARNING, "Invalid calltokenoptional address range - '%s' line %d\n", v->value, v->lineno);
- }
- } else if (!strcasecmp(v->name, "subscribe_network_change_event")) {
- if (ast_true(v->value)) {
- subscribe_network_change = 1;
- } else if (ast_false(v->value)) {
- subscribe_network_change = 0;
- } else {
- ast_log(LOG_WARNING, "subscribe_network_change_event value %s is not valid at line %d.\n", v->value, v->lineno);
- }
- } else if (!strcasecmp(v->name, "shrinkcallerid")) {
- if (ast_true(v->value)) {
- ast_set_flag64((&globalflags), IAX_SHRINKCALLERID);
- } else if (ast_false(v->value)) {
- ast_clear_flag64((&globalflags), IAX_SHRINKCALLERID);
- } else {
- ast_log(LOG_WARNING, "shrinkcallerid value %s is not valid at line %d.\n", v->value, v->lineno);
- }
- }/*else if (strcasecmp(v->name,"type")) */
- /* ast_log(LOG_WARNING, "Ignoring %s\n", v->name); */
- v = v->next;
- }
- if (subscribe_network_change) {
- network_change_event_subscribe();
- } else {
- network_change_event_unsubscribe();
- }
- if (defaultsockfd < 0) {
- if (!(ns = ast_netsock_bind(netsock, io, "0.0.0.0", portno, qos.tos, qos.cos, socket_read, NULL))) {
- ast_log(LOG_ERROR, "Unable to create network socket: %s\n", strerror(errno));
- } else {
- ast_verb(2, "Binding IAX2 to default address 0.0.0.0:%d\n", portno);
- defaultsockfd = ast_netsock_sockfd(ns);
- ast_netsock_unref(ns);
- }
- }
- if (reload) {
- ast_netsock_release(outsock);
- outsock = ast_netsock_list_alloc();
- if (!outsock) {
- ast_log(LOG_ERROR, "Could not allocate outsock list.\n");
- return -1;
- }
- ast_netsock_init(outsock);
- }
- if (min_reg_expire > max_reg_expire) {
- ast_log(LOG_WARNING, "Minimum registration interval of %d is more than maximum of %d, resetting minimum to %d\n",
- min_reg_expire, max_reg_expire, max_reg_expire);
- min_reg_expire = max_reg_expire;
- }
- iax2_capability = capability;
-
- if (ucfg) {
- struct ast_variable *gen;
- int genhasiax;
- int genregisteriax;
- const char *hasiax, *registeriax;
-
- genhasiax = ast_true(ast_variable_retrieve(ucfg, "general", "hasiax"));
- genregisteriax = ast_true(ast_variable_retrieve(ucfg, "general", "registeriax"));
- gen = ast_variable_browse(ucfg, "general");
- cat = ast_category_browse(ucfg, NULL);
- while (cat) {
- if (strcasecmp(cat, "general")) {
- hasiax = ast_variable_retrieve(ucfg, cat, "hasiax");
- registeriax = ast_variable_retrieve(ucfg, cat, "registeriax");
- if (ast_true(hasiax) || (!hasiax && genhasiax)) {
- /* Start with general parameters, then specific parameters, user and peer */
- user = build_user(cat, gen, ast_variable_browse(ucfg, cat), 0);
- if (user) {
- ao2_link(users, user);
- user = user_unref(user);
- }
- peer = build_peer(cat, gen, ast_variable_browse(ucfg, cat), 0);
- if (peer) {
- if (ast_test_flag64(peer, IAX_DYNAMIC))
- reg_source_db(peer);
- ao2_link(peers, peer);
- peer = peer_unref(peer);
- }
- }
- if (ast_true(registeriax) || (!registeriax && genregisteriax)) {
- char tmp[256];
- const char *host = ast_variable_retrieve(ucfg, cat, "host");
- const char *username = ast_variable_retrieve(ucfg, cat, "username");
- const char *secret = ast_variable_retrieve(ucfg, cat, "secret");
- if (!host)
- host = ast_variable_retrieve(ucfg, "general", "host");
- if (!username)
- username = ast_variable_retrieve(ucfg, "general", "username");
- if (!secret)
- secret = ast_variable_retrieve(ucfg, "general", "secret");
- if (!ast_strlen_zero(username) && !ast_strlen_zero(host)) {
- if (!ast_strlen_zero(secret))
- snprintf(tmp, sizeof(tmp), "%s:%s@%s", username, secret, host);
- else
- snprintf(tmp, sizeof(tmp), "%s@%s", username, host);
- iax2_register(tmp, 0);
- }
- }
- }
- cat = ast_category_browse(ucfg, cat);
- }
- ast_config_destroy(ucfg);
- }
-
- cat = ast_category_browse(cfg, NULL);
- while(cat) {
- if (strcasecmp(cat, "general")) {
- utype = ast_variable_retrieve(cfg, cat, "type");
- if (!strcasecmp(cat, "callnumberlimits")) {
- build_callno_limits(ast_variable_browse(cfg, cat));
- } else if (utype) {
- if (!strcasecmp(utype, "user") || !strcasecmp(utype, "friend")) {
- user = build_user(cat, ast_variable_browse(cfg, cat), NULL, 0);
- if (user) {
- ao2_link(users, user);
- user = user_unref(user);
- }
- }
- if (!strcasecmp(utype, "peer") || !strcasecmp(utype, "friend")) {
- peer = build_peer(cat, ast_variable_browse(cfg, cat), NULL, 0);
- if (peer) {
- if (ast_test_flag64(peer, IAX_DYNAMIC))
- reg_source_db(peer);
- ao2_link(peers, peer);
- peer = peer_unref(peer);
- }
- } else if (strcasecmp(utype, "user")) {
- ast_log(LOG_WARNING, "Unknown type '%s' for '%s' in %s\n", utype, cat, config_file);
- }
- } else
- ast_log(LOG_WARNING, "Section '%s' lacks type\n", cat);
- }
- cat = ast_category_browse(cfg, cat);
- }
- ast_config_destroy(cfg);
- return 1;
- }
- static void poke_all_peers(void)
- {
- struct ao2_iterator i;
- struct iax2_peer *peer;
- i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- iax2_poke_peer(peer, 0);
- peer_unref(peer);
- }
- ao2_iterator_destroy(&i);
- }
- static int reload_config(int forced_reload)
- {
- static const char config[] = "iax.conf";
- struct iax2_registry *reg;
- if (set_config(config, 1, forced_reload) > 0) {
- prune_peers();
- prune_users();
- ao2_callback(callno_limits, OBJ_NODATA | OBJ_UNLINK | OBJ_MULTIPLE, prune_addr_range_cb, NULL);
- ao2_callback(calltoken_ignores, OBJ_NODATA | OBJ_UNLINK | OBJ_MULTIPLE, prune_addr_range_cb, NULL);
- ao2_callback(peercnts, OBJ_NODATA, set_peercnt_limit_all_cb, NULL);
- trunk_timed = trunk_untimed = 0;
- trunk_nmaxmtu = trunk_maxmtu = 0;
- memset(&debugaddr, '\0', sizeof(debugaddr));
- AST_LIST_LOCK(®istrations);
- AST_LIST_TRAVERSE(®istrations, reg, entry)
- iax2_do_register(reg);
- AST_LIST_UNLOCK(®istrations);
- /* Qualify hosts, too */
- poke_all_peers();
- }
-
- reload_firmware(0);
- iax_provision_reload(1);
- ast_unload_realtime("iaxpeers");
- return 0;
- }
- static char *handle_cli_iax2_reload(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
- {
- switch (cmd) {
- case CLI_INIT:
- e->command = "iax2 reload";
- e->usage =
- "Usage: iax2 reload\n"
- " Reloads IAX configuration from iax.conf\n";
- return NULL;
- case CLI_GENERATE:
- return NULL;
- }
- reload_config(0);
- return CLI_SUCCESS;
- }
- static int reload(void)
- {
- return reload_config(0);
- }
- static int cache_get_callno_locked(const char *data)
- {
- struct sockaddr_in sin;
- int x;
- int callno;
- struct iax_ie_data ied;
- struct create_addr_info cai;
- struct parsed_dial_string pds;
- char *tmpstr;
- for (x = 0; x < ARRAY_LEN(iaxs); x++) {
- /* Look for an *exact match* call. Once a call is negotiated, it can only
- look up entries for a single context */
- if (!ast_mutex_trylock(&iaxsl[x])) {
- if (iaxs[x] && !strcasecmp(data, iaxs[x]->dproot))
- return x;
- ast_mutex_unlock(&iaxsl[x]);
- }
- }
- /* No match found, we need to create a new one */
- memset(&cai, 0, sizeof(cai));
- memset(&ied, 0, sizeof(ied));
- memset(&pds, 0, sizeof(pds));
- tmpstr = ast_strdupa(data);
- parse_dial_string(tmpstr, &pds);
- if (ast_strlen_zero(pds.peer)) {
- ast_log(LOG_WARNING, "No peer provided in the IAX2 dial string '%s'\n", data);
- return -1;
- }
- /* Populate our address from the given */
- if (create_addr(pds.peer, NULL, &sin, &cai))
- return -1;
- ast_debug(1, "peer: %s, username: %s, password: %s, context: %s\n",
- pds.peer, pds.username, pds.password, pds.context);
- callno = find_callno_locked(0, 0, &sin, NEW_FORCE, cai.sockfd, 0);
- if (callno < 1) {
- ast_log(LOG_WARNING, "Unable to create call\n");
- return -1;
- }
- ast_string_field_set(iaxs[callno], dproot, data);
- iaxs[callno]->capability = IAX_CAPABILITY_FULLBANDWIDTH;
- iax_ie_append_short(&ied, IAX_IE_VERSION, IAX_PROTO_VERSION);
- iax_ie_append_str(&ied, IAX_IE_CALLED_NUMBER, "TBD");
- /* the string format is slightly different from a standard dial string,
- because the context appears in the 'exten' position
- */
- if (pds.exten)
- iax_ie_append_str(&ied, IAX_IE_CALLED_CONTEXT, pds.exten);
- if (pds.username)
- iax_ie_append_str(&ied, IAX_IE_USERNAME, pds.username);
- iax_ie_append_int(&ied, IAX_IE_FORMAT, IAX_CAPABILITY_FULLBANDWIDTH);
- iax_ie_append_int(&ied, IAX_IE_CAPABILITY, IAX_CAPABILITY_FULLBANDWIDTH);
- /* Keep password handy */
- if (pds.password)
- ast_string_field_set(iaxs[callno], secret, pds.password);
- if (pds.key)
- ast_string_field_set(iaxs[callno], outkey, pds.key);
- /* Start the call going */
- add_empty_calltoken_ie(iaxs[callno], &ied); /* this _MUST_ be the last ie added */
- send_command(iaxs[callno], AST_FRAME_IAX, IAX_COMMAND_NEW, 0, ied.buf, ied.pos, -1);
- return callno;
- }
- static struct iax2_dpcache *find_cache(struct ast_channel *chan, const char *data, const char *context, const char *exten, int priority)
- {
- struct iax2_dpcache *dp = NULL;
- struct timeval now = ast_tvnow();
- int x, com[2], timeout, old = 0, outfd, doabort, callno;
- struct ast_channel *c = NULL;
- struct ast_frame *f = NULL;
- AST_LIST_TRAVERSE_SAFE_BEGIN(&dpcache, dp, cache_list) {
- if (ast_tvcmp(now, dp->expiry) > 0) {
- AST_LIST_REMOVE_CURRENT(cache_list);
- if ((dp->flags & CACHE_FLAG_PENDING) || dp->callno)
- ast_log(LOG_WARNING, "DP still has peer field or pending or callno (flags = %d, peer = blah, callno = %d)\n", dp->flags, dp->callno);
- else
- ast_free(dp);
- continue;
- }
- if (!strcmp(dp->peercontext, data) && !strcmp(dp->exten, exten))
- break;
- }
- AST_LIST_TRAVERSE_SAFE_END;
- if (!dp) {
- /* No matching entry. Create a new one. */
- /* First, can we make a callno? */
- if ((callno = cache_get_callno_locked(data)) < 0) {
- ast_log(LOG_WARNING, "Unable to generate call for '%s'\n", data);
- return NULL;
- }
- if (!(dp = ast_calloc(1, sizeof(*dp)))) {
- ast_mutex_unlock(&iaxsl[callno]);
- return NULL;
- }
- ast_copy_string(dp->peercontext, data, sizeof(dp->peercontext));
- ast_copy_string(dp->exten, exten, sizeof(dp->exten));
- dp->expiry = ast_tvnow();
- dp->orig = dp->expiry;
- /* Expires in 30 mins by default */
- dp->expiry.tv_sec += iaxdefaultdpcache;
- dp->flags = CACHE_FLAG_PENDING;
- for (x = 0; x < ARRAY_LEN(dp->waiters); x++)
- dp->waiters[x] = -1;
- /* Insert into the lists */
- AST_LIST_INSERT_TAIL(&dpcache, dp, cache_list);
- AST_LIST_INSERT_TAIL(&iaxs[callno]->dpentries, dp, peer_list);
- /* Send the request if we're already up */
- if (ast_test_flag(&iaxs[callno]->state, IAX_STATE_STARTED))
- iax2_dprequest(dp, callno);
- ast_mutex_unlock(&iaxsl[callno]);
- }
- /* By here we must have a dp */
- if (dp->flags & CACHE_FLAG_PENDING) {
- struct timeval start;
- int ms;
- /* Okay, here it starts to get nasty. We need a pipe now to wait
- for a reply to come back so long as it's pending */
- for (x = 0; x < ARRAY_LEN(dp->waiters); x++) {
- /* Find an empty slot */
- if (dp->waiters[x] < 0)
- break;
- }
- if (x >= ARRAY_LEN(dp->waiters)) {
- ast_log(LOG_WARNING, "No more waiter positions available\n");
- return NULL;
- }
- if (pipe(com)) {
- ast_log(LOG_WARNING, "Unable to create pipe for comm\n");
- return NULL;
- }
- dp->waiters[x] = com[1];
- /* Okay, now we wait */
- timeout = iaxdefaulttimeout * 1000;
- /* Temporarily unlock */
- AST_LIST_UNLOCK(&dpcache);
- /* Defer any dtmf */
- if (chan)
- old = ast_channel_defer_dtmf(chan);
- doabort = 0;
- start = ast_tvnow();
- while ((ms = ast_remaining_ms(start, timeout))) {
- c = ast_waitfor_nandfds(&chan, chan ? 1 : 0, &com[0], 1, NULL, &outfd, &ms);
- if (outfd > -1)
- break;
- if (!c)
- continue;
- if (!(f = ast_read(c))) {
- doabort = 1;
- break;
- }
- ast_frfree(f);
- }
- if (!ms) {
- ast_log(LOG_WARNING, "Timeout waiting for %s exten %s\n", data, exten);
- }
- AST_LIST_LOCK(&dpcache);
- dp->waiters[x] = -1;
- close(com[1]);
- close(com[0]);
- if (doabort) {
- /* Don't interpret anything, just abort. Not sure what th epoint
- of undeferring dtmf on a hung up channel is but hey whatever */
- if (!old && chan)
- ast_channel_undefer_dtmf(chan);
- return NULL;
- }
- if (!(dp->flags & CACHE_FLAG_TIMEOUT)) {
- /* Now to do non-independent analysis the results of our wait */
- if (dp->flags & CACHE_FLAG_PENDING) {
- /* Still pending... It's a timeout. Wake everybody up. Consider it no longer
- pending. Don't let it take as long to timeout. */
- dp->flags &= ~CACHE_FLAG_PENDING;
- dp->flags |= CACHE_FLAG_TIMEOUT;
- /* Expire after only 60 seconds now. This is designed to help reduce backlog in heavily loaded
- systems without leaving it unavailable once the server comes back online */
- dp->expiry.tv_sec = dp->orig.tv_sec + 60;
- for (x = 0; x < ARRAY_LEN(dp->waiters); x++) {
- if (dp->waiters[x] > -1) {
- if (write(dp->waiters[x], "asdf", 4) < 0) {
- ast_log(LOG_WARNING, "write() failed: %s\n", strerror(errno));
- }
- }
- }
- }
- }
- /* Our caller will obtain the rest */
- if (!old && chan)
- ast_channel_undefer_dtmf(chan);
- }
- return dp;
- }
- /*! \brief Part of the IAX2 switch interface */
- static int iax2_exists(struct ast_channel *chan, const char *context, const char *exten, int priority, const char *callerid, const char *data)
- {
- int res = 0;
- struct iax2_dpcache *dp = NULL;
- #if 0
- ast_log(LOG_NOTICE, "iax2_exists: con: %s, exten: %s, pri: %d, cid: %s, data: %s\n", context, exten, priority, callerid ? callerid : "<unknown>", data);
- #endif
- if ((priority != 1) && (priority != 2))
- return 0;
- AST_LIST_LOCK(&dpcache);
- if ((dp = find_cache(chan, data, context, exten, priority))) {
- if (dp->flags & CACHE_FLAG_EXISTS)
- res = 1;
- } else {
- ast_log(LOG_WARNING, "Unable to make DP cache\n");
- }
- AST_LIST_UNLOCK(&dpcache);
- return res;
- }
- /*! \brief part of the IAX2 dial plan switch interface */
- static int iax2_canmatch(struct ast_channel *chan, const char *context, const char *exten, int priority, const char *callerid, const char *data)
- {
- int res = 0;
- struct iax2_dpcache *dp = NULL;
- #if 0
- ast_log(LOG_NOTICE, "iax2_canmatch: con: %s, exten: %s, pri: %d, cid: %s, data: %s\n", context, exten, priority, callerid ? callerid : "<unknown>", data);
- #endif
- if ((priority != 1) && (priority != 2))
- return 0;
- AST_LIST_LOCK(&dpcache);
- if ((dp = find_cache(chan, data, context, exten, priority))) {
- if (dp->flags & CACHE_FLAG_CANEXIST)
- res = 1;
- } else {
- ast_log(LOG_WARNING, "Unable to make DP cache\n");
- }
- AST_LIST_UNLOCK(&dpcache);
- return res;
- }
- /*! \brief Part of the IAX2 Switch interface */
- static int iax2_matchmore(struct ast_channel *chan, const char *context, const char *exten, int priority, const char *callerid, const char *data)
- {
- int res = 0;
- struct iax2_dpcache *dp = NULL;
- #if 0
- ast_log(LOG_NOTICE, "iax2_matchmore: con: %s, exten: %s, pri: %d, cid: %s, data: %s\n", context, exten, priority, callerid ? callerid : "<unknown>", data);
- #endif
- if ((priority != 1) && (priority != 2))
- return 0;
- AST_LIST_LOCK(&dpcache);
- if ((dp = find_cache(chan, data, context, exten, priority))) {
- if (dp->flags & CACHE_FLAG_MATCHMORE)
- res = 1;
- } else {
- ast_log(LOG_WARNING, "Unable to make DP cache\n");
- }
- AST_LIST_UNLOCK(&dpcache);
- return res;
- }
- /*! \brief Execute IAX2 dialplan switch */
- static int iax2_exec(struct ast_channel *chan, const char *context, const char *exten, int priority, const char *callerid, const char *data)
- {
- char odata[256];
- char req[256];
- char *ncontext;
- struct iax2_dpcache *dp = NULL;
- struct ast_app *dial = NULL;
- #if 0
- ast_log(LOG_NOTICE, "iax2_exec: con: %s, exten: %s, pri: %d, cid: %s, data: %s, newstack: %d\n", context, exten, priority, callerid ? callerid : "<unknown>", data, newstack);
- #endif
- if (priority == 2) {
- /* Indicate status, can be overridden in dialplan */
- const char *dialstatus = pbx_builtin_getvar_helper(chan, "DIALSTATUS");
- if (dialstatus) {
- dial = pbx_findapp(dialstatus);
- if (dial)
- pbx_exec(chan, dial, "");
- }
- return -1;
- } else if (priority != 1)
- return -1;
- AST_LIST_LOCK(&dpcache);
- if ((dp = find_cache(chan, data, context, exten, priority))) {
- if (dp->flags & CACHE_FLAG_EXISTS) {
- ast_copy_string(odata, data, sizeof(odata));
- ncontext = strchr(odata, '/');
- if (ncontext) {
- *ncontext = '\0';
- ncontext++;
- snprintf(req, sizeof(req), "IAX2/%s/%s@%s", odata, exten, ncontext);
- } else {
- snprintf(req, sizeof(req), "IAX2/%s/%s", odata, exten);
- }
- ast_verb(3, "Executing Dial('%s')\n", req);
- } else {
- AST_LIST_UNLOCK(&dpcache);
- ast_log(LOG_WARNING, "Can't execute nonexistent extension '%s[@%s]' in data '%s'\n", exten, context, data);
- return -1;
- }
- }
- AST_LIST_UNLOCK(&dpcache);
- if ((dial = pbx_findapp("Dial")))
- return pbx_exec(chan, dial, req);
- else
- ast_log(LOG_WARNING, "No dial application registered\n");
- return -1;
- }
- static int function_iaxpeer(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
- {
- struct iax2_peer *peer;
- char *peername, *colname;
- peername = ast_strdupa(data);
- /* if our channel, return the IP address of the endpoint of current channel */
- if (!strcmp(peername,"CURRENTCHANNEL")) {
- unsigned short callno;
- if (!chan || ast_channel_tech(chan) != &iax2_tech) {
- return -1;
- }
- callno = PTR_TO_CALLNO(ast_channel_tech_pvt(chan));
- ast_copy_string(buf, iaxs[callno]->addr.sin_addr.s_addr ? ast_inet_ntoa(iaxs[callno]->addr.sin_addr) : "", len);
- return 0;
- }
- if ((colname = strchr(peername, ',')))
- *colname++ = '\0';
- else
- colname = "ip";
- if (!(peer = find_peer(peername, 1)))
- return -1;
- if (!strcasecmp(colname, "ip")) {
- ast_copy_string(buf, ast_sockaddr_stringify_addr(&peer->addr), len);
- } else if (!strcasecmp(colname, "status")) {
- peer_status(peer, buf, len);
- } else if (!strcasecmp(colname, "mailbox")) {
- ast_copy_string(buf, peer->mailbox, len);
- } else if (!strcasecmp(colname, "context")) {
- ast_copy_string(buf, peer->context, len);
- } else if (!strcasecmp(colname, "expire")) {
- snprintf(buf, len, "%d", peer->expire);
- } else if (!strcasecmp(colname, "dynamic")) {
- ast_copy_string(buf, (ast_test_flag64(peer, IAX_DYNAMIC) ? "yes" : "no"), len);
- } else if (!strcasecmp(colname, "callerid_name")) {
- ast_copy_string(buf, peer->cid_name, len);
- } else if (!strcasecmp(colname, "callerid_num")) {
- ast_copy_string(buf, peer->cid_num, len);
- } else if (!strcasecmp(colname, "codecs")) {
- iax2_getformatname_multiple(buf, len -1, peer->capability);
- } else if (!strncasecmp(colname, "codec[", 6)) {
- char *codecnum, *ptr;
- struct ast_format tmpfmt;
- /* skip over "codec" to the '[' */
- codecnum = colname + 5;
- *codecnum = '\0';
- codecnum++;
- if ((ptr = strchr(codecnum, ']'))) {
- *ptr = '\0';
- }
- if((ast_codec_pref_index(&peer->prefs, atoi(codecnum), &tmpfmt))) {
- ast_copy_string(buf, ast_getformatname(&tmpfmt), len);
- } else {
- buf[0] = '\0';
- }
- } else {
- buf[0] = '\0';
- }
- peer_unref(peer);
- return 0;
- }
- static struct ast_custom_function iaxpeer_function = {
- .name = "IAXPEER",
- .read = function_iaxpeer,
- };
- static int acf_channel_read(struct ast_channel *chan, const char *funcname, char *args, char *buf, size_t buflen)
- {
- struct chan_iax2_pvt *pvt;
- unsigned int callno;
- int res = 0;
- if (!chan || ast_channel_tech(chan) != &iax2_tech) {
- ast_log(LOG_ERROR, "This function requires a valid IAX2 channel\n");
- return -1;
- }
- callno = PTR_TO_CALLNO(ast_channel_tech_pvt(chan));
- ast_mutex_lock(&iaxsl[callno]);
- if (!(pvt = iaxs[callno])) {
- ast_mutex_unlock(&iaxsl[callno]);
- return -1;
- }
- if (!strcasecmp(args, "osptoken")) {
- ast_copy_string(buf, pvt->osptoken, buflen);
- } else if (!strcasecmp(args, "peerip")) {
- ast_copy_string(buf, pvt->addr.sin_addr.s_addr ? ast_inet_ntoa(pvt->addr.sin_addr) : "", buflen);
- } else if (!strcasecmp(args, "peername")) {
- ast_copy_string(buf, pvt->username, buflen);
- } else if (!strcasecmp(args, "secure_signaling") || !strcasecmp(args, "secure_media")) {
- snprintf(buf, buflen, "%s", IAX_CALLENCRYPTED(pvt) ? "1" : "");
- } else {
- res = -1;
- }
- ast_mutex_unlock(&iaxsl[callno]);
- return res;
- }
- /*! \brief Part of the device state notification system ---*/
- static int iax2_devicestate(const char *data)
- {
- struct parsed_dial_string pds;
- char *tmp = ast_strdupa(data);
- struct iax2_peer *p;
- int res = AST_DEVICE_INVALID;
- memset(&pds, 0, sizeof(pds));
- parse_dial_string(tmp, &pds);
- if (ast_strlen_zero(pds.peer)) {
- ast_log(LOG_WARNING, "No peer provided in the IAX2 dial string '%s'\n", data);
- return res;
- }
- ast_debug(3, "Checking device state for device %s\n", pds.peer);
- /* SLD: FIXME: second call to find_peer during registration */
- if (!(p = find_peer(pds.peer, 1)))
- return res;
- res = AST_DEVICE_UNAVAILABLE;
- ast_debug(3, "Found peer. What's device state of %s? addr=%u, defaddr=%u maxms=%d, lastms=%d\n",
- pds.peer, ast_sockaddr_ipv4(&p->addr), p->defaddr.sin_addr.s_addr, p->maxms, p->lastms);
- if ((ast_sockaddr_ipv4(&p->addr) || p->defaddr.sin_addr.s_addr) &&
- (!p->maxms || ((p->lastms > -1) && (p->historicms <= p->maxms)))) {
- /* Peer is registered, or have default IP address
- and a valid registration */
- if (p->historicms == 0 || p->historicms <= p->maxms)
- /* let the core figure out whether it is in use or not */
- res = AST_DEVICE_UNKNOWN;
- }
- peer_unref(p);
- return res;
- }
- static struct ast_switch iax2_switch =
- {
- .name = "IAX2",
- .description = "IAX Remote Dialplan Switch",
- .exists = iax2_exists,
- .canmatch = iax2_canmatch,
- .exec = iax2_exec,
- .matchmore = iax2_matchmore,
- };
- /*
- { { "iax2", "show", "cache", NULL },
- iax2_show_cache, "Display IAX cached dialplan",
- show_cache_usage },
- { { "iax2", "show", "channels", NULL },
- iax2_show_channels, "List active IAX channels",
- show_channels_usage },
- { { "iax2", "show", "firmware", NULL },
- iax2_show_firmware, "List available IAX firmwares",
- show_firmware_usage },
- { { "iax2", "show", "netstats", NULL },
- iax2_show_netstats, "List active IAX channel netstats",
- show_netstats_usage },
- { { "iax2", "show", "peers", NULL },
- iax2_show_peers, "List defined IAX peers",
- show_peers_usage },
- { { "iax2", "show", "registry", NULL },
- iax2_show_registry, "Display IAX registration status",
- show_reg_usage },
- { { "iax2", "show", "stats", NULL },
- iax2_show_stats, "Display IAX statistics",
- show_stats_usage },
- { { "iax2", "show", "threads", NULL },
- iax2_show_threads, "Display IAX helper thread info",
- show_threads_usage },
- { { "iax2", "unregister", NULL },
- iax2_unregister, "Unregister (force expiration) an IAX2 peer from the registry",
- unregister_usage, complete_iax2_unregister },
- { { "iax2", "set", "mtu", NULL },
- iax2_set_mtu, "Set the IAX systemwide trunking MTU",
- set_mtu_usage, NULL, NULL },
- { { "iax2", "show", "users", NULL },
- iax2_show_users, "List defined IAX users",
- show_users_usage },
- { { "iax2", "prune", "realtime", NULL },
- iax2_prune_realtime, "Prune a cached realtime lookup",
- prune_realtime_usage, complete_iax2_show_peer },
- { { "iax2", "reload", NULL },
- iax2_reload, "Reload IAX configuration",
- iax2_reload_usage },
- { { "iax2", "show", "peer", NULL },
- iax2_show_peer, "Show details on specific IAX peer",
- show_peer_usage, complete_iax2_show_peer },
- { { "iax2", "set", "debug", NULL },
- iax2_do_debug, "Enable IAX debugging",
- debug_usage },
- { { "iax2", "set", "debug", "trunk", NULL },
- iax2_do_trunk_debug, "Enable IAX trunk debugging",
- debug_trunk_usage },
- { { "iax2", "set", "debug", "jb", NULL },
- iax2_do_jb_debug, "Enable IAX jitterbuffer debugging",
- debug_jb_usage },
- { { "iax2", "set", "debug", "off", NULL },
- iax2_no_debug, "Disable IAX debugging",
- no_debug_usage },
- { { "iax2", "set", "debug", "trunk", "off", NULL },
- iax2_no_trunk_debug, "Disable IAX trunk debugging",
- no_debug_trunk_usage },
- { { "iax2", "set", "debug", "jb", "off", NULL },
- iax2_no_jb_debug, "Disable IAX jitterbuffer debugging",
- no_debug_jb_usage },
- { { "iax2", "test", "losspct", NULL },
- iax2_test_losspct, "Set IAX2 incoming frame loss percentage",
- iax2_test_losspct_usage },
- { { "iax2", "provision", NULL },
- iax2_prov_cmd, "Provision an IAX device",
- show_prov_usage, iax2_prov_complete_template_3rd },
- #ifdef IAXTESTS
- { { "iax2", "test", "late", NULL },
- iax2_test_late, "Test the receipt of a late frame",
- iax2_test_late_usage },
- { { "iax2", "test", "resync", NULL },
- iax2_test_resync, "Test a resync in received timestamps",
- iax2_test_resync_usage },
- { { "iax2", "test", "jitter", NULL },
- iax2_test_jitter, "Simulates jitter for testing",
- iax2_test_jitter_usage },
- #endif
- */
- static struct ast_cli_entry cli_iax2[] = {
- AST_CLI_DEFINE(handle_cli_iax2_provision, "Provision an IAX device"),
- AST_CLI_DEFINE(handle_cli_iax2_prune_realtime, "Prune a cached realtime lookup"),
- AST_CLI_DEFINE(handle_cli_iax2_reload, "Reload IAX configuration"),
- AST_CLI_DEFINE(handle_cli_iax2_set_mtu, "Set the IAX systemwide trunking MTU"),
- AST_CLI_DEFINE(handle_cli_iax2_set_debug, "Enable/Disable IAX debugging"),
- AST_CLI_DEFINE(handle_cli_iax2_set_debug_trunk, "Enable/Disable IAX trunk debugging"),
- AST_CLI_DEFINE(handle_cli_iax2_set_debug_jb, "Enable/Disable IAX jitterbuffer debugging"),
- AST_CLI_DEFINE(handle_cli_iax2_show_cache, "Display IAX cached dialplan"),
- AST_CLI_DEFINE(handle_cli_iax2_show_channels, "List active IAX channels"),
- AST_CLI_DEFINE(handle_cli_iax2_show_firmware, "List available IAX firmware"),
- AST_CLI_DEFINE(handle_cli_iax2_show_netstats, "List active IAX channel netstats"),
- AST_CLI_DEFINE(handle_cli_iax2_show_peer, "Show details on specific IAX peer"),
- AST_CLI_DEFINE(handle_cli_iax2_show_peers, "List defined IAX peers"),
- AST_CLI_DEFINE(handle_cli_iax2_show_registry, "Display IAX registration status"),
- AST_CLI_DEFINE(handle_cli_iax2_show_stats, "Display IAX statistics"),
- AST_CLI_DEFINE(handle_cli_iax2_show_threads, "Display IAX helper thread info"),
- AST_CLI_DEFINE(handle_cli_iax2_show_users, "List defined IAX users"),
- AST_CLI_DEFINE(handle_cli_iax2_test_losspct, "Set IAX2 incoming frame loss percentage"),
- AST_CLI_DEFINE(handle_cli_iax2_unregister, "Unregister (force expiration) an IAX2 peer from the registry"),
- AST_CLI_DEFINE(handle_cli_iax2_show_callno_limits, "Show current entries in IP call number limit table"),
- #ifdef IAXTESTS
- AST_CLI_DEFINE(handle_cli_iax2_test_jitter, "Simulates jitter for testing"),
- AST_CLI_DEFINE(handle_cli_iax2_test_late, "Test the receipt of a late frame"),
- AST_CLI_DEFINE(handle_cli_iax2_test_resync, "Test a resync in received timestamps"),
- #endif /* IAXTESTS */
- };
- #ifdef TEST_FRAMEWORK
- AST_TEST_DEFINE(test_iax2_peers_get)
- {
- struct ast_data_query query = {
- .path = "/asterisk/channel/iax2/peers",
- .search = "peers/peer/name=test_peer_data_provider"
- };
- struct ast_data *node;
- struct iax2_peer *peer;
- switch (cmd) {
- case TEST_INIT:
- info->name = "iax2_peers_get_data_test";
- info->category = "/main/data/iax2/peers/";
- info->summary = "IAX2 peers data providers unit test";
- info->description =
- "Tests whether the IAX2 peers data provider implementation works as expected.";
- return AST_TEST_NOT_RUN;
- case TEST_EXECUTE:
- break;
- }
- /* build a test peer */
- peer = build_peer("test_peer_data_provider", NULL, NULL, 0);
- if (!peer) {
- return AST_TEST_FAIL;
- }
- peer->expiry= 1010;
- ao2_link(peers, peer);
- node = ast_data_get(&query);
- if (!node) {
- ao2_unlink(peers, peer);
- peer_unref(peer);
- return AST_TEST_FAIL;
- }
- /* check returned data node. */
- if (strcmp(ast_data_retrieve_string(node, "peer/name"), "test_peer_data_provider")) {
- ao2_unlink(peers, peer);
- peer_unref(peer);
- ast_data_free(node);
- return AST_TEST_FAIL;
- }
- if (ast_data_retrieve_int(node, "peer/expiry") != 1010) {
- ao2_unlink(peers, peer);
- peer_unref(peer);
- ast_data_free(node);
- return AST_TEST_FAIL;
- }
- /* release resources */
- ast_data_free(node);
- ao2_unlink(peers, peer);
- peer_unref(peer);
- return AST_TEST_PASS;
- }
- AST_TEST_DEFINE(test_iax2_users_get)
- {
- struct ast_data_query query = {
- .path = "/asterisk/channel/iax2/users",
- .search = "users/user/name=test_user_data_provider"
- };
- struct ast_data *node;
- struct iax2_user *user;
- switch (cmd) {
- case TEST_INIT:
- info->name = "iax2_users_get_data_test";
- info->category = "/main/data/iax2/users/";
- info->summary = "IAX2 users data providers unit test";
- info->description =
- "Tests whether the IAX2 users data provider implementation works as expected.";
- return AST_TEST_NOT_RUN;
- case TEST_EXECUTE:
- break;
- }
- user = build_user("test_user_data_provider", NULL, NULL, 0);
- if (!user) {
- ast_test_status_update(test, "Failed to build a test user\n");
- return AST_TEST_FAIL;
- }
- user->amaflags = 1010;
- ao2_link(users, user);
- node = ast_data_get(&query);
- if (!node) {
- ast_test_status_update(test, "The data query to find our test user failed\n");
- ao2_unlink(users, user);
- user_unref(user);
- return AST_TEST_FAIL;
- }
- if (strcmp(ast_data_retrieve_string(node, "user/name"), "test_user_data_provider")) {
- ast_test_status_update(test, "Our data results did not return the test user created in the previous step.\n");
- ao2_unlink(users, user);
- user_unref(user);
- ast_data_free(node);
- return AST_TEST_FAIL;
- }
- if (ast_data_retrieve_int(node, "user/amaflags/value") != 1010) {
- ast_test_status_update(test, "The amaflags field in our test user was '%d' not the expected value '1010'\n", ast_data_retrieve_int(node, "user/amaflags/value"));
- ao2_unlink(users, user);
- user_unref(user);
- ast_data_free(node);
- return AST_TEST_FAIL;
- }
- ast_data_free(node);
- ao2_unlink(users, user);
- user_unref(user);
- return AST_TEST_PASS;
- }
- #endif
- static void cleanup_thread_list(void *head)
- {
- AST_LIST_HEAD(iax2_thread_list, iax2_thread);
- struct iax2_thread_list *list_head = head;
- struct iax2_thread *thread;
- AST_LIST_LOCK(list_head);
- while ((thread = AST_LIST_REMOVE_HEAD(list_head, list))) {
- pthread_t thread_id = thread->threadid;
- thread->stop = 1;
- signal_condition(&thread->lock, &thread->cond);
- AST_LIST_UNLOCK(list_head);
- pthread_join(thread_id, NULL);
- AST_LIST_LOCK(list_head);
- }
- AST_LIST_UNLOCK(list_head);
- }
- static int __unload_module(void)
- {
- struct ast_context *con;
- int x;
- network_change_event_unsubscribe();
- acl_change_event_unsubscribe();
- ast_manager_unregister("IAXpeers");
- ast_manager_unregister("IAXpeerlist");
- ast_manager_unregister("IAXnetstats");
- ast_manager_unregister("IAXregistry");
- ast_unregister_application(papp);
- ast_cli_unregister_multiple(cli_iax2, ARRAY_LEN(cli_iax2));
- ast_unregister_switch(&iax2_switch);
- ast_channel_unregister(&iax2_tech);
- if (netthreadid != AST_PTHREADT_NULL) {
- pthread_cancel(netthreadid);
- pthread_kill(netthreadid, SIGURG);
- pthread_join(netthreadid, NULL);
- }
- for (x = 0; x < ARRAY_LEN(iaxs); x++) {
- if (iaxs[x]) {
- iax2_destroy(x);
- }
- }
- /* Call for all threads to halt */
- cleanup_thread_list(&active_list);
- cleanup_thread_list(&dynamic_list);
- cleanup_thread_list(&idle_list);
- ast_netsock_release(netsock);
- ast_netsock_release(outsock);
- for (x = 0; x < ARRAY_LEN(iaxs); x++) {
- if (iaxs[x]) {
- iax2_destroy(x);
- }
- }
- ast_manager_unregister( "IAXpeers" );
- ast_manager_unregister( "IAXpeerlist" );
- ast_manager_unregister( "IAXnetstats" );
- ast_manager_unregister( "IAXregistry" );
- ast_unregister_application(papp);
- #ifdef TEST_FRAMEWORK
- AST_TEST_UNREGISTER(test_iax2_peers_get);
- AST_TEST_UNREGISTER(test_iax2_users_get);
- #endif
- ast_data_unregister(NULL);
- ast_cli_unregister_multiple(cli_iax2, ARRAY_LEN(cli_iax2));
- ast_unregister_switch(&iax2_switch);
- ast_channel_unregister(&iax2_tech);
- delete_users();
- iax_provision_unload();
- reload_firmware(1);
- for (x = 0; x < ARRAY_LEN(iaxsl); x++) {
- ast_mutex_destroy(&iaxsl[x]);
- }
- ao2_ref(peers, -1);
- ao2_ref(users, -1);
- ao2_ref(iax_peercallno_pvts, -1);
- ao2_ref(iax_transfercallno_pvts, -1);
- ao2_ref(callno_limits, -1);
- ao2_ref(calltoken_ignores, -1);
- if (timer) {
- ast_timer_close(timer);
- timer = NULL;
- }
- transmit_processor = ast_taskprocessor_unreference(transmit_processor);
- ast_sched_clean_by_callback(sched, peercnt_remove_cb, peercnt_remove_cb);
- ast_sched_clean_by_callback(sched, replace_callno, replace_callno);
- ast_sched_context_destroy(sched);
- sched = NULL;
- ao2_ref(peercnts, -1);
- ao2_ref(callno_pool, -1);
- ao2_ref(callno_pool_trunk, -1);
- con = ast_context_find(regcontext);
- if (con)
- ast_context_destroy(con, "IAX2");
- ast_unload_realtime("iaxpeers");
- iax2_tech.capabilities = ast_format_cap_destroy(iax2_tech.capabilities);
- return 0;
- }
- static int unload_module(void)
- {
- ast_custom_function_unregister(&iaxpeer_function);
- ast_custom_function_unregister(&iaxvar_function);
- return __unload_module();
- }
- static int peer_set_sock_cb(void *obj, void *arg, int flags)
- {
- struct iax2_peer *peer = obj;
- if (peer->sockfd < 0)
- peer->sockfd = defaultsockfd;
- return 0;
- }
- static int pvt_hash_cb(const void *obj, const int flags)
- {
- const struct chan_iax2_pvt *pvt = obj;
- return pvt->peercallno;
- }
- static int pvt_cmp_cb(void *obj, void *arg, int flags)
- {
- struct chan_iax2_pvt *pvt = obj, *pvt2 = arg;
- /* The frames_received field is used to hold whether we're matching
- * against a full frame or not ... */
- return match(&pvt2->addr, pvt2->peercallno, pvt2->callno, pvt,
- pvt2->frames_received) ? CMP_MATCH | CMP_STOP : 0;
- }
- static int transfercallno_pvt_hash_cb(const void *obj, const int flags)
- {
- const struct chan_iax2_pvt *pvt = obj;
- return pvt->transfercallno;
- }
- static int transfercallno_pvt_cmp_cb(void *obj, void *arg, int flags)
- {
- struct chan_iax2_pvt *pvt = obj, *pvt2 = arg;
- /* The frames_received field is used to hold whether we're matching
- * against a full frame or not ... */
- return match(&pvt2->transfer, pvt2->transfercallno, pvt2->callno, pvt,
- pvt2->frames_received) ? CMP_MATCH | CMP_STOP : 0;
- }
- static int load_objects(void)
- {
- peers = users = iax_peercallno_pvts = iax_transfercallno_pvts = NULL;
- peercnts = callno_limits = calltoken_ignores = callno_pool = callno_pool_trunk = NULL;
- if (!(peers = ao2_container_alloc(MAX_PEER_BUCKETS, peer_hash_cb, peer_cmp_cb))) {
- goto container_fail;
- } else if (!(users = ao2_container_alloc(MAX_USER_BUCKETS, user_hash_cb, user_cmp_cb))) {
- goto container_fail;
- } else if (!(iax_peercallno_pvts = ao2_container_alloc(IAX_MAX_CALLS, pvt_hash_cb, pvt_cmp_cb))) {
- goto container_fail;
- } else if (!(iax_transfercallno_pvts = ao2_container_alloc(IAX_MAX_CALLS, transfercallno_pvt_hash_cb, transfercallno_pvt_cmp_cb))) {
- goto container_fail;
- } else if (!(peercnts = ao2_container_alloc(MAX_PEER_BUCKETS, peercnt_hash_cb, peercnt_cmp_cb))) {
- goto container_fail;
- } else if (!(callno_limits = ao2_container_alloc(MAX_PEER_BUCKETS, addr_range_hash_cb, addr_range_cmp_cb))) {
- goto container_fail;
- } else if (!(calltoken_ignores = ao2_container_alloc(MAX_PEER_BUCKETS, addr_range_hash_cb, addr_range_cmp_cb))) {
- goto container_fail;
- } else if (create_callno_pools()) {
- goto container_fail;
- } else if (!(transmit_processor = ast_taskprocessor_get("iax2_transmit", TPS_REF_DEFAULT))) {
- goto container_fail;
- }
- return 0;
- container_fail:
- if (peers) {
- ao2_ref(peers, -1);
- }
- if (users) {
- ao2_ref(users, -1);
- }
- if (iax_peercallno_pvts) {
- ao2_ref(iax_peercallno_pvts, -1);
- }
- if (iax_transfercallno_pvts) {
- ao2_ref(iax_transfercallno_pvts, -1);
- }
- if (peercnts) {
- ao2_ref(peercnts, -1);
- }
- if (callno_limits) {
- ao2_ref(callno_limits, -1);
- }
- if (calltoken_ignores) {
- ao2_ref(calltoken_ignores, -1);
- }
- if (callno_pool) {
- ao2_ref(callno_pool, -1);
- }
- if (callno_pool_trunk) {
- ao2_ref(callno_pool_trunk, -1);
- }
- return AST_MODULE_LOAD_FAILURE;
- }
- #define DATA_EXPORT_IAX2_PEER(MEMBER) \
- MEMBER(iax2_peer, name, AST_DATA_STRING) \
- MEMBER(iax2_peer, username, AST_DATA_STRING) \
- MEMBER(iax2_peer, secret, AST_DATA_PASSWORD) \
- MEMBER(iax2_peer, dbsecret, AST_DATA_PASSWORD) \
- MEMBER(iax2_peer, outkey, AST_DATA_STRING) \
- MEMBER(iax2_peer, regexten, AST_DATA_STRING) \
- MEMBER(iax2_peer, context, AST_DATA_STRING) \
- MEMBER(iax2_peer, peercontext, AST_DATA_STRING) \
- MEMBER(iax2_peer, mailbox, AST_DATA_STRING) \
- MEMBER(iax2_peer, mohinterpret, AST_DATA_STRING) \
- MEMBER(iax2_peer, mohsuggest, AST_DATA_STRING) \
- MEMBER(iax2_peer, inkeys, AST_DATA_STRING) \
- MEMBER(iax2_peer, cid_num, AST_DATA_STRING) \
- MEMBER(iax2_peer, cid_name, AST_DATA_STRING) \
- MEMBER(iax2_peer, zonetag, AST_DATA_STRING) \
- MEMBER(iax2_peer, parkinglot, AST_DATA_STRING) \
- MEMBER(iax2_peer, expiry, AST_DATA_SECONDS) \
- MEMBER(iax2_peer, callno, AST_DATA_INTEGER) \
- MEMBER(iax2_peer, lastms, AST_DATA_MILLISECONDS) \
- MEMBER(iax2_peer, maxms, AST_DATA_MILLISECONDS) \
- MEMBER(iax2_peer, pokefreqok, AST_DATA_MILLISECONDS) \
- MEMBER(iax2_peer, pokefreqnotok, AST_DATA_MILLISECONDS) \
- MEMBER(iax2_peer, historicms, AST_DATA_INTEGER) \
- MEMBER(iax2_peer, smoothing, AST_DATA_BOOLEAN) \
- MEMBER(iax2_peer, maxcallno, AST_DATA_INTEGER)
- AST_DATA_STRUCTURE(iax2_peer, DATA_EXPORT_IAX2_PEER);
- static int peers_data_provider_get(const struct ast_data_search *search,
- struct ast_data *data_root)
- {
- struct ast_data *data_peer;
- struct iax2_peer *peer;
- struct ao2_iterator i;
- char status[20];
- struct ast_str *encmethods = ast_str_alloca(256);
- i = ao2_iterator_init(peers, 0);
- while ((peer = ao2_iterator_next(&i))) {
- data_peer = ast_data_add_node(data_root, "peer");
- if (!data_peer) {
- peer_unref(peer);
- continue;
- }
- ast_data_add_structure(iax2_peer, data_peer, peer);
- iax2_data_add_codecs(data_peer, "codecs", peer->capability);
- peer_status(peer, status, sizeof(status));
- ast_data_add_str(data_peer, "status", status);
- ast_data_add_str(data_peer, "host", ast_sockaddr_stringify_host(&peer->addr));
- ast_data_add_str(data_peer, "mask", ast_inet_ntoa(peer->mask));
- ast_data_add_int(data_peer, "port", ast_sockaddr_port(&peer->addr));
- ast_data_add_bool(data_peer, "trunk", ast_test_flag64(peer, IAX_TRUNK));
- ast_data_add_bool(data_peer, "dynamic", ast_test_flag64(peer, IAX_DYNAMIC));
- encmethods_to_str(peer->encmethods, &encmethods);
- ast_data_add_str(data_peer, "encryption", peer->encmethods ? ast_str_buffer(encmethods) : "no");
- peer_unref(peer);
- if (!ast_data_search_match(search, data_peer)) {
- ast_data_remove_node(data_root, data_peer);
- }
- }
- ao2_iterator_destroy(&i);
- return 0;
- }
- #define DATA_EXPORT_IAX2_USER(MEMBER) \
- MEMBER(iax2_user, name, AST_DATA_STRING) \
- MEMBER(iax2_user, dbsecret, AST_DATA_PASSWORD) \
- MEMBER(iax2_user, accountcode, AST_DATA_STRING) \
- MEMBER(iax2_user, mohinterpret, AST_DATA_STRING) \
- MEMBER(iax2_user, mohsuggest, AST_DATA_STRING) \
- MEMBER(iax2_user, inkeys, AST_DATA_STRING) \
- MEMBER(iax2_user, language, AST_DATA_STRING) \
- MEMBER(iax2_user, cid_num, AST_DATA_STRING) \
- MEMBER(iax2_user, cid_name, AST_DATA_STRING) \
- MEMBER(iax2_user, parkinglot, AST_DATA_STRING) \
- MEMBER(iax2_user, maxauthreq, AST_DATA_INTEGER) \
- MEMBER(iax2_user, curauthreq, AST_DATA_INTEGER)
- AST_DATA_STRUCTURE(iax2_user, DATA_EXPORT_IAX2_USER);
- static int users_data_provider_get(const struct ast_data_search *search,
- struct ast_data *data_root)
- {
- struct ast_data *data_user, *data_authmethods, *data_enum_node;
- struct iax2_user *user;
- struct ao2_iterator i;
- char auth[90];
- char *pstr = "";
- i = ao2_iterator_init(users, 0);
- for (; (user = ao2_iterator_next(&i)); user_unref(user)) {
- data_user = ast_data_add_node(data_root, "user");
- if (!data_user) {
- continue;
- }
- ast_data_add_structure(iax2_user, data_user, user);
- iax2_data_add_codecs(data_user, "codecs", user->capability);
- if (!ast_strlen_zero(user->secret)) {
- ast_copy_string(auth, user->secret, sizeof(auth));
- } else if (!ast_strlen_zero(user->inkeys)) {
- snprintf(auth, sizeof(auth), "Key: %s", user->inkeys);
- } else {
- ast_copy_string(auth, "no secret", sizeof(auth));
- }
- ast_data_add_password(data_user, "secret", auth);
- ast_data_add_str(data_user, "context", user->contexts ? user->contexts->context : DEFAULT_CONTEXT);
- /* authmethods */
- data_authmethods = ast_data_add_node(data_user, "authmethods");
- if (!data_authmethods) {
- ast_data_remove_node(data_root, data_user);
- continue;
- }
- ast_data_add_bool(data_authmethods, "rsa", user->authmethods & IAX_AUTH_RSA);
- ast_data_add_bool(data_authmethods, "md5", user->authmethods & IAX_AUTH_MD5);
- ast_data_add_bool(data_authmethods, "plaintext", user->authmethods & IAX_AUTH_PLAINTEXT);
- /* amaflags */
- data_enum_node = ast_data_add_node(data_user, "amaflags");
- if (!data_enum_node) {
- ast_data_remove_node(data_root, data_user);
- continue;
- }
- ast_data_add_int(data_enum_node, "value", user->amaflags);
- ast_data_add_str(data_enum_node, "text", ast_cdr_flags2str(user->amaflags));
- ast_data_add_bool(data_user, "access-control", ast_acl_list_is_empty(user->acl) ? 0 : 1);
- if (ast_test_flag64(user, IAX_CODEC_NOCAP)) {
- pstr = "REQ only";
- } else if (ast_test_flag64(user, IAX_CODEC_NOPREFS)) {
- pstr = "disabled";
- } else {
- pstr = ast_test_flag64(user, IAX_CODEC_USER_FIRST) ? "caller" : "host";
- }
- ast_data_add_str(data_user, "codec-preferences", pstr);
- if (!ast_data_search_match(search, data_user)) {
- ast_data_remove_node(data_root, data_user);
- }
- }
- ao2_iterator_destroy(&i);
- return 0;
- }
- static const struct ast_data_handler peers_data_provider = {
- .version = AST_DATA_HANDLER_VERSION,
- .get = peers_data_provider_get
- };
- static const struct ast_data_handler users_data_provider = {
- .version = AST_DATA_HANDLER_VERSION,
- .get = users_data_provider_get
- };
- static const struct ast_data_entry iax2_data_providers[] = {
- AST_DATA_ENTRY("asterisk/channel/iax2/peers", &peers_data_provider),
- AST_DATA_ENTRY("asterisk/channel/iax2/users", &users_data_provider),
- };
- /*! \brief Load IAX2 module, load configuraiton ---*/
- static int load_module(void)
- {
- static const char config[] = "iax.conf";
- int x = 0;
- struct iax2_registry *reg = NULL;
- if (!(iax2_tech.capabilities = ast_format_cap_alloc())) {
- return AST_MODULE_LOAD_FAILURE;
- }
- ast_format_cap_add_all(iax2_tech.capabilities);
- if (load_objects()) {
- return AST_MODULE_LOAD_FAILURE;
- }
- memset(iaxs, 0, sizeof(iaxs));
- for (x = 0; x < ARRAY_LEN(iaxsl); x++) {
- ast_mutex_init(&iaxsl[x]);
- }
- if (!(sched = ast_sched_context_create())) {
- ast_log(LOG_ERROR, "Failed to create scheduler thread\n");
- return AST_MODULE_LOAD_FAILURE;
- }
- if (ast_sched_start_thread(sched)) {
- ast_sched_context_destroy(sched);
- sched = NULL;
- return AST_MODULE_LOAD_FAILURE;
- }
- if (!(io = io_context_create())) {
- ast_log(LOG_ERROR, "Failed to create I/O context\n");
- ast_sched_context_destroy(sched);
- sched = NULL;
- return AST_MODULE_LOAD_FAILURE;
- }
- if (!(netsock = ast_netsock_list_alloc())) {
- ast_log(LOG_ERROR, "Failed to create netsock list\n");
- io_context_destroy(io);
- ast_sched_context_destroy(sched);
- sched = NULL;
- return AST_MODULE_LOAD_FAILURE;
- }
- ast_netsock_init(netsock);
-
- outsock = ast_netsock_list_alloc();
- if (!outsock) {
- ast_log(LOG_ERROR, "Could not allocate outsock list.\n");
- io_context_destroy(io);
- ast_sched_context_destroy(sched);
- sched = NULL;
- return AST_MODULE_LOAD_FAILURE;
- }
- ast_netsock_init(outsock);
- randomcalltokendata = ast_random();
- iax_set_output(iax_debug_output);
- iax_set_error(iax_error_output);
- jb_setoutput(jb_error_output, jb_warning_output, NULL);
-
- if ((timer = ast_timer_open())) {
- ast_timer_set_rate(timer, 1000 / trunkfreq);
- }
- if (set_config(config, 0, 0) == -1) {
- if (timer) {
- ast_timer_close(timer);
- timer = NULL;
- }
- return AST_MODULE_LOAD_DECLINE;
- }
- #ifdef TEST_FRAMEWORK
- AST_TEST_REGISTER(test_iax2_peers_get);
- AST_TEST_REGISTER(test_iax2_users_get);
- #endif
- /* Register AstData providers */
- ast_data_register_multiple(iax2_data_providers, ARRAY_LEN(iax2_data_providers));
- ast_cli_register_multiple(cli_iax2, ARRAY_LEN(cli_iax2));
- ast_register_application_xml(papp, iax2_prov_app);
- ast_custom_function_register(&iaxpeer_function);
- ast_custom_function_register(&iaxvar_function);
- ast_manager_register_xml("IAXpeers", EVENT_FLAG_SYSTEM | EVENT_FLAG_REPORTING, manager_iax2_show_peers);
- ast_manager_register_xml("IAXpeerlist", EVENT_FLAG_SYSTEM | EVENT_FLAG_REPORTING, manager_iax2_show_peer_list);
- ast_manager_register_xml("IAXnetstats", EVENT_FLAG_SYSTEM | EVENT_FLAG_REPORTING, manager_iax2_show_netstats);
- ast_manager_register_xml("IAXregistry", EVENT_FLAG_SYSTEM | EVENT_FLAG_REPORTING, manager_iax2_show_registry);
- if (ast_channel_register(&iax2_tech)) {
- ast_log(LOG_ERROR, "Unable to register channel class %s\n", "IAX2");
- __unload_module();
- return AST_MODULE_LOAD_FAILURE;
- }
- if (ast_register_switch(&iax2_switch)) {
- ast_log(LOG_ERROR, "Unable to register IAX switch\n");
- }
- if (start_network_thread()) {
- ast_log(LOG_ERROR, "Unable to start network thread\n");
- __unload_module();
- return AST_MODULE_LOAD_FAILURE;
- } else {
- ast_verb(2, "IAX Ready and Listening\n");
- }
- AST_LIST_LOCK(®istrations);
- AST_LIST_TRAVERSE(®istrations, reg, entry)
- iax2_do_register(reg);
- AST_LIST_UNLOCK(®istrations);
-
- ao2_callback(peers, 0, peer_set_sock_cb, NULL);
- ao2_callback(peers, 0, iax2_poke_peer_cb, NULL);
- reload_firmware(0);
- iax_provision_reload(0);
- ast_realtime_require_field("iaxpeers", "name", RQ_CHAR, 10, "ipaddr", RQ_CHAR, 15, "port", RQ_UINTEGER2, 5, "regseconds", RQ_UINTEGER2, 6, SENTINEL);
- network_change_event_subscribe();
- return AST_MODULE_LOAD_SUCCESS;
- }
- AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "Inter Asterisk eXchange (Ver 2)",
- .load = load_module,
- .unload = unload_module,
- .reload = reload,
- .load_pri = AST_MODPRI_CHANNEL_DRIVER,
- .nonoptreq = "res_crypto",
- );
|