| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948 |
- // SPDX-License-Identifier: GPL-2.0-or-later
- /*
- *
- * BlueZ - Bluetooth protocol stack for Linux
- *
- * Copyright (C) 2001-2002 Nokia Corporation
- * Copyright (C) 2002-2003 Maxim Krasnyansky <maxk@qualcomm.com>
- * Copyright (C) 2002-2010 Marcel Holtmann <marcel@holtmann.org>
- * Copyright (C) 2002-2003 Stephen Crane <steve.crane@rococosoft.com>
- *
- *
- */
- #ifdef HAVE_CONFIG_H
- #include <config.h>
- #endif
- #include <stdio.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <unistd.h>
- #include <stdlib.h>
- #include <limits.h>
- #include <string.h>
- #include <syslog.h>
- #include <sys/time.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <sys/un.h>
- #include <netinet/in.h>
- #include "bluetooth.h"
- #include "hci.h"
- #include "hci_lib.h"
- #include "l2cap.h"
- #include "sdp.h"
- #include "sdp_lib.h"
- #define SDPINF(fmt, arg...) syslog(LOG_INFO, fmt "\n", ## arg)
- #define SDPERR(fmt, arg...) syslog(LOG_ERR, "%s: " fmt "\n", __func__ , ## arg)
- #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
- #ifdef SDP_DEBUG
- #define SDPDBG(fmt, arg...) syslog(LOG_DEBUG, "%s: " fmt "\n", __func__ , ## arg)
- #else
- #define SDPDBG(fmt...)
- #endif
- static uint128_t bluetooth_base_uuid = {
- .data = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00,
- 0x80, 0x00, 0x00, 0x80, 0x5F, 0x9B, 0x34, 0xFB }
- };
- #define SDP_MAX_ATTR_LEN 65535
- /* match MTU used by RFCOMM */
- #define SDP_LARGE_L2CAP_MTU 1013
- static sdp_data_t *sdp_copy_seq(sdp_data_t *data);
- static int sdp_attr_add_new_with_length(sdp_record_t *rec,
- uint16_t attr, uint8_t dtd, const void *value, uint32_t len);
- static int sdp_gen_buffer(sdp_buf_t *buf, sdp_data_t *d);
- /* Message structure. */
- struct tupla {
- int index;
- char *str;
- };
- static struct tupla Protocol[] = {
- { SDP_UUID, "SDP" },
- { UDP_UUID, "UDP" },
- { RFCOMM_UUID, "RFCOMM" },
- { TCP_UUID, "TCP" },
- { TCS_BIN_UUID, "TCS-BIN" },
- { TCS_AT_UUID, "TCS-AT" },
- { OBEX_UUID, "OBEX" },
- { IP_UUID, "IP" },
- { FTP_UUID, "FTP" },
- { HTTP_UUID, "HTTP" },
- { WSP_UUID, "WSP" },
- { BNEP_UUID, "BNEP" },
- { UPNP_UUID, "UPNP" },
- { HIDP_UUID, "HIDP" },
- { HCRP_CTRL_UUID, "HCRP-Ctrl" },
- { HCRP_DATA_UUID, "HCRP-Data" },
- { HCRP_NOTE_UUID, "HCRP-Notify" },
- { AVCTP_UUID, "AVCTP" },
- { AVDTP_UUID, "AVDTP" },
- { CMTP_UUID, "CMTP" },
- { UDI_UUID, "UDI" },
- { MCAP_CTRL_UUID, "MCAP-Ctrl" },
- { MCAP_DATA_UUID, "MCAP-Data" },
- { L2CAP_UUID, "L2CAP" },
- { ATT_UUID, "ATT" },
- { 0 }
- };
- static struct tupla ServiceClass[] = {
- { SDP_SERVER_SVCLASS_ID, "SDP Server" },
- { BROWSE_GRP_DESC_SVCLASS_ID, "Browse Group Descriptor" },
- { PUBLIC_BROWSE_GROUP, "Public Browse Group" },
- { SERIAL_PORT_SVCLASS_ID, "Serial Port" },
- { LAN_ACCESS_SVCLASS_ID, "LAN Access Using PPP" },
- { DIALUP_NET_SVCLASS_ID, "Dialup Networking" },
- { IRMC_SYNC_SVCLASS_ID, "IrMC Sync" },
- { OBEX_OBJPUSH_SVCLASS_ID, "OBEX Object Push" },
- { OBEX_FILETRANS_SVCLASS_ID, "OBEX File Transfer" },
- { IRMC_SYNC_CMD_SVCLASS_ID, "IrMC Sync Command" },
- { HEADSET_SVCLASS_ID, "Headset" },
- { CORDLESS_TELEPHONY_SVCLASS_ID, "Cordless Telephony" },
- { AUDIO_SOURCE_SVCLASS_ID, "Audio Source" },
- { AUDIO_SINK_SVCLASS_ID, "Audio Sink" },
- { AV_REMOTE_TARGET_SVCLASS_ID, "AV Remote Target" },
- { ADVANCED_AUDIO_SVCLASS_ID, "Advanced Audio" },
- { AV_REMOTE_SVCLASS_ID, "AV Remote" },
- { AV_REMOTE_CONTROLLER_SVCLASS_ID, "AV Remote Controller" },
- { INTERCOM_SVCLASS_ID, "Intercom" },
- { FAX_SVCLASS_ID, "Fax" },
- { HEADSET_AGW_SVCLASS_ID, "Headset Audio Gateway" },
- { WAP_SVCLASS_ID, "WAP" },
- { WAP_CLIENT_SVCLASS_ID, "WAP Client" },
- { PANU_SVCLASS_ID, "PAN User" },
- { NAP_SVCLASS_ID, "Network Access Point" },
- { GN_SVCLASS_ID, "PAN Group Network" },
- { DIRECT_PRINTING_SVCLASS_ID, "Direct Printing" },
- { REFERENCE_PRINTING_SVCLASS_ID, "Reference Printing" },
- { IMAGING_SVCLASS_ID, "Imaging" },
- { IMAGING_RESPONDER_SVCLASS_ID, "Imaging Responder" },
- { IMAGING_ARCHIVE_SVCLASS_ID, "Imaging Automatic Archive" },
- { IMAGING_REFOBJS_SVCLASS_ID, "Imaging Referenced Objects" },
- { HANDSFREE_SVCLASS_ID, "Handsfree" },
- { HANDSFREE_AGW_SVCLASS_ID, "Handsfree Audio Gateway" },
- { DIRECT_PRT_REFOBJS_SVCLASS_ID, "Direct Printing Ref. Objects" },
- { REFLECTED_UI_SVCLASS_ID, "Reflected UI" },
- { BASIC_PRINTING_SVCLASS_ID, "Basic Printing" },
- { PRINTING_STATUS_SVCLASS_ID, "Printing Status" },
- { HID_SVCLASS_ID, "Human Interface Device" },
- { HCR_SVCLASS_ID, "Hardcopy Cable Replacement" },
- { HCR_PRINT_SVCLASS_ID, "HCR Print" },
- { HCR_SCAN_SVCLASS_ID, "HCR Scan" },
- { CIP_SVCLASS_ID, "Common ISDN Access" },
- { VIDEO_CONF_GW_SVCLASS_ID, "Video Conferencing Gateway" },
- { UDI_MT_SVCLASS_ID, "UDI MT" },
- { UDI_TA_SVCLASS_ID, "UDI TA" },
- { AV_SVCLASS_ID, "Audio/Video" },
- { SAP_SVCLASS_ID, "SIM Access" },
- { PBAP_PCE_SVCLASS_ID, "Phonebook Access - PCE" },
- { PBAP_PSE_SVCLASS_ID, "Phonebook Access - PSE" },
- { PBAP_SVCLASS_ID, "Phonebook Access" },
- { MAP_MSE_SVCLASS_ID, "Message Access - MAS" },
- { MAP_MCE_SVCLASS_ID, "Message Access - MNS" },
- { MAP_SVCLASS_ID, "Message Access" },
- { PNP_INFO_SVCLASS_ID, "PnP Information" },
- { GENERIC_NETWORKING_SVCLASS_ID, "Generic Networking" },
- { GENERIC_FILETRANS_SVCLASS_ID, "Generic File Transfer" },
- { GENERIC_AUDIO_SVCLASS_ID, "Generic Audio" },
- { GENERIC_TELEPHONY_SVCLASS_ID, "Generic Telephony" },
- { UPNP_SVCLASS_ID, "UPnP" },
- { UPNP_IP_SVCLASS_ID, "UPnP IP" },
- { UPNP_PAN_SVCLASS_ID, "UPnP PAN" },
- { UPNP_LAP_SVCLASS_ID, "UPnP LAP" },
- { UPNP_L2CAP_SVCLASS_ID, "UPnP L2CAP" },
- { VIDEO_SOURCE_SVCLASS_ID, "Video Source" },
- { VIDEO_SINK_SVCLASS_ID, "Video Sink" },
- { VIDEO_DISTRIBUTION_SVCLASS_ID, "Video Distribution" },
- { HDP_SVCLASS_ID, "HDP" },
- { HDP_SOURCE_SVCLASS_ID, "HDP Source" },
- { HDP_SINK_SVCLASS_ID, "HDP Sink" },
- { GENERIC_ACCESS_SVCLASS_ID, "Generic Access" },
- { GENERIC_ATTRIB_SVCLASS_ID, "Generic Attribute" },
- { APPLE_AGENT_SVCLASS_ID, "Apple Agent" },
- { 0 }
- };
- #define Profile ServiceClass
- static char *string_lookup(struct tupla *pt0, int index)
- {
- struct tupla *pt;
- for (pt = pt0; pt->index; pt++)
- if (pt->index == index)
- return pt->str;
- return "";
- }
- static char *string_lookup_uuid(struct tupla *pt0, const uuid_t *uuid)
- {
- uuid_t tmp_uuid;
- memcpy(&tmp_uuid, uuid, sizeof(tmp_uuid));
- if (sdp_uuid128_to_uuid(&tmp_uuid)) {
- switch (tmp_uuid.type) {
- case SDP_UUID16:
- return string_lookup(pt0, tmp_uuid.value.uuid16);
- case SDP_UUID32:
- return string_lookup(pt0, tmp_uuid.value.uuid32);
- }
- }
- return "";
- }
- /*
- * Prints into a string the Protocol UUID
- * coping a maximum of n characters.
- */
- static int uuid2str(struct tupla *message, const uuid_t *uuid, char *str, size_t n)
- {
- char *str2;
- if (!uuid) {
- snprintf(str, n, "NULL");
- return -2;
- }
- switch (uuid->type) {
- case SDP_UUID16:
- str2 = string_lookup(message, uuid->value.uuid16);
- snprintf(str, n, "%s", str2);
- break;
- case SDP_UUID32:
- str2 = string_lookup(message, uuid->value.uuid32);
- snprintf(str, n, "%s", str2);
- break;
- case SDP_UUID128:
- str2 = string_lookup_uuid(message, uuid);
- snprintf(str, n, "%s", str2);
- break;
- default:
- snprintf(str, n, "Type of UUID (%x) unknown.", uuid->type);
- return -1;
- }
- return 0;
- }
- int sdp_proto_uuid2strn(const uuid_t *uuid, char *str, size_t n)
- {
- return uuid2str(Protocol, uuid, str, n);
- }
- int sdp_svclass_uuid2strn(const uuid_t *uuid, char *str, size_t n)
- {
- return uuid2str(ServiceClass, uuid, str, n);
- }
- int sdp_profile_uuid2strn(const uuid_t *uuid, char *str, size_t n)
- {
- return uuid2str(Profile, uuid, str, n);
- }
- /*
- * convert the UUID to string, copying a maximum of n characters.
- */
- int sdp_uuid2strn(const uuid_t *uuid, char *str, size_t n)
- {
- if (!uuid) {
- snprintf(str, n, "NULL");
- return -2;
- }
- switch (uuid->type) {
- case SDP_UUID16:
- snprintf(str, n, "%.4x", uuid->value.uuid16);
- break;
- case SDP_UUID32:
- snprintf(str, n, "%.8x", uuid->value.uuid32);
- break;
- case SDP_UUID128:{
- unsigned int data0;
- unsigned short data1;
- unsigned short data2;
- unsigned short data3;
- unsigned int data4;
- unsigned short data5;
- memcpy(&data0, &uuid->value.uuid128.data[0], 4);
- memcpy(&data1, &uuid->value.uuid128.data[4], 2);
- memcpy(&data2, &uuid->value.uuid128.data[6], 2);
- memcpy(&data3, &uuid->value.uuid128.data[8], 2);
- memcpy(&data4, &uuid->value.uuid128.data[10], 4);
- memcpy(&data5, &uuid->value.uuid128.data[14], 2);
- snprintf(str, n, "%.8x-%.4x-%.4x-%.4x-%.8x%.4x",
- ntohl(data0), ntohs(data1),
- ntohs(data2), ntohs(data3),
- ntohl(data4), ntohs(data5));
- }
- break;
- default:
- snprintf(str, n, "Type of UUID (%x) unknown.", uuid->type);
- return -1; /* Enum type of UUID not set */
- }
- return 0;
- }
- #ifdef SDP_DEBUG
- /*
- * Function prints the UUID in hex as per defined syntax -
- *
- * 4bytes-2bytes-2bytes-2bytes-6bytes
- *
- * There is some ugly code, including hardcoding, but
- * that is just the way it is converting 16 and 32 bit
- * UUIDs to 128 bit as defined in the SDP doc
- */
- void sdp_uuid_print(const uuid_t *uuid)
- {
- if (uuid == NULL) {
- SDPERR("Null passed to print UUID");
- return;
- }
- if (uuid->type == SDP_UUID16) {
- SDPDBG(" uint16_t : 0x%.4x", uuid->value.uuid16);
- } else if (uuid->type == SDP_UUID32) {
- SDPDBG(" uint32_t : 0x%.8x", uuid->value.uuid32);
- } else if (uuid->type == SDP_UUID128) {
- unsigned int data0;
- unsigned short data1;
- unsigned short data2;
- unsigned short data3;
- unsigned int data4;
- unsigned short data5;
- memcpy(&data0, &uuid->value.uuid128.data[0], 4);
- memcpy(&data1, &uuid->value.uuid128.data[4], 2);
- memcpy(&data2, &uuid->value.uuid128.data[6], 2);
- memcpy(&data3, &uuid->value.uuid128.data[8], 2);
- memcpy(&data4, &uuid->value.uuid128.data[10], 4);
- memcpy(&data5, &uuid->value.uuid128.data[14], 2);
- SDPDBG(" uint128_t : 0x%.8x-%.4x-%.4x-%.4x-%.8x%.4x",
- ntohl(data0), ntohs(data1), ntohs(data2),
- ntohs(data3), ntohl(data4), ntohs(data5));
- } else
- SDPERR("Enum type of UUID not set");
- }
- #endif
- sdp_data_t *sdp_data_alloc_with_length(uint8_t dtd, const void *value,
- uint32_t length)
- {
- sdp_data_t *seq;
- sdp_data_t *d = bt_malloc0(sizeof(sdp_data_t));
- if (!d)
- return NULL;
- d->dtd = dtd;
- d->unitSize = sizeof(uint8_t);
- switch (dtd) {
- case SDP_DATA_NIL:
- break;
- case SDP_UINT8:
- d->val.uint8 = *(uint8_t *) value;
- d->unitSize += sizeof(uint8_t);
- break;
- case SDP_INT8:
- case SDP_BOOL:
- d->val.int8 = *(int8_t *) value;
- d->unitSize += sizeof(int8_t);
- break;
- case SDP_UINT16:
- d->val.uint16 = bt_get_unaligned((uint16_t *) value);
- d->unitSize += sizeof(uint16_t);
- break;
- case SDP_INT16:
- d->val.int16 = bt_get_unaligned((int16_t *) value);
- d->unitSize += sizeof(int16_t);
- break;
- case SDP_UINT32:
- d->val.uint32 = bt_get_unaligned((uint32_t *) value);
- d->unitSize += sizeof(uint32_t);
- break;
- case SDP_INT32:
- d->val.int32 = bt_get_unaligned((int32_t *) value);
- d->unitSize += sizeof(int32_t);
- break;
- case SDP_INT64:
- d->val.int64 = bt_get_unaligned((int64_t *) value);
- d->unitSize += sizeof(int64_t);
- break;
- case SDP_UINT64:
- d->val.uint64 = bt_get_unaligned((uint64_t *) value);
- d->unitSize += sizeof(uint64_t);
- break;
- case SDP_UINT128:
- memcpy(&d->val.uint128.data, value, sizeof(uint128_t));
- d->unitSize += sizeof(uint128_t);
- break;
- case SDP_INT128:
- memcpy(&d->val.int128.data, value, sizeof(uint128_t));
- d->unitSize += sizeof(uint128_t);
- break;
- case SDP_UUID16:
- sdp_uuid16_create(&d->val.uuid, bt_get_unaligned((uint16_t *) value));
- d->unitSize += sizeof(uint16_t);
- break;
- case SDP_UUID32:
- sdp_uuid32_create(&d->val.uuid, bt_get_unaligned((uint32_t *) value));
- d->unitSize += sizeof(uint32_t);
- break;
- case SDP_UUID128:
- sdp_uuid128_create(&d->val.uuid, value);
- d->unitSize += sizeof(uint128_t);
- break;
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- if (!value) {
- free(d);
- return NULL;
- }
- d->unitSize += length;
- if (length <= USHRT_MAX) {
- d->val.str = malloc(length);
- if (!d->val.str) {
- free(d);
- return NULL;
- }
- memcpy(d->val.str, value, length);
- } else {
- SDPERR("Strings of size > USHRT_MAX not supported");
- free(d);
- d = NULL;
- }
- break;
- case SDP_URL_STR32:
- case SDP_TEXT_STR32:
- SDPERR("Strings of size > USHRT_MAX not supported");
- break;
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- if (dtd == SDP_ALT8 || dtd == SDP_SEQ8)
- d->unitSize += sizeof(uint8_t);
- else if (dtd == SDP_ALT16 || dtd == SDP_SEQ16)
- d->unitSize += sizeof(uint16_t);
- else if (dtd == SDP_ALT32 || dtd == SDP_SEQ32)
- d->unitSize += sizeof(uint32_t);
- seq = (sdp_data_t *)value;
- d->val.dataseq = seq;
- for (; seq; seq = seq->next)
- d->unitSize += seq->unitSize;
- break;
- default:
- free(d);
- d = NULL;
- }
- return d;
- }
- sdp_data_t *sdp_data_alloc(uint8_t dtd, const void *value)
- {
- uint32_t length;
- switch (dtd) {
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- if (!value)
- return NULL;
- length = strlen((char *) value);
- break;
- default:
- length = 0;
- break;
- }
- return sdp_data_alloc_with_length(dtd, value, length);
- }
- sdp_data_t *sdp_seq_append(sdp_data_t *seq, sdp_data_t *d)
- {
- if (seq) {
- sdp_data_t *p;
- for (p = seq; p->next; p = p->next);
- p->next = d;
- } else
- seq = d;
- d->next = NULL;
- return seq;
- }
- sdp_data_t *sdp_seq_alloc_with_length(void **dtds, void **values, int *length,
- int len)
- {
- sdp_data_t *curr = NULL, *seq = NULL;
- int i;
- for (i = 0; i < len; i++) {
- sdp_data_t *data;
- int8_t dtd = *(uint8_t *) dtds[i];
- if (dtd >= SDP_SEQ8 && dtd <= SDP_ALT32)
- data = (sdp_data_t *) values[i];
- else
- data = sdp_data_alloc_with_length(dtd, values[i], length[i]);
- if (!data)
- return NULL;
- if (curr)
- curr->next = data;
- else
- seq = data;
- curr = data;
- }
- return sdp_data_alloc(SDP_SEQ8, seq);
- }
- sdp_data_t *sdp_seq_alloc(void **dtds, void **values, int len)
- {
- sdp_data_t *curr = NULL, *seq = NULL;
- int i;
- for (i = 0; i < len; i++) {
- sdp_data_t *data;
- uint8_t dtd = *(uint8_t *) dtds[i];
- if (dtd >= SDP_SEQ8 && dtd <= SDP_ALT32)
- data = (sdp_data_t *) values[i];
- else
- data = sdp_data_alloc(dtd, values[i]);
- if (!data)
- return NULL;
- if (curr)
- curr->next = data;
- else
- seq = data;
- curr = data;
- }
- return sdp_data_alloc(SDP_SEQ8, seq);
- }
- static void extract_svclass_uuid(sdp_data_t *data, uuid_t *uuid)
- {
- sdp_data_t *d;
- if (!data || !SDP_IS_SEQ(data->dtd))
- return;
- d = data->val.dataseq;
- if (!d)
- return;
- if (d->dtd < SDP_UUID16 || d->dtd > SDP_UUID128)
- return;
- *uuid = d->val.uuid;
- }
- int sdp_attr_add(sdp_record_t *rec, uint16_t attr, sdp_data_t *d)
- {
- sdp_data_t *p = sdp_data_get(rec, attr);
- if (p)
- return -1;
- d->attrId = attr;
- rec->attrlist = sdp_list_insert_sorted(rec->attrlist, d, sdp_attrid_comp_func);
- if (attr == SDP_ATTR_SVCLASS_ID_LIST)
- extract_svclass_uuid(d, &rec->svclass);
- return 0;
- }
- void sdp_attr_remove(sdp_record_t *rec, uint16_t attr)
- {
- sdp_data_t *d = sdp_data_get(rec, attr);
- if (d)
- rec->attrlist = sdp_list_remove(rec->attrlist, d);
- if (attr == SDP_ATTR_SVCLASS_ID_LIST)
- memset(&rec->svclass, 0, sizeof(rec->svclass));
- }
- void sdp_set_seq_len(uint8_t *ptr, uint32_t length)
- {
- uint8_t dtd = *ptr++;
- switch (dtd) {
- case SDP_SEQ8:
- case SDP_ALT8:
- case SDP_TEXT_STR8:
- case SDP_URL_STR8:
- *ptr = (uint8_t) length;
- break;
- case SDP_SEQ16:
- case SDP_ALT16:
- case SDP_TEXT_STR16:
- case SDP_URL_STR16:
- bt_put_be16(length, ptr);
- break;
- case SDP_SEQ32:
- case SDP_ALT32:
- case SDP_TEXT_STR32:
- case SDP_URL_STR32:
- bt_put_be32(length, ptr);
- break;
- }
- }
- static int sdp_get_data_type_size(uint8_t dtd)
- {
- int size = sizeof(uint8_t);
- switch (dtd) {
- case SDP_SEQ8:
- case SDP_TEXT_STR8:
- case SDP_URL_STR8:
- case SDP_ALT8:
- size += sizeof(uint8_t);
- break;
- case SDP_SEQ16:
- case SDP_TEXT_STR16:
- case SDP_URL_STR16:
- case SDP_ALT16:
- size += sizeof(uint16_t);
- break;
- case SDP_SEQ32:
- case SDP_TEXT_STR32:
- case SDP_URL_STR32:
- case SDP_ALT32:
- size += sizeof(uint32_t);
- break;
- }
- return size;
- }
- void sdp_set_attrid(sdp_buf_t *buf, uint16_t attr)
- {
- uint8_t *p = buf->data;
- /* data type for attr */
- *p++ = SDP_UINT16;
- buf->data_size = sizeof(uint8_t);
- bt_put_be16(attr, p);
- buf->data_size += sizeof(uint16_t);
- }
- static int get_data_size(sdp_buf_t *buf, sdp_data_t *sdpdata)
- {
- sdp_data_t *d;
- int n = 0;
- for (d = sdpdata->val.dataseq; d; d = d->next) {
- if (buf->data)
- n += sdp_gen_pdu(buf, d);
- else
- n += sdp_gen_buffer(buf, d);
- }
- return n;
- }
- static int sdp_get_data_size(sdp_buf_t *buf, sdp_data_t *d)
- {
- uint32_t data_size = 0;
- uint8_t dtd = d->dtd;
- switch (dtd) {
- case SDP_DATA_NIL:
- break;
- case SDP_UINT8:
- data_size = sizeof(uint8_t);
- break;
- case SDP_UINT16:
- data_size = sizeof(uint16_t);
- break;
- case SDP_UINT32:
- data_size = sizeof(uint32_t);
- break;
- case SDP_UINT64:
- data_size = sizeof(uint64_t);
- break;
- case SDP_UINT128:
- data_size = sizeof(uint128_t);
- break;
- case SDP_INT8:
- case SDP_BOOL:
- data_size = sizeof(int8_t);
- break;
- case SDP_INT16:
- data_size = sizeof(int16_t);
- break;
- case SDP_INT32:
- data_size = sizeof(int32_t);
- break;
- case SDP_INT64:
- data_size = sizeof(int64_t);
- break;
- case SDP_INT128:
- data_size = sizeof(uint128_t);
- break;
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- case SDP_TEXT_STR32:
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_URL_STR32:
- data_size = d->unitSize - sizeof(uint8_t);
- break;
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- data_size = get_data_size(buf, d);
- break;
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- data_size = get_data_size(buf, d);
- break;
- case SDP_UUID16:
- data_size = sizeof(uint16_t);
- break;
- case SDP_UUID32:
- data_size = sizeof(uint32_t);
- break;
- case SDP_UUID128:
- data_size = sizeof(uint128_t);
- break;
- default:
- break;
- }
- return data_size;
- }
- static int sdp_gen_buffer(sdp_buf_t *buf, sdp_data_t *d)
- {
- int orig = buf->buf_size;
- if (buf->buf_size == 0 && d->dtd == 0) {
- /* create initial sequence */
- buf->buf_size += sizeof(uint8_t);
- /* reserve space for sequence size */
- buf->buf_size += sizeof(uint8_t);
- }
- /* attribute length */
- buf->buf_size += sizeof(uint8_t) + sizeof(uint16_t);
- buf->buf_size += sdp_get_data_type_size(d->dtd);
- buf->buf_size += sdp_get_data_size(buf, d);
- if (buf->buf_size > UCHAR_MAX && d->dtd == SDP_SEQ8)
- buf->buf_size += sizeof(uint8_t);
- return buf->buf_size - orig;
- }
- int sdp_gen_pdu(sdp_buf_t *buf, sdp_data_t *d)
- {
- uint32_t pdu_size, data_size;
- unsigned char *src = NULL, is_seq = 0, is_alt = 0;
- uint16_t u16;
- uint32_t u32;
- uint64_t u64;
- uint128_t u128;
- uint8_t *seqp = buf->data + buf->data_size;
- uint32_t orig_data_size = buf->data_size;
- recalculate:
- pdu_size = sdp_get_data_type_size(d->dtd);
- buf->data_size += pdu_size;
- data_size = sdp_get_data_size(buf, d);
- if (data_size > UCHAR_MAX && d->dtd == SDP_SEQ8) {
- buf->data_size = orig_data_size;
- d->dtd = SDP_SEQ16;
- goto recalculate;
- }
- *seqp = d->dtd;
- switch (d->dtd) {
- case SDP_DATA_NIL:
- break;
- case SDP_UINT8:
- src = &d->val.uint8;
- break;
- case SDP_UINT16:
- u16 = htons(d->val.uint16);
- src = (unsigned char *) &u16;
- break;
- case SDP_UINT32:
- u32 = htonl(d->val.uint32);
- src = (unsigned char *) &u32;
- break;
- case SDP_UINT64:
- u64 = hton64(d->val.uint64);
- src = (unsigned char *) &u64;
- break;
- case SDP_UINT128:
- hton128(&d->val.uint128, &u128);
- src = (unsigned char *) &u128;
- break;
- case SDP_INT8:
- case SDP_BOOL:
- src = (unsigned char *) &d->val.int8;
- break;
- case SDP_INT16:
- u16 = htons(d->val.int16);
- src = (unsigned char *) &u16;
- break;
- case SDP_INT32:
- u32 = htonl(d->val.int32);
- src = (unsigned char *) &u32;
- break;
- case SDP_INT64:
- u64 = hton64(d->val.int64);
- src = (unsigned char *) &u64;
- break;
- case SDP_INT128:
- hton128(&d->val.int128, &u128);
- src = (unsigned char *) &u128;
- break;
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- case SDP_TEXT_STR32:
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_URL_STR32:
- src = (unsigned char *) d->val.str;
- sdp_set_seq_len(seqp, data_size);
- break;
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- is_seq = 1;
- sdp_set_seq_len(seqp, data_size);
- break;
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- is_alt = 1;
- sdp_set_seq_len(seqp, data_size);
- break;
- case SDP_UUID16:
- u16 = htons(d->val.uuid.value.uuid16);
- src = (unsigned char *) &u16;
- break;
- case SDP_UUID32:
- u32 = htonl(d->val.uuid.value.uuid32);
- src = (unsigned char *) &u32;
- break;
- case SDP_UUID128:
- src = (unsigned char *) &d->val.uuid.value.uuid128;
- break;
- default:
- break;
- }
- if (!is_seq && !is_alt) {
- if (src && buf->buf_size >= buf->data_size + data_size) {
- memcpy(buf->data + buf->data_size, src, data_size);
- buf->data_size += data_size;
- } else if (d->dtd != SDP_DATA_NIL) {
- SDPDBG("Gen PDU : Can't copy from invalid source or dest");
- }
- }
- pdu_size += data_size;
- return pdu_size;
- }
- static void sdp_attr_pdu(void *value, void *udata)
- {
- sdp_append_to_pdu((sdp_buf_t *)udata, (sdp_data_t *)value);
- }
- static void sdp_attr_size(void *value, void *udata)
- {
- sdp_gen_buffer((sdp_buf_t *)udata, (sdp_data_t *)value);
- }
- int sdp_gen_record_pdu(const sdp_record_t *rec, sdp_buf_t *buf)
- {
- memset(buf, 0, sizeof(sdp_buf_t));
- sdp_list_foreach(rec->attrlist, sdp_attr_size, buf);
- buf->data = bt_malloc0(buf->buf_size);
- if (!buf->data)
- return -ENOMEM;
- buf->data_size = 0;
- sdp_list_foreach(rec->attrlist, sdp_attr_pdu, buf);
- return 0;
- }
- void sdp_attr_replace(sdp_record_t *rec, uint16_t attr, sdp_data_t *d)
- {
- sdp_data_t *p;
- if (!rec)
- return;
- p = sdp_data_get(rec, attr);
- if (p) {
- rec->attrlist = sdp_list_remove(rec->attrlist, p);
- sdp_data_free(p);
- }
- d->attrId = attr;
- rec->attrlist = sdp_list_insert_sorted(rec->attrlist, d, sdp_attrid_comp_func);
- if (attr == SDP_ATTR_SVCLASS_ID_LIST)
- extract_svclass_uuid(d, &rec->svclass);
- }
- int sdp_attrid_comp_func(const void *key1, const void *key2)
- {
- const sdp_data_t *d1 = (const sdp_data_t *)key1;
- const sdp_data_t *d2 = (const sdp_data_t *)key2;
- if (d1 && d2)
- return d1->attrId - d2->attrId;
- return 0;
- }
- static void data_seq_free(sdp_data_t *seq)
- {
- sdp_data_t *d = seq->val.dataseq;
- while (d) {
- sdp_data_t *next = d->next;
- sdp_data_free(d);
- d = next;
- }
- }
- void sdp_data_free(sdp_data_t *d)
- {
- switch (d->dtd) {
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- data_seq_free(d);
- break;
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_URL_STR32:
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- case SDP_TEXT_STR32:
- free(d->val.str);
- break;
- }
- free(d);
- }
- int sdp_uuid_extract(const uint8_t *p, int bufsize, uuid_t *uuid, int *scanned)
- {
- uint8_t type;
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- return -1;
- }
- type = *(const uint8_t *) p;
- if (!SDP_IS_UUID(type)) {
- SDPERR("Unknown data type : %d expecting a svc UUID", type);
- return -1;
- }
- p += sizeof(uint8_t);
- *scanned += sizeof(uint8_t);
- bufsize -= sizeof(uint8_t);
- if (type == SDP_UUID16) {
- if (bufsize < (int) sizeof(uint16_t)) {
- SDPERR("Not enough room for 16-bit UUID");
- return -1;
- }
- sdp_uuid16_create(uuid, bt_get_be16(p));
- *scanned += sizeof(uint16_t);
- } else if (type == SDP_UUID32) {
- if (bufsize < (int) sizeof(uint32_t)) {
- SDPERR("Not enough room for 32-bit UUID");
- return -1;
- }
- sdp_uuid32_create(uuid, bt_get_be32(p));
- *scanned += sizeof(uint32_t);
- } else {
- if (bufsize < (int) sizeof(uint128_t)) {
- SDPERR("Not enough room for 128-bit UUID");
- return -1;
- }
- sdp_uuid128_create(uuid, p);
- *scanned += sizeof(uint128_t);
- }
- return 0;
- }
- static sdp_data_t *extract_int(const void *p, int bufsize, int *len)
- {
- sdp_data_t *d;
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- return NULL;
- }
- d = bt_malloc0(sizeof(sdp_data_t));
- if (!d)
- return NULL;
- SDPDBG("Extracting integer");
- d->dtd = *(uint8_t *) p;
- p += sizeof(uint8_t);
- *len += sizeof(uint8_t);
- bufsize -= sizeof(uint8_t);
- switch (d->dtd) {
- case SDP_DATA_NIL:
- break;
- case SDP_BOOL:
- case SDP_INT8:
- case SDP_UINT8:
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- *len += sizeof(uint8_t);
- d->val.uint8 = *(uint8_t *) p;
- break;
- case SDP_INT16:
- case SDP_UINT16:
- if (bufsize < (int) sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- *len += sizeof(uint16_t);
- d->val.uint16 = bt_get_be16(p);
- break;
- case SDP_INT32:
- case SDP_UINT32:
- if (bufsize < (int) sizeof(uint32_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- *len += sizeof(uint32_t);
- d->val.uint32 = bt_get_be32(p);
- break;
- case SDP_INT64:
- case SDP_UINT64:
- if (bufsize < (int) sizeof(uint64_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- *len += sizeof(uint64_t);
- d->val.uint64 = bt_get_be64(p);
- break;
- case SDP_INT128:
- case SDP_UINT128:
- if (bufsize < (int) sizeof(uint128_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- *len += sizeof(uint128_t);
- ntoh128((uint128_t *) p, &d->val.uint128);
- break;
- default:
- free(d);
- d = NULL;
- }
- return d;
- }
- static sdp_data_t *extract_uuid(const uint8_t *p, int bufsize, int *len,
- sdp_record_t *rec)
- {
- sdp_data_t *d = bt_malloc0(sizeof(sdp_data_t));
- if (!d)
- return NULL;
- SDPDBG("Extracting UUID");
- if (sdp_uuid_extract(p, bufsize, &d->val.uuid, len) < 0) {
- free(d);
- return NULL;
- }
- d->dtd = *p;
- if (rec)
- sdp_pattern_add_uuid(rec, &d->val.uuid);
- return d;
- }
- /*
- * Extract strings from the PDU (could be service description and similar info)
- */
- static sdp_data_t *extract_str(const void *p, int bufsize, int *len)
- {
- char *s;
- int n;
- sdp_data_t *d;
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- return NULL;
- }
- d = bt_malloc0(sizeof(sdp_data_t));
- if (!d)
- return NULL;
- d->dtd = *(uint8_t *) p;
- p += sizeof(uint8_t);
- *len += sizeof(uint8_t);
- bufsize -= sizeof(uint8_t);
- switch (d->dtd) {
- case SDP_TEXT_STR8:
- case SDP_URL_STR8:
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- n = *(uint8_t *) p;
- p += sizeof(uint8_t);
- *len += sizeof(uint8_t);
- bufsize -= sizeof(uint8_t);
- break;
- case SDP_TEXT_STR16:
- case SDP_URL_STR16:
- if (bufsize < (int) sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- free(d);
- return NULL;
- }
- n = bt_get_be16(p);
- p += sizeof(uint16_t);
- *len += sizeof(uint16_t);
- bufsize -= sizeof(uint16_t);
- break;
- default:
- SDPERR("Sizeof text string > UINT16_MAX");
- free(d);
- return NULL;
- }
- if (bufsize < n) {
- SDPERR("String too long to fit in packet");
- free(d);
- return NULL;
- }
- s = bt_malloc0(n + 1);
- if (!s) {
- SDPERR("Not enough memory for incoming string");
- free(d);
- return NULL;
- }
- memcpy(s, p, n);
- *len += n;
- SDPDBG("Len : %d", n);
- SDPDBG("Str : %s", s);
- d->val.str = s;
- d->unitSize = n + sizeof(uint8_t);
- return d;
- }
- /*
- * Extract the sequence type and its length, and return offset into buf
- * or 0 on failure.
- */
- int sdp_extract_seqtype(const uint8_t *buf, int bufsize, uint8_t *dtdp, int *size)
- {
- uint8_t dtd;
- int scanned = sizeof(uint8_t);
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- return 0;
- }
- dtd = *(uint8_t *) buf;
- buf += sizeof(uint8_t);
- bufsize -= sizeof(uint8_t);
- *dtdp = dtd;
- switch (dtd) {
- case SDP_SEQ8:
- case SDP_ALT8:
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- return 0;
- }
- *size = *(uint8_t *) buf;
- scanned += sizeof(uint8_t);
- break;
- case SDP_SEQ16:
- case SDP_ALT16:
- if (bufsize < (int) sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- return 0;
- }
- *size = bt_get_be16(buf);
- scanned += sizeof(uint16_t);
- break;
- case SDP_SEQ32:
- case SDP_ALT32:
- if (bufsize < (int) sizeof(uint32_t)) {
- SDPERR("Unexpected end of packet");
- return 0;
- }
- *size = bt_get_be32(buf);
- scanned += sizeof(uint32_t);
- break;
- default:
- SDPERR("Unknown sequence type, aborting");
- return 0;
- }
- return scanned;
- }
- static sdp_data_t *extract_seq(const void *p, int bufsize, int *len,
- sdp_record_t *rec)
- {
- int seqlen, n = 0;
- sdp_data_t *curr, *prev;
- sdp_data_t *d = bt_malloc0(sizeof(sdp_data_t));
- if (!d)
- return NULL;
- SDPDBG("Extracting SEQ");
- *len = sdp_extract_seqtype(p, bufsize, &d->dtd, &seqlen);
- SDPDBG("Sequence Type : 0x%x length : 0x%x", d->dtd, seqlen);
- if (*len == 0)
- return d;
- if (*len > bufsize) {
- SDPERR("Packet not big enough to hold sequence.");
- free(d);
- return NULL;
- }
- p += *len;
- bufsize -= *len;
- prev = NULL;
- while (n < seqlen) {
- int attrlen = 0;
- curr = sdp_extract_attr(p, bufsize, &attrlen, rec);
- if (curr == NULL)
- break;
- if (prev)
- prev->next = curr;
- else
- d->val.dataseq = curr;
- prev = curr;
- p += attrlen;
- n += attrlen;
- bufsize -= attrlen;
- SDPDBG("Extracted: %d SequenceLength: %d", n, seqlen);
- }
- *len += n;
- return d;
- }
- sdp_data_t *sdp_extract_attr(const uint8_t *p, int bufsize, int *size,
- sdp_record_t *rec)
- {
- sdp_data_t *elem;
- int n = 0;
- uint8_t dtd;
- if (bufsize < (int) sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet");
- return NULL;
- }
- dtd = *(const uint8_t *)p;
- SDPDBG("extract_attr: dtd=0x%x", dtd);
- switch (dtd) {
- case SDP_DATA_NIL:
- case SDP_BOOL:
- case SDP_UINT8:
- case SDP_UINT16:
- case SDP_UINT32:
- case SDP_UINT64:
- case SDP_UINT128:
- case SDP_INT8:
- case SDP_INT16:
- case SDP_INT32:
- case SDP_INT64:
- case SDP_INT128:
- elem = extract_int(p, bufsize, &n);
- break;
- case SDP_UUID16:
- case SDP_UUID32:
- case SDP_UUID128:
- elem = extract_uuid(p, bufsize, &n, rec);
- break;
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- case SDP_TEXT_STR32:
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_URL_STR32:
- elem = extract_str(p, bufsize, &n);
- break;
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- elem = extract_seq(p, bufsize, &n, rec);
- break;
- default:
- SDPERR("Unknown data descriptor : 0x%x terminating", dtd);
- return NULL;
- }
- *size += n;
- return elem;
- }
- #ifdef SDP_DEBUG
- static void attr_print_func(void *value, void *userData)
- {
- sdp_data_t *d = (sdp_data_t *)value;
- SDPDBG("=====================================");
- SDPDBG("ATTRIBUTE IDENTIFIER : 0x%x", d->attrId);
- SDPDBG("ATTRIBUTE VALUE PTR : %p", value);
- if (d)
- sdp_data_print(d);
- else
- SDPDBG("NULL value");
- SDPDBG("=====================================");
- }
- void sdp_print_service_attr(sdp_list_t *svcAttrList)
- {
- SDPDBG("Printing service attr list %p", svcAttrList);
- sdp_list_foreach(svcAttrList, attr_print_func, NULL);
- SDPDBG("Printed service attr list %p", svcAttrList);
- }
- #endif
- sdp_record_t *sdp_extract_pdu(const uint8_t *buf, int bufsize, int *scanned)
- {
- int extracted = 0, seqlen = 0;
- uint8_t dtd;
- uint16_t attr;
- sdp_record_t *rec = sdp_record_alloc();
- const uint8_t *p = buf;
- *scanned = sdp_extract_seqtype(buf, bufsize, &dtd, &seqlen);
- p += *scanned;
- bufsize -= *scanned;
- rec->attrlist = NULL;
- while (extracted < seqlen && bufsize > 0) {
- int n = sizeof(uint8_t), attrlen = 0;
- sdp_data_t *data = NULL;
- SDPDBG("Extract PDU, sequenceLength: %d localExtractedLength: %d",
- seqlen, extracted);
- if (bufsize < n + (int) sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- break;
- }
- dtd = *(uint8_t *) p;
- attr = bt_get_be16(p + n);
- n += sizeof(uint16_t);
- SDPDBG("DTD of attrId : %d Attr id : 0x%x ", dtd, attr);
- data = sdp_extract_attr(p + n, bufsize - n, &attrlen, rec);
- SDPDBG("Attr id : 0x%x attrValueLength : %d", attr, attrlen);
- n += attrlen;
- if (data == NULL) {
- SDPDBG("Terminating extraction of attributes");
- break;
- }
- if (attr == SDP_ATTR_RECORD_HANDLE)
- rec->handle = data->val.uint32;
- if (attr == SDP_ATTR_SVCLASS_ID_LIST)
- extract_svclass_uuid(data, &rec->svclass);
- extracted += n;
- p += n;
- bufsize -= n;
- sdp_attr_replace(rec, attr, data);
- SDPDBG("Extract PDU, seqLength: %d localExtractedLength: %d",
- seqlen, extracted);
- }
- #ifdef SDP_DEBUG
- SDPDBG("Successful extracting of Svc Rec attributes");
- sdp_print_service_attr(rec->attrlist);
- #endif
- *scanned += seqlen;
- return rec;
- }
- static void sdp_copy_pattern(void *value, void *udata)
- {
- uuid_t *uuid = value;
- sdp_record_t *rec = udata;
- sdp_pattern_add_uuid(rec, uuid);
- }
- static void *sdp_data_value(sdp_data_t *data, uint32_t *len)
- {
- void *val = NULL;
- switch (data->dtd) {
- case SDP_DATA_NIL:
- break;
- case SDP_UINT8:
- val = &data->val.uint8;
- break;
- case SDP_INT8:
- case SDP_BOOL:
- val = &data->val.int8;
- break;
- case SDP_UINT16:
- val = &data->val.uint16;
- break;
- case SDP_INT16:
- val = &data->val.int16;
- break;
- case SDP_UINT32:
- val = &data->val.uint32;
- break;
- case SDP_INT32:
- val = &data->val.int32;
- break;
- case SDP_INT64:
- val = &data->val.int64;
- break;
- case SDP_UINT64:
- val = &data->val.uint64;
- break;
- case SDP_UINT128:
- val = &data->val.uint128;
- break;
- case SDP_INT128:
- val = &data->val.int128;
- break;
- case SDP_UUID16:
- val = &data->val.uuid.value.uuid16;
- break;
- case SDP_UUID32:
- val = &data->val.uuid.value.uuid32;
- break;
- case SDP_UUID128:
- val = &data->val.uuid.value.uuid128;
- break;
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- case SDP_URL_STR32:
- case SDP_TEXT_STR32:
- val = data->val.str;
- if (len)
- *len = data->unitSize - 1;
- break;
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- val = sdp_copy_seq(data->val.dataseq);
- break;
- }
- return val;
- }
- static sdp_data_t *sdp_copy_seq(sdp_data_t *data)
- {
- sdp_data_t *tmp, *seq = NULL, *cur = NULL;
- for (tmp = data; tmp; tmp = tmp->next) {
- sdp_data_t *datatmp;
- void *value;
- value = sdp_data_value(tmp, NULL);
- datatmp = sdp_data_alloc_with_length(tmp->dtd, value,
- tmp->unitSize);
- if (cur)
- cur->next = datatmp;
- else
- seq = datatmp;
- cur = datatmp;
- }
- return seq;
- }
- static void sdp_copy_attrlist(void *value, void *udata)
- {
- sdp_data_t *data = value;
- sdp_record_t *rec = udata;
- void *val;
- uint32_t len = 0;
- val = sdp_data_value(data, &len);
- if (!len)
- sdp_attr_add_new(rec, data->attrId, data->dtd, val);
- else
- sdp_attr_add_new_with_length(rec, data->attrId,
- data->dtd, val, len);
- }
- sdp_record_t *sdp_copy_record(sdp_record_t *rec)
- {
- sdp_record_t *cpy;
- cpy = sdp_record_alloc();
- cpy->handle = rec->handle;
- sdp_list_foreach(rec->pattern, sdp_copy_pattern, cpy);
- sdp_list_foreach(rec->attrlist, sdp_copy_attrlist, cpy);
- cpy->svclass = rec->svclass;
- return cpy;
- }
- #ifdef SDP_DEBUG
- static void print_dataseq(sdp_data_t *p)
- {
- sdp_data_t *d;
- for (d = p; d; d = d->next)
- sdp_data_print(d);
- }
- #endif
- void sdp_record_print(const sdp_record_t *rec)
- {
- sdp_data_t *d = sdp_data_get(rec, SDP_ATTR_SVCNAME_PRIMARY);
- if (d && SDP_IS_TEXT_STR(d->dtd))
- printf("Service Name: %.*s\n", d->unitSize, d->val.str);
- d = sdp_data_get(rec, SDP_ATTR_SVCDESC_PRIMARY);
- if (d && SDP_IS_TEXT_STR(d->dtd))
- printf("Service Description: %.*s\n", d->unitSize, d->val.str);
- d = sdp_data_get(rec, SDP_ATTR_PROVNAME_PRIMARY);
- if (d && SDP_IS_TEXT_STR(d->dtd))
- printf("Service Provider: %.*s\n", d->unitSize, d->val.str);
- }
- #ifdef SDP_DEBUG
- void sdp_data_print(sdp_data_t *d)
- {
- switch (d->dtd) {
- case SDP_DATA_NIL:
- SDPDBG("NIL");
- break;
- case SDP_BOOL:
- case SDP_UINT8:
- case SDP_UINT16:
- case SDP_UINT32:
- case SDP_UINT64:
- case SDP_UINT128:
- case SDP_INT8:
- case SDP_INT16:
- case SDP_INT32:
- case SDP_INT64:
- case SDP_INT128:
- SDPDBG("Integer : 0x%x", d->val.uint32);
- break;
- case SDP_UUID16:
- case SDP_UUID32:
- case SDP_UUID128:
- SDPDBG("UUID");
- sdp_uuid_print(&d->val.uuid);
- break;
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- case SDP_TEXT_STR32:
- SDPDBG("Text : %s", d->val.str);
- break;
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_URL_STR32:
- SDPDBG("URL : %s", d->val.str);
- break;
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- print_dataseq(d->val.dataseq);
- break;
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- SDPDBG("Data Sequence Alternates");
- print_dataseq(d->val.dataseq);
- break;
- }
- }
- #endif
- sdp_data_t *sdp_data_get(const sdp_record_t *rec, uint16_t attrId)
- {
- if (rec && rec->attrlist) {
- sdp_data_t sdpTemplate;
- sdp_list_t *p;
- sdpTemplate.attrId = attrId;
- p = sdp_list_find(rec->attrlist, &sdpTemplate, sdp_attrid_comp_func);
- if (p)
- return p->data;
- }
- return NULL;
- }
- static int sdp_send_req(sdp_session_t *session, uint8_t *buf, uint32_t size)
- {
- uint32_t sent = 0;
- while (sent < size) {
- int n = send(session->sock, buf + sent, size - sent, 0);
- if (n < 0)
- return -1;
- sent += n;
- }
- return 0;
- }
- static int sdp_read_rsp(sdp_session_t *session, uint8_t *buf, uint32_t size)
- {
- fd_set readFds;
- struct timeval timeout = { SDP_RESPONSE_TIMEOUT, 0 };
- FD_ZERO(&readFds);
- FD_SET(session->sock, &readFds);
- SDPDBG("Waiting for response");
- if (select(session->sock + 1, &readFds, NULL, NULL, &timeout) == 0) {
- SDPERR("Client timed out");
- errno = ETIMEDOUT;
- return -1;
- }
- return recv(session->sock, buf, size, 0);
- }
- /*
- * generic send request, wait for response method.
- */
- int sdp_send_req_w4_rsp(sdp_session_t *session, uint8_t *reqbuf,
- uint8_t *rspbuf, uint32_t reqsize, uint32_t *rspsize)
- {
- int n;
- sdp_pdu_hdr_t *reqhdr = (sdp_pdu_hdr_t *) reqbuf;
- sdp_pdu_hdr_t *rsphdr = (sdp_pdu_hdr_t *) rspbuf;
- SDPDBG("");
- if (0 > sdp_send_req(session, reqbuf, reqsize)) {
- SDPERR("Error sending data:%m");
- return -1;
- }
- n = sdp_read_rsp(session, rspbuf, SDP_RSP_BUFFER_SIZE);
- if (0 > n)
- return -1;
- SDPDBG("Read : %d", n);
- if (n == 0 || reqhdr->tid != rsphdr->tid) {
- errno = EPROTO;
- return -1;
- }
- *rspsize = n;
- return 0;
- }
- /*
- * singly-linked lists (after openobex implementation)
- */
- sdp_list_t *sdp_list_append(sdp_list_t *p, void *d)
- {
- sdp_list_t *q, *n = malloc(sizeof(sdp_list_t));
- if (!n)
- return NULL;
- n->data = d;
- n->next = 0;
- if (!p)
- return n;
- for (q = p; q->next; q = q->next);
- q->next = n;
- return p;
- }
- sdp_list_t *sdp_list_remove(sdp_list_t *list, void *d)
- {
- sdp_list_t *p, *q;
- for (q = 0, p = list; p; q = p, p = p->next)
- if (p->data == d) {
- if (q)
- q->next = p->next;
- else
- list = p->next;
- free(p);
- break;
- }
- return list;
- }
- sdp_list_t *sdp_list_insert_sorted(sdp_list_t *list, void *d,
- sdp_comp_func_t f)
- {
- sdp_list_t *q, *p, *n;
- n = malloc(sizeof(sdp_list_t));
- if (!n)
- return NULL;
- n->data = d;
- for (q = 0, p = list; p; q = p, p = p->next)
- if (f(p->data, d) >= 0)
- break;
- /* insert between q and p; if !q insert at head */
- if (q)
- q->next = n;
- else
- list = n;
- n->next = p;
- return list;
- }
- /*
- * Every element of the list points to things which need
- * to be free()'d. This method frees the list's contents
- */
- void sdp_list_free(sdp_list_t *list, sdp_free_func_t f)
- {
- sdp_list_t *next;
- while (list) {
- next = list->next;
- if (f)
- f(list->data);
- free(list);
- list = next;
- }
- }
- static inline int __find_port(sdp_data_t *seq, int proto)
- {
- if (!seq || !seq->next)
- return 0;
- if (SDP_IS_UUID(seq->dtd) && sdp_uuid_to_proto(&seq->val.uuid) == proto) {
- seq = seq->next;
- switch (seq->dtd) {
- case SDP_UINT8:
- return seq->val.uint8;
- case SDP_UINT16:
- return seq->val.uint16;
- }
- }
- return 0;
- }
- int sdp_get_proto_port(const sdp_list_t *list, int proto)
- {
- if (proto != L2CAP_UUID && proto != RFCOMM_UUID) {
- errno = EINVAL;
- return -1;
- }
- for (; list; list = list->next) {
- sdp_list_t *p;
- for (p = list->data; p; p = p->next) {
- sdp_data_t *seq = p->data;
- int port = __find_port(seq, proto);
- if (port)
- return port;
- }
- }
- return 0;
- }
- sdp_data_t *sdp_get_proto_desc(sdp_list_t *list, int proto)
- {
- for (; list; list = list->next) {
- sdp_list_t *p;
- for (p = list->data; p; p = p->next) {
- sdp_data_t *seq = p->data;
- if (SDP_IS_UUID(seq->dtd) &&
- sdp_uuid_to_proto(&seq->val.uuid) == proto)
- return seq->next;
- }
- }
- return NULL;
- }
- static int sdp_get_proto_descs(uint16_t attr_id, const sdp_record_t *rec,
- sdp_list_t **pap)
- {
- sdp_data_t *pdlist, *curr;
- sdp_list_t *ap = NULL;
- pdlist = sdp_data_get(rec, attr_id);
- if (pdlist == NULL) {
- errno = ENODATA;
- return -1;
- }
- SDPDBG("Attribute value type: 0x%02x", pdlist->dtd);
- if (attr_id == SDP_ATTR_ADD_PROTO_DESC_LIST) {
- if (!SDP_IS_SEQ(pdlist->dtd)) {
- errno = EINVAL;
- return -1;
- }
- pdlist = pdlist->val.dataseq;
- }
- for (; pdlist; pdlist = pdlist->next) {
- sdp_list_t *pds = NULL;
- if (!SDP_IS_SEQ(pdlist->dtd) && !SDP_IS_ALT(pdlist->dtd))
- goto failed;
- for (curr = pdlist->val.dataseq; curr; curr = curr->next) {
- if (!SDP_IS_SEQ(curr->dtd)) {
- sdp_list_free(pds, NULL);
- goto failed;
- }
- pds = sdp_list_append(pds, curr->val.dataseq);
- }
- ap = sdp_list_append(ap, pds);
- }
- *pap = ap;
- return 0;
- failed:
- sdp_list_foreach(ap, (sdp_list_func_t) sdp_list_free, NULL);
- sdp_list_free(ap, NULL);
- errno = EINVAL;
- return -1;
- }
- int sdp_get_access_protos(const sdp_record_t *rec, sdp_list_t **pap)
- {
- return sdp_get_proto_descs(SDP_ATTR_PROTO_DESC_LIST, rec, pap);
- }
- int sdp_get_add_access_protos(const sdp_record_t *rec, sdp_list_t **pap)
- {
- return sdp_get_proto_descs(SDP_ATTR_ADD_PROTO_DESC_LIST, rec, pap);
- }
- int sdp_get_uuidseq_attr(const sdp_record_t *rec, uint16_t attr,
- sdp_list_t **seqp)
- {
- sdp_data_t *sdpdata = sdp_data_get(rec, attr);
- *seqp = NULL;
- if (sdpdata && SDP_IS_SEQ(sdpdata->dtd)) {
- sdp_data_t *d;
- for (d = sdpdata->val.dataseq; d; d = d->next) {
- uuid_t *u;
- if (d->dtd < SDP_UUID16 || d->dtd > SDP_UUID128) {
- errno = EINVAL;
- goto fail;
- }
- u = malloc(sizeof(uuid_t));
- if (!u)
- goto fail;
- *u = d->val.uuid;
- *seqp = sdp_list_append(*seqp, u);
- }
- return 0;
- }
- fail:
- sdp_list_free(*seqp, free);
- *seqp = NULL;
- return -1;
- }
- int sdp_set_uuidseq_attr(sdp_record_t *rec, uint16_t aid, sdp_list_t *seq)
- {
- int status = 0, i, len;
- void **dtds, **values;
- uint8_t uuid16 = SDP_UUID16;
- uint8_t uuid32 = SDP_UUID32;
- uint8_t uuid128 = SDP_UUID128;
- sdp_list_t *p;
- len = sdp_list_len(seq);
- if (!seq || len == 0)
- return -1;
- dtds = malloc(len * sizeof(void *));
- if (!dtds)
- return -1;
- values = malloc(len * sizeof(void *));
- if (!values) {
- free(dtds);
- return -1;
- }
- for (p = seq, i = 0; i < len; i++, p = p->next) {
- uuid_t *uuid = p->data;
- if (uuid)
- switch (uuid->type) {
- case SDP_UUID16:
- dtds[i] = &uuid16;
- values[i] = &uuid->value.uuid16;
- break;
- case SDP_UUID32:
- dtds[i] = &uuid32;
- values[i] = &uuid->value.uuid32;
- break;
- case SDP_UUID128:
- dtds[i] = &uuid128;
- values[i] = &uuid->value.uuid128;
- break;
- default:
- status = -1;
- break;
- }
- else {
- status = -1;
- break;
- }
- }
- if (status == 0) {
- sdp_data_t *data = sdp_seq_alloc(dtds, values, len);
- sdp_attr_replace(rec, aid, data);
- sdp_pattern_add_uuidseq(rec, seq);
- }
- free(dtds);
- free(values);
- return status;
- }
- int sdp_get_lang_attr(const sdp_record_t *rec, sdp_list_t **langSeq)
- {
- sdp_lang_attr_t *lang;
- sdp_data_t *sdpdata, *curr_data;
- *langSeq = NULL;
- sdpdata = sdp_data_get(rec, SDP_ATTR_LANG_BASE_ATTR_ID_LIST);
- if (sdpdata == NULL) {
- errno = ENODATA;
- return -1;
- }
- if (!SDP_IS_SEQ(sdpdata->dtd))
- goto invalid;
- curr_data = sdpdata->val.dataseq;
- while (curr_data) {
- sdp_data_t *pCode, *pEncoding, *pOffset;
- pCode = curr_data;
- if (pCode->dtd != SDP_UINT16)
- goto invalid;
- /* LanguageBaseAttributeIDList entries are always grouped as
- * triplets */
- if (!pCode->next || !pCode->next->next)
- goto invalid;
- pEncoding = pCode->next;
- if (pEncoding->dtd != SDP_UINT16)
- goto invalid;
- pOffset = pEncoding->next;
- if (pOffset->dtd != SDP_UINT16)
- goto invalid;
- lang = malloc(sizeof(sdp_lang_attr_t));
- if (!lang) {
- sdp_list_free(*langSeq, free);
- *langSeq = NULL;
- return -1;
- }
- lang->code_ISO639 = pCode->val.uint16;
- lang->encoding = pEncoding->val.uint16;
- lang->base_offset = pOffset->val.uint16;
- SDPDBG("code_ISO639 : 0x%02x", lang->code_ISO639);
- SDPDBG("encoding : 0x%02x", lang->encoding);
- SDPDBG("base_offfset : 0x%02x", lang->base_offset);
- *langSeq = sdp_list_append(*langSeq, lang);
- curr_data = pOffset->next;
- }
- return 0;
- invalid:
- sdp_list_free(*langSeq, free);
- *langSeq = NULL;
- errno = EINVAL;
- return -1;
- }
- int sdp_get_profile_descs(const sdp_record_t *rec, sdp_list_t **profDescSeq)
- {
- sdp_profile_desc_t *profDesc;
- sdp_data_t *sdpdata, *seq;
- *profDescSeq = NULL;
- sdpdata = sdp_data_get(rec, SDP_ATTR_PFILE_DESC_LIST);
- if (sdpdata == NULL) {
- errno = ENODATA;
- return -1;
- }
- if (!SDP_IS_SEQ(sdpdata->dtd) || sdpdata->val.dataseq == NULL)
- goto invalid;
- for (seq = sdpdata->val.dataseq; seq; seq = seq->next) {
- uuid_t *uuid = NULL;
- uint16_t version = 0x100;
- if (SDP_IS_UUID(seq->dtd)) {
- /* Mac OS X 10.7.3 and old Samsung phones do not comply
- * to the SDP specification for
- * BluetoothProfileDescriptorList. This workaround
- * allows to properly parse UUID/version from SDP
- * record published by these systems. */
- sdp_data_t *next = seq->next;
- uuid = &seq->val.uuid;
- if (next && next->dtd == SDP_UINT16) {
- version = next->val.uint16;
- seq = next;
- }
- } else if (SDP_IS_SEQ(seq->dtd)) {
- sdp_data_t *puuid, *pVnum;
- puuid = seq->val.dataseq;
- if (puuid == NULL || !SDP_IS_UUID(puuid->dtd))
- goto invalid;
- uuid = &puuid->val.uuid;
- pVnum = puuid->next;
- if (pVnum == NULL || pVnum->dtd != SDP_UINT16)
- goto invalid;
- version = pVnum->val.uint16;
- } else
- goto invalid;
- if (uuid != NULL) {
- profDesc = malloc(sizeof(sdp_profile_desc_t));
- if (!profDesc) {
- sdp_list_free(*profDescSeq, free);
- *profDescSeq = NULL;
- return -1;
- }
- profDesc->uuid = *uuid;
- profDesc->version = version;
- #ifdef SDP_DEBUG
- sdp_uuid_print(&profDesc->uuid);
- SDPDBG("Vnum : 0x%04x", profDesc->version);
- #endif
- *profDescSeq = sdp_list_append(*profDescSeq, profDesc);
- }
- }
- return 0;
- invalid:
- sdp_list_free(*profDescSeq, free);
- *profDescSeq = NULL;
- errno = EINVAL;
- return -1;
- }
- int sdp_get_server_ver(const sdp_record_t *rec, sdp_list_t **u16)
- {
- sdp_data_t *d, *curr;
- *u16 = NULL;
- d = sdp_data_get(rec, SDP_ATTR_VERSION_NUM_LIST);
- if (d == NULL) {
- errno = ENODATA;
- return -1;
- }
- if (!SDP_IS_SEQ(d->dtd) || d->val.dataseq == NULL)
- goto invalid;
- for (curr = d->val.dataseq; curr; curr = curr->next) {
- if (curr->dtd != SDP_UINT16)
- goto invalid;
- *u16 = sdp_list_append(*u16, &curr->val.uint16);
- }
- return 0;
- invalid:
- sdp_list_free(*u16, NULL);
- *u16 = NULL;
- errno = EINVAL;
- return -1;
- }
- /* flexible extraction of basic attributes - Jean II */
- /* How do we expect caller to extract predefined data sequences? */
- int sdp_get_int_attr(const sdp_record_t *rec, uint16_t attrid, int *value)
- {
- sdp_data_t *sdpdata = sdp_data_get(rec, attrid);
- if (sdpdata)
- /* Verify that it is what the caller expects */
- if (sdpdata->dtd == SDP_BOOL || sdpdata->dtd == SDP_UINT8 ||
- sdpdata->dtd == SDP_UINT16 || sdpdata->dtd == SDP_UINT32 ||
- sdpdata->dtd == SDP_INT8 || sdpdata->dtd == SDP_INT16 ||
- sdpdata->dtd == SDP_INT32) {
- *value = sdpdata->val.uint32;
- return 0;
- }
- errno = EINVAL;
- return -1;
- }
- int sdp_get_string_attr(const sdp_record_t *rec, uint16_t attrid, char *value,
- int valuelen)
- {
- sdp_data_t *sdpdata = sdp_data_get(rec, attrid);
- if (sdpdata)
- /* Verify that it is what the caller expects */
- if (SDP_IS_TEXT_STR(sdpdata->dtd))
- if ((int) strlen(sdpdata->val.str) < valuelen) {
- strcpy(value, sdpdata->val.str);
- return 0;
- }
- errno = EINVAL;
- return -1;
- }
- #define get_basic_attr(attrID, pAttrValue, fieldName) \
- sdp_data_t *data = sdp_data_get(rec, attrID); \
- if (data) { \
- *pAttrValue = data->val.fieldName; \
- return 0; \
- } \
- errno = EINVAL; \
- return -1;
- int sdp_get_service_id(const sdp_record_t *rec, uuid_t *uuid)
- {
- get_basic_attr(SDP_ATTR_SERVICE_ID, uuid, uuid);
- }
- int sdp_get_group_id(const sdp_record_t *rec, uuid_t *uuid)
- {
- get_basic_attr(SDP_ATTR_GROUP_ID, uuid, uuid);
- }
- int sdp_get_record_state(const sdp_record_t *rec, uint32_t *svcRecState)
- {
- get_basic_attr(SDP_ATTR_RECORD_STATE, svcRecState, uint32);
- }
- int sdp_get_service_avail(const sdp_record_t *rec, uint8_t *svcAvail)
- {
- get_basic_attr(SDP_ATTR_SERVICE_AVAILABILITY, svcAvail, uint8);
- }
- int sdp_get_service_ttl(const sdp_record_t *rec, uint32_t *svcTTLInfo)
- {
- get_basic_attr(SDP_ATTR_SVCINFO_TTL, svcTTLInfo, uint32);
- }
- int sdp_get_database_state(const sdp_record_t *rec, uint32_t *svcDBState)
- {
- get_basic_attr(SDP_ATTR_SVCDB_STATE, svcDBState, uint32);
- }
- /*
- * NOTE that none of the setXXX() functions below will
- * actually update the SDP server, unless the
- * {register, update}sdp_record_t() function is invoked.
- */
- int sdp_attr_add_new(sdp_record_t *rec, uint16_t attr, uint8_t dtd,
- const void *value)
- {
- sdp_data_t *d = sdp_data_alloc(dtd, value);
- if (d) {
- sdp_attr_replace(rec, attr, d);
- return 0;
- }
- return -1;
- }
- static int sdp_attr_add_new_with_length(sdp_record_t *rec,
- uint16_t attr, uint8_t dtd, const void *value, uint32_t len)
- {
- sdp_data_t *d;
- d = sdp_data_alloc_with_length(dtd, value, len);
- if (!d)
- return -1;
- sdp_attr_replace(rec, attr, d);
- return 0;
- }
- /*
- * Set the information attributes of the service
- * pointed to by rec. The attributes are
- * service name, description and provider name
- */
- void sdp_set_info_attr(sdp_record_t *rec, const char *name, const char *prov,
- const char *desc)
- {
- if (name)
- sdp_attr_add_new(rec, SDP_ATTR_SVCNAME_PRIMARY,
- SDP_TEXT_STR8, name);
- if (prov)
- sdp_attr_add_new(rec, SDP_ATTR_PROVNAME_PRIMARY,
- SDP_TEXT_STR8, prov);
- if (desc)
- sdp_attr_add_new(rec, SDP_ATTR_SVCDESC_PRIMARY,
- SDP_TEXT_STR8, desc);
- }
- static sdp_data_t *access_proto_to_dataseq(sdp_record_t *rec, sdp_list_t *proto)
- {
- sdp_data_t *seq = NULL;
- void *dtds[10], *values[10];
- void **seqDTDs, **seqs;
- int i, seqlen;
- sdp_list_t *p;
- seqlen = sdp_list_len(proto);
- seqDTDs = malloc(seqlen * sizeof(void *));
- if (!seqDTDs)
- return NULL;
- seqs = malloc(seqlen * sizeof(void *));
- if (!seqs) {
- free(seqDTDs);
- return NULL;
- }
- for (i = 0, p = proto; p; p = p->next, i++) {
- sdp_list_t *elt = p->data;
- sdp_data_t *s;
- uuid_t *uuid = NULL;
- unsigned int pslen = 0;
- for (; elt && pslen < ARRAY_SIZE(dtds); elt = elt->next, pslen++) {
- sdp_data_t *d = elt->data;
- dtds[pslen] = &d->dtd;
- switch (d->dtd) {
- case SDP_UUID16:
- uuid = (uuid_t *) d;
- values[pslen] = &uuid->value.uuid16;
- break;
- case SDP_UUID32:
- uuid = (uuid_t *) d;
- values[pslen] = &uuid->value.uuid32;
- break;
- case SDP_UUID128:
- uuid = (uuid_t *) d;
- values[pslen] = &uuid->value.uuid128;
- break;
- case SDP_UINT8:
- values[pslen] = &d->val.uint8;
- break;
- case SDP_UINT16:
- values[pslen] = &d->val.uint16;
- break;
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- values[pslen] = d;
- break;
- /* FIXME: more */
- }
- }
- s = sdp_seq_alloc(dtds, values, pslen);
- if (s) {
- seqDTDs[i] = &s->dtd;
- seqs[i] = s;
- if (uuid)
- sdp_pattern_add_uuid(rec, uuid);
- }
- }
- seq = sdp_seq_alloc(seqDTDs, seqs, seqlen);
- free(seqDTDs);
- free(seqs);
- return seq;
- }
- /*
- * sets the access protocols of the service specified
- * to the value specified in "access_proto"
- *
- * Note that if there are alternate mechanisms by
- * which the service is accessed, then they should
- * be specified as sequences
- *
- * Using a value of NULL for accessProtocols has
- * effect of removing this attribute (if previously set)
- *
- * This function replaces the existing sdp_access_proto_t
- * structure (if any) with the new one specified.
- *
- * returns 0 if successful or -1 if there is a failure.
- */
- int sdp_set_access_protos(sdp_record_t *rec, const sdp_list_t *ap)
- {
- const sdp_list_t *p;
- sdp_data_t *protos = NULL;
- for (p = ap; p; p = p->next) {
- sdp_data_t *seq = access_proto_to_dataseq(rec, p->data);
- protos = sdp_seq_append(protos, seq);
- }
- sdp_attr_add(rec, SDP_ATTR_PROTO_DESC_LIST, protos);
- return 0;
- }
- int sdp_set_add_access_protos(sdp_record_t *rec, const sdp_list_t *ap)
- {
- const sdp_list_t *p;
- sdp_data_t *protos = NULL;
- for (p = ap; p; p = p->next) {
- sdp_data_t *seq = access_proto_to_dataseq(rec, p->data);
- protos = sdp_seq_append(protos, seq);
- }
- sdp_attr_add(rec, SDP_ATTR_ADD_PROTO_DESC_LIST,
- protos ? sdp_data_alloc(SDP_SEQ8, protos) : NULL);
- return 0;
- }
- /*
- * set the "LanguageBase" attributes of the service record
- * record to the value specified in "langAttrList".
- *
- * "langAttrList" is a linked list of "sdp_lang_attr_t"
- * objects, one for each language in which user visible
- * attributes are present in the service record.
- *
- * Using a value of NULL for langAttrList has
- * effect of removing this attribute (if previously set)
- *
- * This function replaces the exisiting sdp_lang_attr_t
- * structure (if any) with the new one specified.
- *
- * returns 0 if successful or -1 if there is a failure.
- */
- int sdp_set_lang_attr(sdp_record_t *rec, const sdp_list_t *seq)
- {
- uint8_t uint16 = SDP_UINT16;
- int status = 0, i = 0, seqlen = sdp_list_len(seq);
- void **dtds, **values;
- const sdp_list_t *p;
- dtds = malloc(3 * seqlen * sizeof(void *));
- if (!dtds)
- return -1;
- values = malloc(3 * seqlen * sizeof(void *));
- if (!values) {
- free(dtds);
- return -1;
- }
- for (p = seq; p; p = p->next) {
- sdp_lang_attr_t *lang = p->data;
- if (!lang) {
- status = -1;
- break;
- }
- dtds[i] = &uint16;
- values[i] = &lang->code_ISO639;
- i++;
- dtds[i] = &uint16;
- values[i] = &lang->encoding;
- i++;
- dtds[i] = &uint16;
- values[i] = &lang->base_offset;
- i++;
- }
- if (status == 0) {
- sdp_data_t *seq = sdp_seq_alloc(dtds, values, 3 * seqlen);
- sdp_attr_add(rec, SDP_ATTR_LANG_BASE_ATTR_ID_LIST, seq);
- }
- free(dtds);
- free(values);
- return status;
- }
- /*
- * set the "ServiceID" attribute of the service.
- *
- * This is the UUID of the service.
- *
- * returns 0 if successful or -1 if there is a failure.
- */
- void sdp_set_service_id(sdp_record_t *rec, uuid_t uuid)
- {
- switch (uuid.type) {
- case SDP_UUID16:
- sdp_attr_add_new(rec, SDP_ATTR_SERVICE_ID, SDP_UUID16,
- &uuid.value.uuid16);
- break;
- case SDP_UUID32:
- sdp_attr_add_new(rec, SDP_ATTR_SERVICE_ID, SDP_UUID32,
- &uuid.value.uuid32);
- break;
- case SDP_UUID128:
- sdp_attr_add_new(rec, SDP_ATTR_SERVICE_ID, SDP_UUID128,
- &uuid.value.uuid128);
- break;
- }
- sdp_pattern_add_uuid(rec, &uuid);
- }
- /*
- * set the GroupID attribute of the service record defining a group.
- *
- * This is the UUID of the group.
- *
- * returns 0 if successful or -1 if there is a failure.
- */
- void sdp_set_group_id(sdp_record_t *rec, uuid_t uuid)
- {
- switch (uuid.type) {
- case SDP_UUID16:
- sdp_attr_add_new(rec, SDP_ATTR_GROUP_ID, SDP_UUID16,
- &uuid.value.uuid16);
- break;
- case SDP_UUID32:
- sdp_attr_add_new(rec, SDP_ATTR_GROUP_ID, SDP_UUID32,
- &uuid.value.uuid32);
- break;
- case SDP_UUID128:
- sdp_attr_add_new(rec, SDP_ATTR_GROUP_ID, SDP_UUID128,
- &uuid.value.uuid128);
- break;
- }
- sdp_pattern_add_uuid(rec, &uuid);
- }
- /*
- * set the ProfileDescriptorList attribute of the service record
- * pointed to by record to the value specified in "profileDesc".
- *
- * Each element in the list is an object of type
- * sdp_profile_desc_t which is a definition of the
- * Bluetooth profile that this service conforms to.
- *
- * Using a value of NULL for profileDesc has
- * effect of removing this attribute (if previously set)
- *
- * This function replaces the exisiting ProfileDescriptorList
- * structure (if any) with the new one specified.
- *
- * returns 0 if successful or -1 if there is a failure.
- */
- int sdp_set_profile_descs(sdp_record_t *rec, const sdp_list_t *profiles)
- {
- int status = 0;
- uint8_t uuid16 = SDP_UUID16;
- uint8_t uuid32 = SDP_UUID32;
- uint8_t uuid128 = SDP_UUID128;
- uint8_t uint16 = SDP_UINT16;
- int i = 0, seqlen = sdp_list_len(profiles);
- void **seqDTDs, **seqs;
- const sdp_list_t *p;
- sdp_data_t *pAPSeq;
- seqDTDs = malloc(seqlen * sizeof(void *));
- if (!seqDTDs)
- return -1;
- seqs = malloc(seqlen * sizeof(void *));
- if (!seqs) {
- free(seqDTDs);
- return -1;
- }
- for (p = profiles; p; p = p->next) {
- sdp_data_t *seq;
- void *dtds[2], *values[2];
- sdp_profile_desc_t *profile = p->data;
- if (!profile) {
- status = -1;
- goto end;
- }
- switch (profile->uuid.type) {
- case SDP_UUID16:
- dtds[0] = &uuid16;
- values[0] = &profile->uuid.value.uuid16;
- break;
- case SDP_UUID32:
- dtds[0] = &uuid32;
- values[0] = &profile->uuid.value.uuid32;
- break;
- case SDP_UUID128:
- dtds[0] = &uuid128;
- values[0] = &profile->uuid.value.uuid128;
- break;
- default:
- status = -1;
- goto end;
- }
- dtds[1] = &uint16;
- values[1] = &profile->version;
- seq = sdp_seq_alloc(dtds, values, 2);
- if (seq == NULL) {
- status = -1;
- goto end;
- }
- seqDTDs[i] = &seq->dtd;
- seqs[i] = seq;
- sdp_pattern_add_uuid(rec, &profile->uuid);
- i++;
- }
- pAPSeq = sdp_seq_alloc(seqDTDs, seqs, seqlen);
- sdp_attr_add(rec, SDP_ATTR_PFILE_DESC_LIST, pAPSeq);
- end:
- free(seqDTDs);
- free(seqs);
- return status;
- }
- /*
- * sets various URL attributes of the service
- * pointed to by record. The URL include
- *
- * client: a URL to the client's
- * platform specific (WinCE, PalmOS) executable
- * code that can be used to access this service.
- *
- * doc: a URL pointing to service documentation
- *
- * icon: a URL to an icon that can be used to represent
- * this service.
- *
- * Note that you need to pass NULL for any URLs
- * that you don't want to set or remove
- */
- void sdp_set_url_attr(sdp_record_t *rec, const char *client, const char *doc,
- const char *icon)
- {
- sdp_attr_add_new(rec, SDP_ATTR_CLNT_EXEC_URL, SDP_URL_STR8, client);
- sdp_attr_add_new(rec, SDP_ATTR_DOC_URL, SDP_URL_STR8, doc);
- sdp_attr_add_new(rec, SDP_ATTR_ICON_URL, SDP_URL_STR8, icon);
- }
- uuid_t *sdp_uuid16_create(uuid_t *u, uint16_t val)
- {
- memset(u, 0, sizeof(uuid_t));
- u->type = SDP_UUID16;
- u->value.uuid16 = val;
- return u;
- }
- uuid_t *sdp_uuid32_create(uuid_t *u, uint32_t val)
- {
- memset(u, 0, sizeof(uuid_t));
- u->type = SDP_UUID32;
- u->value.uuid32 = val;
- return u;
- }
- uuid_t *sdp_uuid128_create(uuid_t *u, const void *val)
- {
- memset(u, 0, sizeof(uuid_t));
- u->type = SDP_UUID128;
- memcpy(&u->value.uuid128, val, sizeof(uint128_t));
- return u;
- }
- /*
- * UUID comparison function
- * returns 0 if uuidValue1 == uuidValue2 else -1
- */
- int sdp_uuid_cmp(const void *p1, const void *p2)
- {
- uuid_t *u1 = sdp_uuid_to_uuid128(p1);
- uuid_t *u2 = sdp_uuid_to_uuid128(p2);
- int ret;
- ret = sdp_uuid128_cmp(u1, u2);
- bt_free(u1);
- bt_free(u2);
- return ret;
- }
- /*
- * UUID comparison function
- * returns 0 if uuidValue1 == uuidValue2 else -1
- */
- int sdp_uuid16_cmp(const void *p1, const void *p2)
- {
- const uuid_t *u1 = p1;
- const uuid_t *u2 = p2;
- return memcmp(&u1->value.uuid16, &u2->value.uuid16, sizeof(uint16_t));
- }
- /*
- * UUID comparison function
- * returns 0 if uuidValue1 == uuidValue2 else -1
- */
- int sdp_uuid128_cmp(const void *p1, const void *p2)
- {
- const uuid_t *u1 = p1;
- const uuid_t *u2 = p2;
- return memcmp(&u1->value.uuid128, &u2->value.uuid128, sizeof(uint128_t));
- }
- /*
- * 128 to 16 bit and 32 to 16 bit UUID conversion functions
- * yet to be implemented. Note that the input is in NBO in
- * both 32 and 128 bit UUIDs and conversion is needed
- */
- void sdp_uuid16_to_uuid128(uuid_t *uuid128, const uuid_t *uuid16)
- {
- /*
- * We have a 16 bit value, which needs to be added to
- * bytes 3 and 4 (at indices 2 and 3) of the Bluetooth base
- */
- unsigned short data1;
- /* allocate a 128bit UUID and init to the Bluetooth base UUID */
- uuid128->value.uuid128 = bluetooth_base_uuid;
- uuid128->type = SDP_UUID128;
- /* extract bytes 2 and 3 of 128bit BT base UUID */
- memcpy(&data1, &bluetooth_base_uuid.data[2], 2);
- /* add the given UUID (16 bits) */
- data1 += htons(uuid16->value.uuid16);
- /* set bytes 2 and 3 of the 128 bit value */
- memcpy(&uuid128->value.uuid128.data[2], &data1, 2);
- }
- void sdp_uuid32_to_uuid128(uuid_t *uuid128, const uuid_t *uuid32)
- {
- /*
- * We have a 32 bit value, which needs to be added to
- * bytes 1->4 (at indices 0 thru 3) of the Bluetooth base
- */
- unsigned int data0;
- /* allocate a 128bit UUID and init to the Bluetooth base UUID */
- uuid128->value.uuid128 = bluetooth_base_uuid;
- uuid128->type = SDP_UUID128;
- /* extract first 4 bytes */
- memcpy(&data0, &bluetooth_base_uuid.data[0], 4);
- /* add the given UUID (32bits) */
- data0 += htonl(uuid32->value.uuid32);
- /* set the 4 bytes of the 128 bit value */
- memcpy(&uuid128->value.uuid128.data[0], &data0, 4);
- }
- uuid_t *sdp_uuid_to_uuid128(const uuid_t *uuid)
- {
- uuid_t *uuid128 = bt_malloc0(sizeof(uuid_t));
- if (!uuid128)
- return NULL;
- switch (uuid->type) {
- case SDP_UUID128:
- *uuid128 = *uuid;
- break;
- case SDP_UUID32:
- sdp_uuid32_to_uuid128(uuid128, uuid);
- break;
- case SDP_UUID16:
- sdp_uuid16_to_uuid128(uuid128, uuid);
- break;
- }
- return uuid128;
- }
- /*
- * converts a 128-bit uuid to a 16/32-bit one if possible
- * returns true if uuid contains a 16/32-bit UUID at exit
- */
- int sdp_uuid128_to_uuid(uuid_t *uuid)
- {
- uint128_t *b = &bluetooth_base_uuid;
- uint128_t *u = &uuid->value.uuid128;
- uint32_t data;
- unsigned int i;
- if (uuid->type != SDP_UUID128)
- return 1;
- for (i = 4; i < sizeof(b->data); i++)
- if (b->data[i] != u->data[i])
- return 0;
- memcpy(&data, u->data, 4);
- data = htonl(data);
- if (data <= 0xffff) {
- uuid->type = SDP_UUID16;
- uuid->value.uuid16 = (uint16_t) data;
- } else {
- uuid->type = SDP_UUID32;
- uuid->value.uuid32 = data;
- }
- return 1;
- }
- /*
- * convert a UUID to the 16-bit short-form
- */
- int sdp_uuid_to_proto(uuid_t *uuid)
- {
- uuid_t u = *uuid;
- if (sdp_uuid128_to_uuid(&u)) {
- switch (u.type) {
- case SDP_UUID16:
- return u.value.uuid16;
- case SDP_UUID32:
- return u.value.uuid32;
- }
- }
- return 0;
- }
- /*
- * This function appends data to the PDU buffer "dst" from source "src".
- * The data length is also computed and set.
- * Should the PDU length exceed 2^8, then sequence type is
- * set accordingly and the data is memmove()'d.
- */
- void sdp_append_to_buf(sdp_buf_t *dst, uint8_t *data, uint32_t len)
- {
- uint8_t *p = dst->data;
- uint8_t dtd = *p;
- SDPDBG("Append src size: %d", len);
- SDPDBG("Append dst size: %d", dst->data_size);
- SDPDBG("Dst buffer size: %d", dst->buf_size);
- if (dst->data_size + len > dst->buf_size) {
- SDPERR("Cannot append");
- return;
- }
- if (dst->data_size == 0 && dtd == 0) {
- /* create initial sequence */
- *p = SDP_SEQ8;
- dst->data_size += sizeof(uint8_t);
- /* reserve space for sequence size */
- dst->data_size += sizeof(uint8_t);
- }
- memcpy(dst->data + dst->data_size, data, len);
- dst->data_size += len;
- dtd = *(uint8_t *) dst->data;
- if (dst->data_size > UCHAR_MAX && dtd == SDP_SEQ8) {
- short offset = sizeof(uint8_t) + sizeof(uint8_t);
- memmove(dst->data + offset + 1, dst->data + offset,
- dst->data_size - offset);
- *p = SDP_SEQ16;
- dst->data_size += 1;
- }
- dtd = *(uint8_t *) p;
- p += sizeof(uint8_t);
- switch (dtd) {
- case SDP_SEQ8:
- *(uint8_t *) p = dst->data_size - sizeof(uint8_t) - sizeof(uint8_t);
- break;
- case SDP_SEQ16:
- bt_put_be16(dst->data_size - sizeof(uint8_t) - sizeof(uint16_t), p);
- break;
- case SDP_SEQ32:
- bt_put_be32(dst->data_size - sizeof(uint8_t) - sizeof(uint32_t), p);
- break;
- }
- }
- void sdp_append_to_pdu(sdp_buf_t *pdu, sdp_data_t *d)
- {
- sdp_buf_t append;
- memset(&append, 0, sizeof(sdp_buf_t));
- sdp_gen_buffer(&append, d);
- append.data = malloc(append.buf_size);
- if (!append.data)
- return;
- sdp_set_attrid(&append, d->attrId);
- sdp_gen_pdu(&append, d);
- sdp_append_to_buf(pdu, append.data, append.data_size);
- free(append.data);
- }
- /*
- * Registers an sdp record.
- *
- * It is incorrect to call this method on a record that
- * has been already registered with the server.
- *
- * Returns zero on success, otherwise -1 (and sets errno).
- */
- int sdp_device_record_register_binary(sdp_session_t *session, bdaddr_t *device, uint8_t *data, uint32_t size, uint8_t flags, uint32_t *handle)
- {
- uint8_t *req, *rsp, *p;
- uint32_t reqsize, rspsize;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- int status;
- SDPDBG("");
- if (!session->local) {
- errno = EREMOTE;
- return -1;
- }
- req = malloc(SDP_REQ_BUFFER_SIZE);
- rsp = malloc(SDP_RSP_BUFFER_SIZE);
- if (req == NULL || rsp == NULL) {
- status = -1;
- errno = ENOMEM;
- goto end;
- }
- reqhdr = (sdp_pdu_hdr_t *)req;
- reqhdr->pdu_id = SDP_SVC_REGISTER_REQ;
- reqhdr->tid = htons(sdp_gen_tid(session));
- reqsize = sizeof(sdp_pdu_hdr_t) + 1;
- p = req + sizeof(sdp_pdu_hdr_t);
- if (bacmp(device, BDADDR_ANY)) {
- *p++ = flags | SDP_DEVICE_RECORD;
- bacpy((bdaddr_t *) p, device);
- p += sizeof(bdaddr_t);
- reqsize += sizeof(bdaddr_t);
- } else
- *p++ = flags;
- memcpy(p, data, size);
- reqsize += size;
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- status = sdp_send_req_w4_rsp(session, req, rsp, reqsize, &rspsize);
- if (status < 0)
- goto end;
- if (rspsize < sizeof(sdp_pdu_hdr_t)) {
- SDPERR("Unexpected end of packet");
- errno = EPROTO;
- status = -1;
- goto end;
- }
- rsphdr = (sdp_pdu_hdr_t *) rsp;
- p = rsp + sizeof(sdp_pdu_hdr_t);
- if (rsphdr->pdu_id == SDP_ERROR_RSP) {
- /* Invalid service record */
- errno = EINVAL;
- status = -1;
- } else if (rsphdr->pdu_id != SDP_SVC_REGISTER_RSP) {
- errno = EPROTO;
- status = -1;
- } else {
- if (rspsize < sizeof(sdp_pdu_hdr_t) + sizeof(uint32_t)) {
- SDPERR("Unexpected end of packet");
- errno = EPROTO;
- status = -1;
- goto end;
- }
- if (handle)
- *handle = bt_get_be32(p);
- }
- end:
- free(req);
- free(rsp);
- return status;
- }
- int sdp_device_record_register(sdp_session_t *session, bdaddr_t *device, sdp_record_t *rec, uint8_t flags)
- {
- sdp_buf_t pdu;
- uint32_t handle;
- int err;
- SDPDBG("");
- if (rec->handle && rec->handle != 0xffffffff) {
- uint32_t handle = rec->handle;
- sdp_data_t *data = sdp_data_alloc(SDP_UINT32, &handle);
- sdp_attr_replace(rec, SDP_ATTR_RECORD_HANDLE, data);
- }
- if (sdp_gen_record_pdu(rec, &pdu) < 0) {
- errno = ENOMEM;
- return -1;
- }
- err = sdp_device_record_register_binary(session, device,
- pdu.data, pdu.data_size, flags, &handle);
- free(pdu.data);
- if (err == 0) {
- sdp_data_t *data = sdp_data_alloc(SDP_UINT32, &handle);
- rec->handle = handle;
- sdp_attr_replace(rec, SDP_ATTR_RECORD_HANDLE, data);
- }
- return err;
- }
- int sdp_record_register(sdp_session_t *session, sdp_record_t *rec, uint8_t flags)
- {
- return sdp_device_record_register(session, BDADDR_ANY, rec, flags);
- }
- /*
- * unregister a service record
- */
- int sdp_device_record_unregister_binary(sdp_session_t *session, bdaddr_t *device, uint32_t handle)
- {
- uint8_t *reqbuf, *rspbuf, *p;
- uint32_t reqsize = 0, rspsize = 0;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- int status;
- SDPDBG("");
- if (handle == SDP_SERVER_RECORD_HANDLE) {
- errno = EINVAL;
- return -1;
- }
- if (!session->local) {
- errno = EREMOTE;
- return -1;
- }
- reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- rspbuf = malloc(SDP_RSP_BUFFER_SIZE);
- if (!reqbuf || !rspbuf) {
- errno = ENOMEM;
- status = -1;
- goto end;
- }
- reqhdr = (sdp_pdu_hdr_t *) reqbuf;
- reqhdr->pdu_id = SDP_SVC_REMOVE_REQ;
- reqhdr->tid = htons(sdp_gen_tid(session));
- p = reqbuf + sizeof(sdp_pdu_hdr_t);
- reqsize = sizeof(sdp_pdu_hdr_t);
- bt_put_be32(handle, p);
- reqsize += sizeof(uint32_t);
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- status = sdp_send_req_w4_rsp(session, reqbuf, rspbuf, reqsize, &rspsize);
- if (status < 0)
- goto end;
- if (rspsize < sizeof(sdp_pdu_hdr_t) + sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- errno = EPROTO;
- status = -1;
- goto end;
- }
- rsphdr = (sdp_pdu_hdr_t *) rspbuf;
- p = rspbuf + sizeof(sdp_pdu_hdr_t);
- if (rsphdr->pdu_id == SDP_ERROR_RSP) {
- /* For this case the status always is invalid record handle */
- errno = EINVAL;
- status = -1;
- } else if (rsphdr->pdu_id != SDP_SVC_REMOVE_RSP) {
- errno = EPROTO;
- status = -1;
- } else {
- uint16_t tmp;
- memcpy(&tmp, p, sizeof(tmp));
- status = tmp;
- }
- end:
- free(reqbuf);
- free(rspbuf);
- return status;
- }
- int sdp_device_record_unregister(sdp_session_t *session, bdaddr_t *device, sdp_record_t *rec)
- {
- int err;
- err = sdp_device_record_unregister_binary(session, device, rec->handle);
- if (err == 0)
- sdp_record_free(rec);
- return err;
- }
- int sdp_record_unregister(sdp_session_t *session, sdp_record_t *rec)
- {
- return sdp_device_record_unregister(session, BDADDR_ANY, rec);
- }
- /*
- * modify an existing service record
- */
- int sdp_device_record_update_binary(sdp_session_t *session, bdaddr_t *device, uint32_t handle, uint8_t *data, uint32_t size)
- {
- return -1;
- }
- int sdp_device_record_update(sdp_session_t *session, bdaddr_t *device, const sdp_record_t *rec)
- {
- uint8_t *reqbuf, *rspbuf, *p;
- uint32_t reqsize, rspsize;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- uint32_t handle;
- sdp_buf_t pdu;
- int status;
- SDPDBG("");
- handle = rec->handle;
- if (handle == SDP_SERVER_RECORD_HANDLE) {
- errno = EINVAL;
- return -1;
- }
- if (!session->local) {
- errno = EREMOTE;
- return -1;
- }
- reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- rspbuf = malloc(SDP_RSP_BUFFER_SIZE);
- if (!reqbuf || !rspbuf) {
- errno = ENOMEM;
- status = -1;
- goto end;
- }
- reqhdr = (sdp_pdu_hdr_t *) reqbuf;
- reqhdr->pdu_id = SDP_SVC_UPDATE_REQ;
- reqhdr->tid = htons(sdp_gen_tid(session));
- p = reqbuf + sizeof(sdp_pdu_hdr_t);
- reqsize = sizeof(sdp_pdu_hdr_t);
- bt_put_be32(handle, p);
- reqsize += sizeof(uint32_t);
- p += sizeof(uint32_t);
- if (sdp_gen_record_pdu(rec, &pdu) < 0) {
- errno = ENOMEM;
- status = -1;
- goto end;
- }
- memcpy(p, pdu.data, pdu.data_size);
- reqsize += pdu.data_size;
- free(pdu.data);
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- status = sdp_send_req_w4_rsp(session, reqbuf, rspbuf, reqsize, &rspsize);
- if (status < 0)
- goto end;
- if (rspsize < sizeof(sdp_pdu_hdr_t) + sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- errno = EPROTO;
- status = -1;
- goto end;
- }
- SDPDBG("Send req status : %d", status);
- rsphdr = (sdp_pdu_hdr_t *) rspbuf;
- p = rspbuf + sizeof(sdp_pdu_hdr_t);
- if (rsphdr->pdu_id == SDP_ERROR_RSP) {
- /* The status can be invalid sintax or invalid record handle */
- errno = EINVAL;
- status = -1;
- } else if (rsphdr->pdu_id != SDP_SVC_UPDATE_RSP) {
- errno = EPROTO;
- status = -1;
- } else {
- uint16_t tmp;
- memcpy(&tmp, p, sizeof(tmp));
- status = tmp;
- }
- end:
- free(reqbuf);
- free(rspbuf);
- return status;
- }
- int sdp_record_update(sdp_session_t *session, const sdp_record_t *rec)
- {
- return sdp_device_record_update(session, BDADDR_ANY, rec);
- }
- sdp_record_t *sdp_record_alloc(void)
- {
- sdp_record_t *rec = bt_malloc0(sizeof(sdp_record_t));
- if (!rec)
- return NULL;
- rec->handle = 0xffffffff;
- return rec;
- }
- /*
- * Free the contents of a service record
- */
- void sdp_record_free(sdp_record_t *rec)
- {
- sdp_list_free(rec->attrlist, (sdp_free_func_t) sdp_data_free);
- sdp_list_free(rec->pattern, free);
- free(rec);
- }
- void sdp_pattern_add_uuid(sdp_record_t *rec, uuid_t *uuid)
- {
- uuid_t *uuid128 = sdp_uuid_to_uuid128(uuid);
- SDPDBG("Elements in target pattern : %d", sdp_list_len(rec->pattern));
- SDPDBG("Trying to add : 0x%lx", (unsigned long) uuid128);
- if (sdp_list_find(rec->pattern, uuid128, sdp_uuid128_cmp) == NULL)
- rec->pattern = sdp_list_insert_sorted(rec->pattern, uuid128, sdp_uuid128_cmp);
- else
- bt_free(uuid128);
- SDPDBG("Elements in target pattern : %d", sdp_list_len(rec->pattern));
- }
- void sdp_pattern_add_uuidseq(sdp_record_t *rec, sdp_list_t *seq)
- {
- for (; seq; seq = seq->next) {
- uuid_t *uuid = (uuid_t *)seq->data;
- sdp_pattern_add_uuid(rec, uuid);
- }
- }
- /*
- * Extract a sequence of service record handles from a PDU buffer
- * and add the entries to a sdp_list_t. Note that the service record
- * handles are not in "data element sequence" form, but just like
- * an array of service handles
- */
- static void extract_record_handle_seq(uint8_t *pdu, int bufsize, sdp_list_t **seq, int count, unsigned int *scanned)
- {
- sdp_list_t *pSeq = *seq;
- uint8_t *pdata = pdu;
- int n;
- for (n = 0; n < count; n++) {
- uint32_t *pSvcRec;
- if (bufsize < (int) sizeof(uint32_t)) {
- SDPERR("Unexpected end of packet");
- break;
- }
- pSvcRec = malloc(sizeof(uint32_t));
- if (!pSvcRec)
- break;
- *pSvcRec = bt_get_be32(pdata);
- pSeq = sdp_list_append(pSeq, pSvcRec);
- pdata += sizeof(uint32_t);
- *scanned += sizeof(uint32_t);
- bufsize -= sizeof(uint32_t);
- }
- *seq = pSeq;
- }
- /*
- * Generate the attribute sequence pdu form
- * from sdp_list_t elements. Return length of attr seq
- */
- static int gen_dataseq_pdu(uint8_t *dst, const sdp_list_t *seq, uint8_t dtd)
- {
- sdp_data_t *dataseq;
- void **types, **values;
- sdp_buf_t buf;
- int i, seqlen = sdp_list_len(seq);
- /* Fill up the value and the dtd arrays */
- SDPDBG("");
- SDPDBG("Seq length : %d", seqlen);
- types = malloc(seqlen * sizeof(void *));
- if (!types)
- return -ENOMEM;
- values = malloc(seqlen * sizeof(void *));
- if (!values) {
- free(types);
- return -ENOMEM;
- }
- for (i = 0; i < seqlen; i++) {
- void *data = seq->data;
- types[i] = &dtd;
- if (SDP_IS_UUID(dtd))
- data = &((uuid_t *)data)->value;
- values[i] = data;
- seq = seq->next;
- }
- dataseq = sdp_seq_alloc(types, values, seqlen);
- if (!dataseq) {
- free(types);
- free(values);
- return -ENOMEM;
- }
- memset(&buf, 0, sizeof(sdp_buf_t));
- sdp_gen_buffer(&buf, dataseq);
- buf.data = malloc(buf.buf_size);
- if (!buf.data) {
- sdp_data_free(dataseq);
- free(types);
- free(values);
- return -ENOMEM;
- }
- SDPDBG("Data Seq : 0x%p", seq);
- seqlen = sdp_gen_pdu(&buf, dataseq);
- SDPDBG("Copying : %d", buf.data_size);
- memcpy(dst, buf.data, buf.data_size);
- sdp_data_free(dataseq);
- free(types);
- free(values);
- free(buf.data);
- return seqlen;
- }
- static int gen_searchseq_pdu(uint8_t *dst, const sdp_list_t *seq)
- {
- uuid_t *uuid = seq->data;
- return gen_dataseq_pdu(dst, seq, uuid->type);
- }
- static int gen_attridseq_pdu(uint8_t *dst, const sdp_list_t *seq, uint8_t dataType)
- {
- return gen_dataseq_pdu(dst, seq, dataType);
- }
- typedef struct {
- uint8_t length;
- unsigned char data[16];
- } __attribute__ ((packed)) sdp_cstate_t;
- static int copy_cstate(uint8_t *pdata, int pdata_len, const sdp_cstate_t *cstate)
- {
- if (cstate) {
- uint8_t len = cstate->length;
- if (len >= pdata_len) {
- SDPERR("Continuation state size exceeds internal buffer");
- len = pdata_len - 1;
- }
- *pdata++ = len;
- memcpy(pdata, cstate->data, len);
- return len + 1;
- }
- *pdata = 0;
- return 1;
- }
- /*
- * This is a service search request.
- *
- * INPUT :
- *
- * sdp_list_t *search
- * Singly linked list containing elements of the search
- * pattern. Each entry in the list is a UUID (DataTypeSDP_UUID16)
- * of the service to be searched
- *
- * uint16_t max_rec_num
- * A 16 bit integer which tells the service, the maximum
- * entries that the client can handle in the response. The
- * server is obliged not to return > max_rec_num entries
- *
- * OUTPUT :
- *
- * int return value
- * 0:
- * The request completed successfully. This does not
- * mean the requested services were found
- * -1:
- * On any failure and sets errno
- *
- * sdp_list_t **rsp_list
- * This variable is set on a successful return if there are
- * non-zero service handles. It is a singly linked list of
- * service record handles (uint16_t)
- */
- int sdp_service_search_req(sdp_session_t *session, const sdp_list_t *search,
- uint16_t max_rec_num, sdp_list_t **rsp)
- {
- int status = 0;
- uint32_t reqsize = 0, _reqsize;
- uint32_t rspsize = 0, rsplen;
- int seqlen = 0;
- int rec_count;
- unsigned scanned, pdata_len;
- uint8_t *pdata, *_pdata;
- uint8_t *reqbuf, *rspbuf;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- sdp_cstate_t *cstate = NULL;
- reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- rspbuf = malloc(SDP_RSP_BUFFER_SIZE);
- if (!reqbuf || !rspbuf) {
- errno = ENOMEM;
- status = -1;
- goto end;
- }
- reqhdr = (sdp_pdu_hdr_t *) reqbuf;
- reqhdr->pdu_id = SDP_SVC_SEARCH_REQ;
- pdata = reqbuf + sizeof(sdp_pdu_hdr_t);
- reqsize = sizeof(sdp_pdu_hdr_t);
- /* add service class IDs for search */
- seqlen = gen_searchseq_pdu(pdata, search);
- if (seqlen < 0) {
- errno = EINVAL;
- status = -1;
- goto end;
- }
- SDPDBG("Data seq added : %d", seqlen);
- /* set the length and increment the pointer */
- reqsize += seqlen;
- pdata += seqlen;
- /* specify the maximum svc rec count that client expects */
- bt_put_be16(max_rec_num, pdata);
- reqsize += sizeof(uint16_t);
- pdata += sizeof(uint16_t);
- _reqsize = reqsize;
- _pdata = pdata;
- *rsp = NULL;
- do {
- /* Add continuation state or NULL (first time) */
- reqsize = _reqsize + copy_cstate(_pdata,
- SDP_REQ_BUFFER_SIZE - _reqsize, cstate);
- /* Set the request header's param length */
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- reqhdr->tid = htons(sdp_gen_tid(session));
- /*
- * Send the request, wait for response and if
- * no error, set the appropriate values and return
- */
- status = sdp_send_req_w4_rsp(session, reqbuf, rspbuf, reqsize, &rspsize);
- if (status < 0)
- goto end;
- if (rspsize < sizeof(sdp_pdu_hdr_t)) {
- SDPERR("Unexpected end of packet");
- status = -1;
- goto end;
- }
- rsphdr = (sdp_pdu_hdr_t *) rspbuf;
- rsplen = ntohs(rsphdr->plen);
- if (rsphdr->pdu_id == SDP_ERROR_RSP) {
- SDPDBG("Status : 0x%x", rsphdr->pdu_id);
- status = -1;
- goto end;
- }
- scanned = 0;
- pdata = rspbuf + sizeof(sdp_pdu_hdr_t);
- pdata_len = rspsize - sizeof(sdp_pdu_hdr_t);
- if (pdata_len < sizeof(uint16_t) + sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- status = -1;
- goto end;
- }
- /* net service record match count */
- pdata += sizeof(uint16_t);
- scanned += sizeof(uint16_t);
- pdata_len -= sizeof(uint16_t);
- rec_count = bt_get_be16(pdata);
- pdata += sizeof(uint16_t);
- scanned += sizeof(uint16_t);
- pdata_len -= sizeof(uint16_t);
- SDPDBG("Current svc count: %d", rec_count);
- SDPDBG("ResponseLength: %d", rsplen);
- if (!rec_count) {
- status = -1;
- goto end;
- }
- extract_record_handle_seq(pdata, pdata_len, rsp, rec_count, &scanned);
- SDPDBG("BytesScanned : %d", scanned);
- if (rsplen > scanned) {
- uint8_t cstate_len;
- if (rspsize < sizeof(sdp_pdu_hdr_t) + scanned + sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet: continuation state data missing");
- status = -1;
- goto end;
- }
- pdata = rspbuf + sizeof(sdp_pdu_hdr_t) + scanned;
- cstate_len = *(uint8_t *) pdata;
- if (cstate_len > 0) {
- cstate = (sdp_cstate_t *)pdata;
- SDPDBG("Cont state length: %d", cstate_len);
- } else
- cstate = NULL;
- }
- } while (cstate);
- end:
- free(reqbuf);
- free(rspbuf);
- return status;
- }
- /*
- * This is a service attribute request.
- *
- * INPUT :
- *
- * uint32_t handle
- * The handle of the service for which the attribute(s) are
- * requested
- *
- * sdp_attrreq_type_t reqtype
- * Attribute identifiers are 16 bit unsigned integers specified
- * in one of 2 ways described below :
- * SDP_ATTR_REQ_INDIVIDUAL - 16bit individual identifiers
- * They are the actual attribute identifiers in ascending order
- *
- * SDP_ATTR_REQ_RANGE - 32bit identifier range
- * The high-order 16bits is the start of range
- * the low-order 16bits are the end of range
- * 0x0000 to 0xFFFF gets all attributes
- *
- * sdp_list_t *attrid
- * Singly linked list containing attribute identifiers desired.
- * Every element is either a uint16_t(attrSpec = SDP_ATTR_REQ_INDIVIDUAL)
- * or a uint32_t(attrSpec=SDP_ATTR_REQ_RANGE)
- *
- * OUTPUT :
- * return sdp_record_t *
- * 0:
- * On any error and sets errno
- * !0:
- * The service record
- */
- sdp_record_t *sdp_service_attr_req(sdp_session_t *session, uint32_t handle,
- sdp_attrreq_type_t reqtype, const sdp_list_t *attrids)
- {
- uint32_t reqsize = 0, _reqsize;
- uint32_t rspsize = 0, rsp_count;
- int attr_list_len = 0;
- int seqlen = 0;
- unsigned int pdata_len;
- uint8_t *pdata, *_pdata;
- uint8_t *reqbuf, *rspbuf;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- sdp_cstate_t *cstate = NULL;
- uint8_t cstate_len = 0;
- sdp_buf_t rsp_concat_buf;
- sdp_record_t *rec = 0;
- if (reqtype != SDP_ATTR_REQ_INDIVIDUAL && reqtype != SDP_ATTR_REQ_RANGE) {
- errno = EINVAL;
- return NULL;
- }
- memset(&rsp_concat_buf, 0, sizeof(sdp_buf_t));
- reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- rspbuf = malloc(SDP_RSP_BUFFER_SIZE);
- if (!reqbuf || !rspbuf) {
- errno = ENOMEM;
- goto end;
- }
- reqhdr = (sdp_pdu_hdr_t *) reqbuf;
- reqhdr->pdu_id = SDP_SVC_ATTR_REQ;
- pdata = reqbuf + sizeof(sdp_pdu_hdr_t);
- reqsize = sizeof(sdp_pdu_hdr_t);
- /* add the service record handle */
- bt_put_be32(handle, pdata);
- reqsize += sizeof(uint32_t);
- pdata += sizeof(uint32_t);
- /* specify the response limit */
- bt_put_be16(65535, pdata);
- reqsize += sizeof(uint16_t);
- pdata += sizeof(uint16_t);
- /* get attr seq PDU form */
- seqlen = gen_attridseq_pdu(pdata, attrids,
- reqtype == SDP_ATTR_REQ_INDIVIDUAL? SDP_UINT16 : SDP_UINT32);
- if (seqlen == -1) {
- errno = EINVAL;
- goto end;
- }
- pdata += seqlen;
- reqsize += seqlen;
- SDPDBG("Attr list length : %d", seqlen);
- /* save before Continuation State */
- _pdata = pdata;
- _reqsize = reqsize;
- do {
- int status;
- /* add NULL continuation state */
- reqsize = _reqsize + copy_cstate(_pdata,
- SDP_REQ_BUFFER_SIZE - _reqsize, cstate);
- /* set the request header's param length */
- reqhdr->tid = htons(sdp_gen_tid(session));
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- status = sdp_send_req_w4_rsp(session, reqbuf, rspbuf, reqsize, &rspsize);
- if (status < 0)
- goto end;
- if (rspsize < sizeof(sdp_pdu_hdr_t)) {
- SDPERR("Unexpected end of packet");
- goto end;
- }
- rsphdr = (sdp_pdu_hdr_t *) rspbuf;
- if (rsphdr->pdu_id == SDP_ERROR_RSP) {
- SDPDBG("PDU ID : 0x%x", rsphdr->pdu_id);
- goto end;
- }
- pdata = rspbuf + sizeof(sdp_pdu_hdr_t);
- pdata_len = rspsize - sizeof(sdp_pdu_hdr_t);
- if (pdata_len < sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- goto end;
- }
- rsp_count = bt_get_be16(pdata);
- attr_list_len += rsp_count;
- pdata += sizeof(uint16_t);
- pdata_len -= sizeof(uint16_t);
- /*
- * if continuation state set need to re-issue request before
- * parsing
- */
- if (pdata_len < rsp_count + sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet: continuation state data missing");
- goto end;
- }
- cstate_len = *(uint8_t *) (pdata + rsp_count);
- SDPDBG("Response id : %d", rsphdr->pdu_id);
- SDPDBG("Attrlist byte count : %d", rsp_count);
- SDPDBG("sdp_cstate_t length : %d", cstate_len);
- /*
- * a split response: concatenate intermediate responses
- * and the last one (which has cstate_len == 0)
- */
- if (cstate_len > 0 || rsp_concat_buf.data_size != 0) {
- uint8_t *targetPtr = NULL;
- cstate = cstate_len > 0 ? (sdp_cstate_t *) (pdata + rsp_count) : 0;
- /* build concatenated response buffer */
- rsp_concat_buf.data = realloc(rsp_concat_buf.data, rsp_concat_buf.data_size + rsp_count);
- rsp_concat_buf.buf_size = rsp_concat_buf.data_size + rsp_count;
- targetPtr = rsp_concat_buf.data + rsp_concat_buf.data_size;
- memcpy(targetPtr, pdata, rsp_count);
- rsp_concat_buf.data_size += rsp_count;
- }
- } while (cstate);
- if (attr_list_len > 0) {
- int scanned = 0;
- if (rsp_concat_buf.data_size != 0) {
- pdata = rsp_concat_buf.data;
- pdata_len = rsp_concat_buf.data_size;
- }
- rec = sdp_extract_pdu(pdata, pdata_len, &scanned);
- }
- end:
- free(reqbuf);
- free(rsp_concat_buf.data);
- free(rspbuf);
- return rec;
- }
- /*
- * SDP transaction structure for asynchronous search
- */
- struct sdp_transaction {
- sdp_callback_t *cb; /* called when the transaction finishes */
- void *udata; /* client user data */
- uint8_t *reqbuf; /* pointer to request PDU */
- sdp_buf_t rsp_concat_buf;
- uint32_t reqsize; /* without cstate */
- int err; /* ZERO if success or the errno if failed */
- };
- /*
- * Creates a new sdp session for asynchronous search
- * INPUT:
- * int sk
- * non-blocking L2CAP socket
- *
- * RETURN:
- * sdp_session_t *
- * NULL - On memory allocation failure
- */
- sdp_session_t *sdp_create(int sk, uint32_t flags)
- {
- sdp_session_t *session;
- struct sdp_transaction *t;
- session = bt_malloc0(sizeof(sdp_session_t));
- if (!session) {
- errno = ENOMEM;
- return NULL;
- }
- session->flags = flags;
- session->sock = sk;
- t = bt_malloc0(sizeof(struct sdp_transaction));
- if (!t) {
- errno = ENOMEM;
- free(session);
- return NULL;
- }
- session->priv = t;
- return session;
- }
- /*
- * Sets the callback function/user data used to notify the application
- * that the asynchronous transaction finished. This function must be
- * called before request an asynchronous search.
- *
- * INPUT:
- * sdp_session_t *session
- * Current sdp session to be handled
- * sdp_callback_t *cb
- * callback to be called when the transaction finishes
- * void *udata
- * user data passed to callback
- * RETURN:
- * 0 - Success
- * -1 - Failure
- */
- int sdp_set_notify(sdp_session_t *session, sdp_callback_t *func, void *udata)
- {
- struct sdp_transaction *t;
- if (!session || !session->priv)
- return -1;
- t = session->priv;
- t->cb = func;
- t->udata = udata;
- return 0;
- }
- /*
- * This function starts an asynchronous service search request.
- * The incoming and outgoing data are stored in the transaction structure
- * buffers. When there is incoming data the sdp_process function must be
- * called to get the data and handle the continuation state.
- *
- * INPUT :
- * sdp_session_t *session
- * Current sdp session to be handled
- *
- * sdp_list_t *search
- * Singly linked list containing elements of the search
- * pattern. Each entry in the list is a UUID (DataTypeSDP_UUID16)
- * of the service to be searched
- *
- * uint16_t max_rec_num
- * A 16 bit integer which tells the service, the maximum
- * entries that the client can handle in the response. The
- * server is obliged not to return > max_rec_num entries
- *
- * OUTPUT :
- *
- * int return value
- * 0 - if the request has been sent properly
- * -1 - On any failure and sets errno
- */
- int sdp_service_search_async(sdp_session_t *session, const sdp_list_t *search, uint16_t max_rec_num)
- {
- struct sdp_transaction *t;
- sdp_pdu_hdr_t *reqhdr;
- uint8_t *pdata;
- int cstate_len, seqlen = 0;
- if (!session || !session->priv)
- return -1;
- t = session->priv;
- /* clean possible allocated buffer */
- free(t->rsp_concat_buf.data);
- memset(&t->rsp_concat_buf, 0, sizeof(sdp_buf_t));
- if (!t->reqbuf) {
- t->reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- if (!t->reqbuf) {
- t->err = ENOMEM;
- goto end;
- }
- }
- memset(t->reqbuf, 0, SDP_REQ_BUFFER_SIZE);
- reqhdr = (sdp_pdu_hdr_t *) t->reqbuf;
- reqhdr->tid = htons(sdp_gen_tid(session));
- reqhdr->pdu_id = SDP_SVC_SEARCH_REQ;
- /* generate PDU */
- pdata = t->reqbuf + sizeof(sdp_pdu_hdr_t);
- t->reqsize = sizeof(sdp_pdu_hdr_t);
- /* add service class IDs for search */
- seqlen = gen_searchseq_pdu(pdata, search);
- if (seqlen < 0) {
- t->err = EINVAL;
- goto end;
- }
- SDPDBG("Data seq added : %d", seqlen);
- /* now set the length and increment the pointer */
- t->reqsize += seqlen;
- pdata += seqlen;
- bt_put_be16(max_rec_num, pdata);
- t->reqsize += sizeof(uint16_t);
- pdata += sizeof(uint16_t);
- /* set the request header's param length */
- cstate_len = copy_cstate(pdata, SDP_REQ_BUFFER_SIZE - t->reqsize, NULL);
- reqhdr->plen = htons((t->reqsize + cstate_len) - sizeof(sdp_pdu_hdr_t));
- if (sdp_send_req(session, t->reqbuf, t->reqsize + cstate_len) < 0) {
- SDPERR("Error sending data:%m");
- t->err = errno;
- goto end;
- }
- return 0;
- end:
- free(t->reqbuf);
- t->reqbuf = NULL;
- return -1;
- }
- /*
- * This function starts an asynchronous service attribute request.
- * The incoming and outgoing data are stored in the transaction structure
- * buffers. When there is incoming data the sdp_process function must be
- * called to get the data and handle the continuation state.
- *
- * INPUT :
- * sdp_session_t *session
- * Current sdp session to be handled
- *
- * uint32_t handle
- * The handle of the service for which the attribute(s) are
- * requested
- *
- * sdp_attrreq_type_t reqtype
- * Attribute identifiers are 16 bit unsigned integers specified
- * in one of 2 ways described below :
- * SDP_ATTR_REQ_INDIVIDUAL - 16bit individual identifiers
- * They are the actual attribute identifiers in ascending order
- *
- * SDP_ATTR_REQ_RANGE - 32bit identifier range
- * The high-order 16bits is the start of range
- * the low-order 16bits are the end of range
- * 0x0000 to 0xFFFF gets all attributes
- *
- * sdp_list_t *attrid_list
- * Singly linked list containing attribute identifiers desired.
- * Every element is either a uint16_t(attrSpec = SDP_ATTR_REQ_INDIVIDUAL)
- * or a uint32_t(attrSpec=SDP_ATTR_REQ_RANGE)
- *
- * OUTPUT :
- * int return value
- * 0 - if the request has been sent properly
- * -1 - On any failure and sets errno
- */
- int sdp_service_attr_async(sdp_session_t *session, uint32_t handle, sdp_attrreq_type_t reqtype, const sdp_list_t *attrid_list)
- {
- struct sdp_transaction *t;
- sdp_pdu_hdr_t *reqhdr;
- uint8_t *pdata;
- int cstate_len, seqlen = 0;
- if (!session || !session->priv)
- return -1;
- t = session->priv;
- /* clean possible allocated buffer */
- free(t->rsp_concat_buf.data);
- memset(&t->rsp_concat_buf, 0, sizeof(sdp_buf_t));
- if (!t->reqbuf) {
- t->reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- if (!t->reqbuf) {
- t->err = ENOMEM;
- goto end;
- }
- }
- memset(t->reqbuf, 0, SDP_REQ_BUFFER_SIZE);
- reqhdr = (sdp_pdu_hdr_t *) t->reqbuf;
- reqhdr->tid = htons(sdp_gen_tid(session));
- reqhdr->pdu_id = SDP_SVC_ATTR_REQ;
- /* generate PDU */
- pdata = t->reqbuf + sizeof(sdp_pdu_hdr_t);
- t->reqsize = sizeof(sdp_pdu_hdr_t);
- /* add the service record handle */
- bt_put_be32(handle, pdata);
- t->reqsize += sizeof(uint32_t);
- pdata += sizeof(uint32_t);
- /* specify the response limit */
- bt_put_be16(65535, pdata);
- t->reqsize += sizeof(uint16_t);
- pdata += sizeof(uint16_t);
- /* get attr seq PDU form */
- seqlen = gen_attridseq_pdu(pdata, attrid_list,
- reqtype == SDP_ATTR_REQ_INDIVIDUAL? SDP_UINT16 : SDP_UINT32);
- if (seqlen == -1) {
- t->err = EINVAL;
- goto end;
- }
- /* now set the length and increment the pointer */
- t->reqsize += seqlen;
- pdata += seqlen;
- SDPDBG("Attr list length : %d", seqlen);
- /* set the request header's param length */
- cstate_len = copy_cstate(pdata, SDP_REQ_BUFFER_SIZE - t->reqsize, NULL);
- reqhdr->plen = htons((t->reqsize + cstate_len) - sizeof(sdp_pdu_hdr_t));
- if (sdp_send_req(session, t->reqbuf, t->reqsize + cstate_len) < 0) {
- SDPERR("Error sending data:%m");
- t->err = errno;
- goto end;
- }
- return 0;
- end:
- free(t->reqbuf);
- t->reqbuf = NULL;
- return -1;
- }
- /*
- * This function starts an asynchronous service search attributes.
- * It is a service search request combined with attribute request. The incoming
- * and outgoing data are stored in the transaction structure buffers. When there
- * is incoming data the sdp_process function must be called to get the data
- * and handle the continuation state.
- *
- * INPUT:
- * sdp_session_t *session
- * Current sdp session to be handled
- *
- * sdp_list_t *search
- * Singly linked list containing elements of the search
- * pattern. Each entry in the list is a UUID(DataTypeSDP_UUID16)
- * of the service to be searched
- *
- * AttributeSpecification attrSpec
- * Attribute identifiers are 16 bit unsigned integers specified
- * in one of 2 ways described below :
- * SDP_ATTR_REQ_INDIVIDUAL - 16bit individual identifiers
- * They are the actual attribute identifiers in ascending order
- *
- * SDP_ATTR_REQ_RANGE - 32bit identifier range
- * The high-order 16bits is the start of range
- * the low-order 16bits are the end of range
- * 0x0000 to 0xFFFF gets all attributes
- *
- * sdp_list_t *attrid_list
- * Singly linked list containing attribute identifiers desired.
- * Every element is either a uint16_t(attrSpec = SDP_ATTR_REQ_INDIVIDUAL)
- * or a uint32_t(attrSpec=SDP_ATTR_REQ_RANGE)
- *
- * RETURN:
- * 0 - if the request has been sent properly
- * -1 - On any failure
- */
- int sdp_service_search_attr_async(sdp_session_t *session, const sdp_list_t *search, sdp_attrreq_type_t reqtype, const sdp_list_t *attrid_list)
- {
- struct sdp_transaction *t;
- sdp_pdu_hdr_t *reqhdr;
- uint8_t *pdata;
- int cstate_len, seqlen = 0;
- if (!session || !session->priv)
- return -1;
- t = session->priv;
- /* clean possible allocated buffer */
- free(t->rsp_concat_buf.data);
- memset(&t->rsp_concat_buf, 0, sizeof(sdp_buf_t));
- if (!t->reqbuf) {
- t->reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- if (!t->reqbuf) {
- t->err = ENOMEM;
- goto end;
- }
- }
- memset(t->reqbuf, 0, SDP_REQ_BUFFER_SIZE);
- reqhdr = (sdp_pdu_hdr_t *) t->reqbuf;
- reqhdr->tid = htons(sdp_gen_tid(session));
- reqhdr->pdu_id = SDP_SVC_SEARCH_ATTR_REQ;
- /* generate PDU */
- pdata = t->reqbuf + sizeof(sdp_pdu_hdr_t);
- t->reqsize = sizeof(sdp_pdu_hdr_t);
- /* add service class IDs for search */
- seqlen = gen_searchseq_pdu(pdata, search);
- if (seqlen < 0) {
- t->err = EINVAL;
- goto end;
- }
- SDPDBG("Data seq added : %d", seqlen);
- /* now set the length and increment the pointer */
- t->reqsize += seqlen;
- pdata += seqlen;
- bt_put_be16(SDP_MAX_ATTR_LEN, pdata);
- t->reqsize += sizeof(uint16_t);
- pdata += sizeof(uint16_t);
- SDPDBG("Max attr byte count : %d", SDP_MAX_ATTR_LEN);
- /* get attr seq PDU form */
- seqlen = gen_attridseq_pdu(pdata, attrid_list,
- reqtype == SDP_ATTR_REQ_INDIVIDUAL ? SDP_UINT16 : SDP_UINT32);
- if (seqlen == -1) {
- t->err = EINVAL;
- goto end;
- }
- pdata += seqlen;
- SDPDBG("Attr list length : %d", seqlen);
- t->reqsize += seqlen;
- /* set the request header's param length */
- cstate_len = copy_cstate(pdata, SDP_REQ_BUFFER_SIZE - t->reqsize, NULL);
- reqhdr->plen = htons((t->reqsize + cstate_len) - sizeof(sdp_pdu_hdr_t));
- if (sdp_send_req(session, t->reqbuf, t->reqsize + cstate_len) < 0) {
- SDPERR("Error sending data:%m");
- t->err = errno;
- goto end;
- }
- return 0;
- end:
- free(t->reqbuf);
- t->reqbuf = NULL;
- return -1;
- }
- /*
- * Function used to get the error reason after sdp_callback_t function has been called
- * and the status is 0xffff or if sdp_service_{search, attr, search_attr}_async returns -1.
- * It indicates that an error NOT related to SDP_ErrorResponse happened. Get errno directly
- * is not safe because multiple transactions can be triggered.
- * This function must be used with asynchronous sdp functions only.
- *
- * INPUT:
- * sdp_session_t *session
- * Current sdp session to be handled
- * RETURN:
- * 0 = No error in the current transaction
- * -1 - if the session is invalid
- * positive value - the errno value
- *
- */
- int sdp_get_error(sdp_session_t *session)
- {
- struct sdp_transaction *t;
- if (!session || !session->priv) {
- SDPERR("Invalid session");
- return -1;
- }
- t = session->priv;
- return t->err;
- }
- /*
- * Receive the incoming SDP PDU. This function must be called when there is data
- * available to be read. On continuation state, the original request (with a new
- * transaction ID) and the continuation state data will be appended in the initial PDU.
- * If an error happens or the transaction finishes the callback function will be called.
- *
- * INPUT:
- * sdp_session_t *session
- * Current sdp session to be handled
- * RETURN:
- * 0 - if the transaction is on continuation state
- * -1 - On any failure or the transaction finished
- */
- int sdp_process(sdp_session_t *session)
- {
- struct sdp_transaction *t;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- sdp_cstate_t *pcstate;
- uint8_t *pdata, *rspbuf, *targetPtr;
- int rsp_count, err = -1;
- size_t size = 0;
- int n, plen;
- uint16_t status = 0xffff;
- uint8_t pdu_id = 0x00;
- if (!session || !session->priv) {
- SDPERR("Invalid session");
- return -1;
- }
- rspbuf = bt_malloc0(SDP_RSP_BUFFER_SIZE);
- if (!rspbuf) {
- SDPERR("Response buffer alloc failure:%m (%d)", errno);
- return -1;
- }
- t = session->priv;
- reqhdr = (sdp_pdu_hdr_t *)t->reqbuf;
- rsphdr = (sdp_pdu_hdr_t *)rspbuf;
- pdata = rspbuf + sizeof(sdp_pdu_hdr_t);
- n = sdp_read_rsp(session, rspbuf, SDP_RSP_BUFFER_SIZE);
- if (n < 0) {
- SDPERR("Read response:%m (%d)", errno);
- t->err = errno;
- goto end;
- }
- if (reqhdr->tid != rsphdr->tid) {
- t->err = EPROTO;
- SDPERR("Protocol error: transaction id does not match");
- goto end;
- }
- if (n != (int) (ntohs(rsphdr->plen) + sizeof(sdp_pdu_hdr_t))) {
- t->err = EPROTO;
- SDPERR("Protocol error: invalid length");
- goto end;
- }
- pdu_id = rsphdr->pdu_id;
- switch (rsphdr->pdu_id) {
- uint8_t *ssr_pdata;
- uint16_t tsrc, csrc;
- case SDP_SVC_SEARCH_RSP:
- /*
- * TSRC: Total Service Record Count (2 bytes)
- * CSRC: Current Service Record Count (2 bytes)
- */
- ssr_pdata = pdata;
- tsrc = bt_get_be16(ssr_pdata);
- ssr_pdata += sizeof(uint16_t);
- csrc = bt_get_be16(ssr_pdata);
- /* csrc should never be larger than tsrc */
- if (csrc > tsrc) {
- t->err = EPROTO;
- SDPERR("Protocol error: wrong current service record count value.");
- goto end;
- }
- SDPDBG("Total svc count: %d", tsrc);
- SDPDBG("Current svc count: %d", csrc);
- /* parameter length without continuation state */
- plen = sizeof(tsrc) + sizeof(csrc) + csrc * 4;
- if (t->rsp_concat_buf.data_size == 0) {
- /* first fragment */
- rsp_count = sizeof(tsrc) + sizeof(csrc) + csrc * 4;
- } else if (t->rsp_concat_buf.data_size >= sizeof(uint16_t) * 2) {
- /* point to the first csrc */
- uint8_t *pcsrc = t->rsp_concat_buf.data + 2;
- uint16_t tcsrc, tcsrc2;
- /* FIXME: update the interface later. csrc doesn't need be passed to clients */
- pdata += sizeof(uint16_t); /* point to csrc */
- /* the first csrc contains the sum of partial csrc responses */
- memcpy(&tcsrc, pcsrc, sizeof(tcsrc));
- memcpy(&tcsrc2, pdata, sizeof(tcsrc2));
- tcsrc += tcsrc2;
- memcpy(pcsrc, &tcsrc, sizeof(tcsrc));
- pdata += sizeof(uint16_t); /* point to the first handle */
- rsp_count = csrc * 4;
- } else {
- t->err = EPROTO;
- SDPERR("Protocol error: invalid PDU size");
- status = SDP_INVALID_PDU_SIZE;
- goto end;
- }
- status = 0x0000;
- break;
- case SDP_SVC_ATTR_RSP:
- case SDP_SVC_SEARCH_ATTR_RSP:
- rsp_count = bt_get_be16(pdata);
- SDPDBG("Attrlist byte count : %d", rsp_count);
- /* Valid range for rsp_count is 0x0002-0xFFFF */
- if (t->rsp_concat_buf.data_size == 0 && rsp_count < 0x0002) {
- t->err = EPROTO;
- SDPERR("Protocol error: invalid AttrList size");
- status = SDP_INVALID_PDU_SIZE;
- goto end;
- }
- /*
- * Number of bytes in the AttributeLists parameter(without
- * continuation state) + AttributeListsByteCount field size.
- */
- plen = sizeof(uint16_t) + rsp_count;
- pdata += sizeof(uint16_t); /* points to attribute list */
- status = 0x0000;
- break;
- case SDP_ERROR_RSP:
- status = bt_get_be16(pdata);
- size = ntohs(rsphdr->plen);
- goto end;
- default:
- t->err = EPROTO;
- SDPERR("Illegal PDU ID: 0x%x", rsphdr->pdu_id);
- goto end;
- }
- /* Out of bound check before using rsp_count as offset for
- * continuation state, which has at least a one byte size
- * field.
- */
- if ((n - (int) sizeof(sdp_pdu_hdr_t)) < plen + 1) {
- t->err = EPROTO;
- SDPERR("Protocol error: invalid PDU size");
- status = SDP_INVALID_PDU_SIZE;
- goto end;
- }
- pcstate = (sdp_cstate_t *) (pdata + rsp_count);
- SDPDBG("Cstate length : %d", pcstate->length);
- /*
- * Check out of bound. Continuation state must have at least
- * 1 byte: ZERO to indicate that it is not a partial response.
- */
- if ((n - (int) sizeof(sdp_pdu_hdr_t)) != (plen + pcstate->length + 1)) {
- t->err = EPROTO;
- SDPERR("Protocol error: wrong PDU size.");
- status = 0xffff;
- goto end;
- }
- /*
- * This is a split response, need to concatenate intermediate
- * responses and the last one which will have cstate length == 0
- */
- t->rsp_concat_buf.data = realloc(t->rsp_concat_buf.data, t->rsp_concat_buf.data_size + rsp_count);
- targetPtr = t->rsp_concat_buf.data + t->rsp_concat_buf.data_size;
- t->rsp_concat_buf.buf_size = t->rsp_concat_buf.data_size + rsp_count;
- memcpy(targetPtr, pdata, rsp_count);
- t->rsp_concat_buf.data_size += rsp_count;
- if (pcstate->length > 0) {
- int reqsize, cstate_len;
- reqhdr->tid = htons(sdp_gen_tid(session));
- /* add continuation state */
- cstate_len = copy_cstate(t->reqbuf + t->reqsize,
- SDP_REQ_BUFFER_SIZE - t->reqsize, pcstate);
- reqsize = t->reqsize + cstate_len;
- /* set the request header's param length */
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- if (sdp_send_req(session, t->reqbuf, reqsize) < 0) {
- SDPERR("Error sending data:%m(%d)", errno);
- status = 0xffff;
- t->err = errno;
- goto end;
- }
- err = 0;
- }
- end:
- if (err) {
- if (t->rsp_concat_buf.data_size != 0) {
- pdata = t->rsp_concat_buf.data;
- size = t->rsp_concat_buf.data_size;
- }
- if (t->cb)
- t->cb(pdu_id, status, pdata, size, t->udata);
- }
- free(rspbuf);
- return err;
- }
- /*
- * This is a service search request combined with the service
- * attribute request. First a service class match is done and
- * for matching service, requested attributes are extracted
- *
- * INPUT :
- *
- * sdp_list_t *search
- * Singly linked list containing elements of the search
- * pattern. Each entry in the list is a UUID(DataTypeSDP_UUID16)
- * of the service to be searched
- *
- * AttributeSpecification attrSpec
- * Attribute identifiers are 16 bit unsigned integers specified
- * in one of 2 ways described below :
- * SDP_ATTR_REQ_INDIVIDUAL - 16bit individual identifiers
- * They are the actual attribute identifiers in ascending order
- *
- * SDP_ATTR_REQ_RANGE - 32bit identifier range
- * The high-order 16bits is the start of range
- * the low-order 16bits are the end of range
- * 0x0000 to 0xFFFF gets all attributes
- *
- * sdp_list_t *attrids
- * Singly linked list containing attribute identifiers desired.
- * Every element is either a uint16_t(attrSpec = SDP_ATTR_REQ_INDIVIDUAL)
- * or a uint32_t(attrSpec=SDP_ATTR_REQ_RANGE)
- *
- * OUTPUT :
- * int return value
- * 0:
- * The request completed successfully. This does not
- * mean the requested services were found
- * -1:
- * On any error and sets errno
- *
- * sdp_list_t **rsp
- * This variable is set on a successful return to point to
- * service(s) found. Each element of this list is of type
- * sdp_record_t* (of the services which matched the search list)
- */
- int sdp_service_search_attr_req(sdp_session_t *session, const sdp_list_t *search, sdp_attrreq_type_t reqtype, const sdp_list_t *attrids, sdp_list_t **rsp)
- {
- int status = 0;
- uint32_t reqsize = 0, _reqsize;
- uint32_t rspsize = 0;
- int seqlen = 0, attr_list_len = 0;
- int rsp_count = 0, cstate_len = 0;
- unsigned int pdata_len;
- uint8_t *pdata, *_pdata;
- uint8_t *reqbuf, *rspbuf;
- sdp_pdu_hdr_t *reqhdr, *rsphdr;
- uint8_t dataType;
- sdp_list_t *rec_list = NULL;
- sdp_buf_t rsp_concat_buf;
- sdp_cstate_t *cstate = NULL;
- if (reqtype != SDP_ATTR_REQ_INDIVIDUAL && reqtype != SDP_ATTR_REQ_RANGE) {
- errno = EINVAL;
- return -1;
- }
- memset(&rsp_concat_buf, 0, sizeof(sdp_buf_t));
- reqbuf = malloc(SDP_REQ_BUFFER_SIZE);
- rspbuf = malloc(SDP_RSP_BUFFER_SIZE);
- if (!reqbuf || !rspbuf) {
- errno = ENOMEM;
- status = -1;
- goto end;
- }
- reqhdr = (sdp_pdu_hdr_t *) reqbuf;
- reqhdr->pdu_id = SDP_SVC_SEARCH_ATTR_REQ;
- /* generate PDU */
- pdata = reqbuf + sizeof(sdp_pdu_hdr_t);
- reqsize = sizeof(sdp_pdu_hdr_t);
- /* add service class IDs for search */
- seqlen = gen_searchseq_pdu(pdata, search);
- if (seqlen < 0) {
- errno = EINVAL;
- status = -1;
- goto end;
- }
- SDPDBG("Data seq added : %d", seqlen);
- /* now set the length and increment the pointer */
- reqsize += seqlen;
- pdata += seqlen;
- bt_put_be16(SDP_MAX_ATTR_LEN, pdata);
- reqsize += sizeof(uint16_t);
- pdata += sizeof(uint16_t);
- SDPDBG("Max attr byte count : %d", SDP_MAX_ATTR_LEN);
- /* get attr seq PDU form */
- seqlen = gen_attridseq_pdu(pdata, attrids,
- reqtype == SDP_ATTR_REQ_INDIVIDUAL ? SDP_UINT16 : SDP_UINT32);
- if (seqlen == -1) {
- errno = EINVAL;
- status = -1;
- goto end;
- }
- pdata += seqlen;
- SDPDBG("Attr list length : %d", seqlen);
- reqsize += seqlen;
- *rsp = 0;
- /* save before Continuation State */
- _pdata = pdata;
- _reqsize = reqsize;
- do {
- reqhdr->tid = htons(sdp_gen_tid(session));
- /* add continuation state (can be null) */
- reqsize = _reqsize + copy_cstate(_pdata,
- SDP_REQ_BUFFER_SIZE - _reqsize, cstate);
- /* set the request header's param length */
- reqhdr->plen = htons(reqsize - sizeof(sdp_pdu_hdr_t));
- rsphdr = (sdp_pdu_hdr_t *) rspbuf;
- status = sdp_send_req_w4_rsp(session, reqbuf, rspbuf, reqsize, &rspsize);
- if (rspsize < sizeof(sdp_pdu_hdr_t)) {
- SDPERR("Unexpected end of packet");
- status = -1;
- goto end;
- }
- if (status < 0) {
- SDPDBG("Status : 0x%x", rsphdr->pdu_id);
- goto end;
- }
- if (rsphdr->pdu_id == SDP_ERROR_RSP) {
- status = -1;
- goto end;
- }
- pdata = rspbuf + sizeof(sdp_pdu_hdr_t);
- pdata_len = rspsize - sizeof(sdp_pdu_hdr_t);
- if (pdata_len < sizeof(uint16_t)) {
- SDPERR("Unexpected end of packet");
- status = -1;
- goto end;
- }
- rsp_count = bt_get_be16(pdata);
- attr_list_len += rsp_count;
- pdata += sizeof(uint16_t); /* pdata points to attribute list */
- pdata_len -= sizeof(uint16_t);
- if (pdata_len < rsp_count + sizeof(uint8_t)) {
- SDPERR("Unexpected end of packet: continuation state data missing");
- status = -1;
- goto end;
- }
- cstate_len = *(uint8_t *) (pdata + rsp_count);
- SDPDBG("Attrlist byte count : %d", attr_list_len);
- SDPDBG("Response byte count : %d", rsp_count);
- SDPDBG("Cstate length : %d", cstate_len);
- /*
- * This is a split response, need to concatenate intermediate
- * responses and the last one which will have cstate_len == 0
- */
- if (cstate_len > 0 || rsp_concat_buf.data_size != 0) {
- uint8_t *targetPtr = NULL;
- cstate = cstate_len > 0 ? (sdp_cstate_t *) (pdata + rsp_count) : 0;
- /* build concatenated response buffer */
- rsp_concat_buf.data = realloc(rsp_concat_buf.data, rsp_concat_buf.data_size + rsp_count);
- targetPtr = rsp_concat_buf.data + rsp_concat_buf.data_size;
- rsp_concat_buf.buf_size = rsp_concat_buf.data_size + rsp_count;
- memcpy(targetPtr, pdata, rsp_count);
- rsp_concat_buf.data_size += rsp_count;
- }
- } while (cstate);
- if (attr_list_len > 0) {
- int scanned = 0;
- if (rsp_concat_buf.data_size != 0) {
- pdata = rsp_concat_buf.data;
- pdata_len = rsp_concat_buf.data_size;
- }
- /*
- * Response is a sequence of sequence(s) for one or
- * more data element sequence(s) representing services
- * for which attributes are returned
- */
- scanned = sdp_extract_seqtype(pdata, pdata_len, &dataType, &seqlen);
- SDPDBG("Bytes scanned : %d", scanned);
- SDPDBG("Seq length : %d", seqlen);
- if (scanned && seqlen) {
- pdata += scanned;
- pdata_len -= scanned;
- do {
- int recsize = 0;
- sdp_record_t *rec = sdp_extract_pdu(pdata, pdata_len, &recsize);
- if (rec == NULL) {
- SDPERR("SVC REC is null");
- status = -1;
- goto end;
- }
- if (!recsize) {
- sdp_record_free(rec);
- break;
- }
- scanned += recsize;
- pdata += recsize;
- pdata_len -= recsize;
- SDPDBG("Loc seq length : %d", recsize);
- SDPDBG("Svc Rec Handle : 0x%x", rec->handle);
- SDPDBG("Bytes scanned : %d", scanned);
- SDPDBG("Attrlist byte count : %d", attr_list_len);
- rec_list = sdp_list_append(rec_list, rec);
- } while (scanned < attr_list_len && pdata_len > 0);
- SDPDBG("Successful scan of service attr lists");
- *rsp = rec_list;
- }
- }
- end:
- free(rsp_concat_buf.data);
- free(reqbuf);
- free(rspbuf);
- return status;
- }
- /*
- * Find devices in the piconet.
- */
- int sdp_general_inquiry(inquiry_info *ii, int num_dev, int duration, uint8_t *found)
- {
- int n = hci_inquiry(-1, 10, num_dev, NULL, &ii, 0);
- if (n < 0) {
- SDPERR("Inquiry failed:%m");
- return -1;
- }
- *found = n;
- return 0;
- }
- int sdp_close(sdp_session_t *session)
- {
- struct sdp_transaction *t;
- int ret;
- if (!session)
- return -1;
- ret = close(session->sock);
- t = session->priv;
- if (t) {
- free(t->reqbuf);
- free(t->rsp_concat_buf.data);
- free(t);
- }
- free(session);
- return ret;
- }
- static inline int sdp_is_local(const bdaddr_t *device)
- {
- return memcmp(device, BDADDR_LOCAL, sizeof(bdaddr_t)) == 0;
- }
- static int sdp_connect_local(sdp_session_t *session)
- {
- struct sockaddr_un sa;
- session->sock = socket(PF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0);
- if (session->sock < 0)
- return -1;
- session->local = 1;
- sa.sun_family = AF_UNIX;
- strcpy(sa.sun_path, SDP_UNIX_PATH);
- return connect(session->sock, (struct sockaddr *) &sa, sizeof(sa));
- }
- static int set_l2cap_mtu(int sk, uint16_t mtu)
- {
- struct l2cap_options l2o;
- socklen_t len;
- memset(&l2o, 0, sizeof(l2o));
- len = sizeof(l2o);
- if (getsockopt(sk, SOL_L2CAP, L2CAP_OPTIONS, &l2o, &len) < 0)
- return -1;
- l2o.imtu = mtu;
- l2o.omtu = mtu;
- if (setsockopt(sk, SOL_L2CAP, L2CAP_OPTIONS, &l2o, sizeof(l2o)) < 0)
- return -1;
- return 0;
- }
- static int sdp_connect_l2cap(const bdaddr_t *src,
- const bdaddr_t *dst, sdp_session_t *session)
- {
- uint32_t flags = session->flags;
- struct sockaddr_l2 sa;
- int sk;
- int sockflags = SOCK_SEQPACKET | SOCK_CLOEXEC;
- if (flags & SDP_NON_BLOCKING)
- sockflags |= SOCK_NONBLOCK;
- session->sock = socket(PF_BLUETOOTH, sockflags, BTPROTO_L2CAP);
- if (session->sock < 0)
- return -1;
- session->local = 0;
- sk = session->sock;
- memset(&sa, 0, sizeof(sa));
- sa.l2_family = AF_BLUETOOTH;
- sa.l2_psm = 0;
- if (bacmp(src, BDADDR_ANY)) {
- sa.l2_bdaddr = *src;
- if (bind(sk, (struct sockaddr *) &sa, sizeof(sa)) < 0)
- return -1;
- }
- if (flags & SDP_WAIT_ON_CLOSE) {
- struct linger l = { .l_onoff = 1, .l_linger = 1 };
- if (setsockopt(sk, SOL_SOCKET, SO_LINGER, &l, sizeof(l)) < 0)
- return -1;
- }
- if ((flags & SDP_LARGE_MTU) &&
- set_l2cap_mtu(sk, SDP_LARGE_L2CAP_MTU) < 0)
- return -1;
- sa.l2_psm = htobs(SDP_PSM);
- sa.l2_bdaddr = *dst;
- do {
- int ret = connect(sk, (struct sockaddr *) &sa, sizeof(sa));
- if (!ret)
- return 0;
- if (ret < 0 && (flags & SDP_NON_BLOCKING) &&
- (errno == EAGAIN || errno == EINPROGRESS))
- return 0;
- } while (errno == EBUSY && (flags & SDP_RETRY_IF_BUSY));
- return -1;
- }
- sdp_session_t *sdp_connect(const bdaddr_t *src,
- const bdaddr_t *dst, uint32_t flags)
- {
- sdp_session_t *session;
- int err;
- if ((flags & SDP_RETRY_IF_BUSY) && (flags & SDP_NON_BLOCKING)) {
- errno = EINVAL;
- return NULL;
- }
- session = sdp_create(-1, flags);
- if (!session)
- return NULL;
- if (sdp_is_local(dst)) {
- if (sdp_connect_local(session) < 0)
- goto fail;
- } else {
- if (sdp_connect_l2cap(src, dst, session) < 0)
- goto fail;
- }
- return session;
- fail:
- err = errno;
- if (session->sock >= 0)
- close(session->sock);
- free(session->priv);
- free(session);
- errno = err;
- return NULL;
- }
- int sdp_get_socket(const sdp_session_t *session)
- {
- return session->sock;
- }
- uint16_t sdp_gen_tid(sdp_session_t *session)
- {
- return session->tid++;
- }
- /*
- * Set the supported features
- */
- int sdp_set_supp_feat(sdp_record_t *rec, const sdp_list_t *sf)
- {
- const sdp_list_t *p, *r;
- sdp_data_t *feat, *seq_feat;
- int seqlen, i;
- void **seqDTDs, **seqVals;
- seqlen = sdp_list_len(sf);
- seqDTDs = malloc(seqlen * sizeof(void *));
- if (!seqDTDs)
- return -1;
- seqVals = malloc(seqlen * sizeof(void *));
- if (!seqVals) {
- free(seqDTDs);
- return -1;
- }
- for (p = sf, i = 0; p; p = p->next, i++) {
- int plen, j;
- void **dtds, **vals;
- int *lengths;
- plen = sdp_list_len(p->data);
- dtds = malloc(plen * sizeof(void *));
- if (!dtds)
- goto fail;
- vals = malloc(plen * sizeof(void *));
- if (!vals) {
- free(dtds);
- goto fail;
- }
- lengths = malloc(plen * sizeof(int));
- if (!lengths) {
- free(dtds);
- free(vals);
- goto fail;
- }
- for (r = p->data, j = 0; r; r = r->next, j++) {
- sdp_data_t *data = (sdp_data_t *) r->data;
- dtds[j] = &data->dtd;
- switch (data->dtd) {
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- vals[j] = data->val.str;
- lengths[j] = data->unitSize - sizeof(uint8_t);
- break;
- case SDP_ALT8:
- case SDP_ALT16:
- case SDP_ALT32:
- case SDP_SEQ8:
- case SDP_SEQ16:
- case SDP_SEQ32:
- vals[j] = data->val.dataseq;
- lengths[j] = 0;
- break;
- default:
- vals[j] = &data->val;
- lengths[j] = 0;
- break;
- }
- }
- feat = sdp_seq_alloc_with_length(dtds, vals, lengths, plen);
- free(dtds);
- free(vals);
- free(lengths);
- if (!feat)
- goto fail;
- seqDTDs[i] = &feat->dtd;
- seqVals[i] = feat;
- }
- seq_feat = sdp_seq_alloc(seqDTDs, seqVals, seqlen);
- if (!seq_feat)
- goto fail;
- sdp_attr_replace(rec, SDP_ATTR_SUPPORTED_FEATURES_LIST, seq_feat);
- free(seqVals);
- free(seqDTDs);
- return 0;
- fail:
- free(seqVals);
- free(seqDTDs);
- return -1;
- }
- /*
- * Get the supported features
- * If an error occurred -1 is returned and errno is set
- */
- int sdp_get_supp_feat(const sdp_record_t *rec, sdp_list_t **seqp)
- {
- sdp_data_t *sdpdata, *d;
- sdp_list_t *tseq;
- tseq = NULL;
- sdpdata = sdp_data_get(rec, SDP_ATTR_SUPPORTED_FEATURES_LIST);
- if (!sdpdata || !SDP_IS_SEQ(sdpdata->dtd))
- return sdp_get_uuidseq_attr(rec,
- SDP_ATTR_SUPPORTED_FEATURES_LIST, seqp);
- for (d = sdpdata->val.dataseq; d; d = d->next) {
- sdp_data_t *dd;
- sdp_list_t *subseq;
- if (!SDP_IS_SEQ(d->dtd))
- goto fail;
- subseq = NULL;
- for (dd = d->val.dataseq; dd; dd = dd->next) {
- sdp_data_t *data;
- void *val;
- int length;
- switch (dd->dtd) {
- case SDP_URL_STR8:
- case SDP_URL_STR16:
- case SDP_TEXT_STR8:
- case SDP_TEXT_STR16:
- val = dd->val.str;
- length = dd->unitSize - sizeof(uint8_t);
- break;
- case SDP_UINT8:
- case SDP_UINT16:
- val = &dd->val;
- length = 0;
- break;
- default:
- sdp_list_free(subseq, free);
- goto fail;
- }
- data = sdp_data_alloc_with_length(dd->dtd, val, length);
- if (data)
- subseq = sdp_list_append(subseq, data);
- }
- tseq = sdp_list_append(tseq, subseq);
- }
- *seqp = tseq;
- return 0;
- fail:
- while (tseq) {
- sdp_list_t * next;
- next = tseq->next;
- sdp_list_free(tseq, free);
- tseq = next;
- }
- errno = EINVAL;
- return -1;
- }
- void sdp_add_lang_attr(sdp_record_t *rec)
- {
- sdp_lang_attr_t base_lang;
- sdp_list_t *langs;
- base_lang.code_ISO639 = (0x65 << 8) | 0x6e;
- base_lang.encoding = 106;
- base_lang.base_offset = SDP_PRIMARY_LANG_BASE;
- langs = sdp_list_append(0, &base_lang);
- sdp_set_lang_attr(rec, langs);
- sdp_list_free(langs, NULL);
- }
|