%% %% %CopyrightBegin% %% %% Copyright Ericsson AB 2006-2016. All Rights Reserved. %% %% Licensed under the Apache License, Version 2.0 (the "License"); %% you may not use this file except in compliance with the License. %% You may obtain a copy of the License at %% %% http://www.apache.org/licenses/LICENSE-2.0 %% %% Unless required by applicable law or agreed to in writing, software %% distributed under the License is distributed on an "AS IS" BASIS, %% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %% See the License for the specific language governing permissions and %% limitations under the License. %% %% %CopyrightEnd% %% -module(bsdecode). -export([?MODULE/0]). -record(protocolErrors, {invalidManIE = false, outOfSequence = false, incorrectOptIE = false}). -record(mvsT_msisdn, {value}). -record(mvsgT_pdpAddressType, {pdpTypeNbr, address}). -record(mvsgT_ipAddress, {version, a1, a2, a3, a4, a5, a6, a7, a8}). -record(mvsgT_imsi, {value}). -record(mvsgT_tid, {imsi, nsapi}). -record(sesT_qualityOfServiceV0, {delayClass, reliabilityClass, peakThroughput, precedenceClass, meanThroughput}). -record(sesT_deleteReqV0, {tid}). -record(sesT_deleteResV0, {tid, cause}). -record(sesT_createReqV0, {tid, tidRaw, qos, recovery, selMode, flowLblData, flowLblSig, endUserAdd, accPointName, protConOpt, sgsnAddSig, sgsnAddUser, msisdn}). -record(sesT_updateReqV0, {tid, tidRaw, qos, recovery, flowLblData, flowLblSig, sgsnAddSig, sgsnAddUser}). -record(masT_ipcpData, {type, ipAddress, rawMessage}). -record(masT_ipcp, {exists, code, id, ipcpList}). -record(masT_pap, {exists, code, id, username, password}). -record(masT_chap, {code, id, value, name}). -record(masT_protocolConfigOptions, {chap, pap, ipcp}). ?MODULE() -> Res = test(), {Res,Res =:= {ok,{sesT_createReqV0,{mvsgT_tid,{mvsgT_imsi,<<81,67,101,7,0,0,0,240>>},6}, [81,67,101,7,0,0,0,96], {sesT_qualityOfServiceV0,1,4,9,2,18},0, subscribed,0,0,{mvsgT_pdpAddressType,ietf_ipv4,[]}, [<<97,112,110,48,49,51,97>>,<<101,114,105,99,115,115,111,110>>,<<115,101>>], {masT_protocolConfigOptions,[], {masT_pap,true,1,5,[117,115,101,114,53],[112,97,115,115,53]},[]}, {mvsgT_ipAddress,ipv4,172,28,12,1,0,0,0,0}, {mvsgT_ipAddress,ipv4,172,28,12,3,0,0,0,0}, {mvsT_msisdn,<<145,148,113,129,0,0,0,0>>}},1}}. test() -> Pdu = <<30, 16, 0, 90, 0, 1, 0, 0, 255, 255, 255, 255, 81, 67, 101, 7, 0, 0, 0, 96, 6, 12, 146, 18, 14, 0, 15, 252, 16, 0, 0, 17, 0, 0, 128, 0, 2, 241, 33, 131, 0, 20, 7, 97, 112, 110, 48, 49, 51, 97, 8, 101, 114, 105, 99, 115, 115, 111, 110, 2, 115, 101, 132, 0, 20, 128, 192, 35, 16, 1, 5, 0, 16, 5, 117, 115, 101, 114, 53, 5, 112, 97, 115, 115, 53, 133, 0, 4, 172, 28, 12, 1, 133, 0, 4, 172, 28, 12, 3, 134, 0, 8, 145, 148, 113, 129, 0, 0, 0, 0>>, decode_v0_opt(10,Pdu). decode_v0_opt(0,Pdu) -> decode_gtpc_msg(Pdu); decode_v0_opt(N,Pdu) -> decode_gtpc_msg(Pdu), decode_v0_opt(N - 1,Pdu). decode_gtpc_msg(<<0:3, _:4, 0:1, 16:8, _Length:16, SequenceNumber:16, _FlowLabel:16, _SNDCP_N_PDU_Number:8, _:3/binary-unit:8, TID:8/binary-unit:8, InformationElements/binary>>) -> Errors = #protocolErrors{}, {ok,TID2} = tid_internal_storage(TID,[]), EmptyCreateReq = #sesT_createReqV0{tid = TID2, tidRaw = binary_to_list(TID)}, case catch decode_ie_create(InformationElements,0,Errors,EmptyCreateReq) of {ok,CreateReq} -> {ok,CreateReq,SequenceNumber}; {fault,Cause,CreateReq} -> {fault,Cause,CreateReq,SequenceNumber}; {'EXIT',_Reason} -> {fault,193,EmptyCreateReq,SequenceNumber} end; decode_gtpc_msg(<<0:3, _:4, 0:1, 18:8, _Length:16, SequenceNumber:16, _FlowLabel:16, _SNDCP_N_PDU_Number:8, _:3/binary-unit:8, TID:8/binary-unit:8, InformationElements/binary>>) -> io:format("hej",[]), Errors = #protocolErrors{}, {ok,TID2} = tid_internal_storage(TID,[]), EmptyUpdateReq = #sesT_updateReqV0{tid = TID2, tidRaw = binary_to_list(TID)}, case catch decode_ie_update(InformationElements,0,Errors,EmptyUpdateReq) of {ok,UpdateReq} -> {ok,UpdateReq,SequenceNumber}; {fault,Cause,UpdateReq} -> {fault,Cause,UpdateReq,SequenceNumber}; {'EXIT',Reason} -> io:format("hej",[]), {fault,193,EmptyUpdateReq,SequenceNumber,Reason} end; decode_gtpc_msg(<<0:3, _:4, 0:1, 20:8, _Length:16, SequenceNumber:16, _FlowLabel:16, _SNDCP_N_PDU_Number:8, _:3/binary-unit:8, TID:8/binary-unit:8, _InformationElements/binary>>) -> {ok,TID2} = tid_internal_storage(TID,[]), DeleteReq = #sesT_deleteReqV0{tid = TID2}, {ok,DeleteReq,SequenceNumber}; decode_gtpc_msg(<<0:3, _:4, 0:1, 21:8, _Length:16, SequenceNumber:16, _FlowLabel:16, _SNDCP_N_PDU_Number:8, _:3/binary-unit:8, TID:8/binary-unit:8, InformationElements/binary>>) -> Errors = #protocolErrors{}, {ok,TID2} = tid_internal_storage(TID,[]), EmptyDeleteRes = #sesT_deleteResV0{tid = TID2}, case catch decode_ie_delete_res(InformationElements,0,Errors,EmptyDeleteRes) of {ok,DeleteRes} -> {ok,DeleteRes,SequenceNumber}; {fault,Cause,DeleteRes} -> {fault,Cause,DeleteRes,SequenceNumber}; {'EXIT',_Reason} -> {fault,193,EmptyDeleteRes,SequenceNumber} end; decode_gtpc_msg(_GTP_C_Message) -> {fault}. decode_ie_create(<<>>,PresentIEs,Errors,CreateReq) -> if PresentIEs band 1917 /= 1917 -> {fault,202,CreateReq}; true -> case Errors of #protocolErrors{invalidManIE = true} -> {fault,201,CreateReq}; #protocolErrors{outOfSequence = true} -> {fault,193,CreateReq}; #protocolErrors{incorrectOptIE = true} -> {fault,203,CreateReq}; _ -> {ok,CreateReq} end end; decode_ie_create(<<6:8, QoSElement:3/binary-unit:8, Rest/binary>>,PresentIEs,Errors,CreateReq) -> if PresentIEs band 1 == 1 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 1 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, <<_:2, DelayClass:3, ReliabilityClass:3, PeakThroughput:4, _:1, PrecedenceClass:3, _:3, MeanThroughput:5>> = QoSElement, QoS = #sesT_qualityOfServiceV0{delayClass = DelayClass, reliabilityClass = ReliabilityClass, peakThroughput = PeakThroughput, precedenceClass = PrecedenceClass, meanThroughput = MeanThroughput}, UpdatedCreateReq = CreateReq#sesT_createReqV0{qos = QoS}, decode_ie_create(Rest,PresentIEs bor 1,UpdatedErrors,UpdatedCreateReq); true -> <<_:2, DelayClass:3, ReliabilityClass:3, PeakThroughput:4, _:1, PrecedenceClass:3, _:3, MeanThroughput:5>> = QoSElement, QoS = #sesT_qualityOfServiceV0{delayClass = DelayClass, reliabilityClass = ReliabilityClass, peakThroughput = PeakThroughput, precedenceClass = PrecedenceClass, meanThroughput = MeanThroughput}, UpdatedCreateReq = CreateReq#sesT_createReqV0{qos = QoS}, decode_ie_create(Rest,PresentIEs bor 1,Errors,UpdatedCreateReq) end; decode_ie_create(<<14:8, Recovery:8, Rest/binary>>,PresentIEs,Errors,CreateReq) -> if PresentIEs band 2 == 2 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 2 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{recovery = Recovery}, decode_ie_create(Rest,PresentIEs bor 2,UpdatedErrors,UpdatedCreateReq); true -> UpdatedCreateReq = CreateReq#sesT_createReqV0{recovery = Recovery}, decode_ie_create(Rest,PresentIEs bor 2,Errors,UpdatedCreateReq) end; decode_ie_create(<<15:8, _:6, SelectionMode:2, Rest/binary>>,PresentIEs,Errors,CreateReq) -> if PresentIEs band 4 == 4 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 4 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{selMode = selection_mode_internal_storage(SelectionMode)}, decode_ie_create(Rest,PresentIEs bor 4,UpdatedErrors,UpdatedCreateReq); true -> UpdatedCreateReq = CreateReq#sesT_createReqV0{selMode = selection_mode_internal_storage(SelectionMode)}, decode_ie_create(Rest,PresentIEs bor 4,Errors,UpdatedCreateReq) end; decode_ie_create(<<16:8, FlowLabel:16, Rest/binary>>,PresentIEs,Errors,CreateReq) -> if PresentIEs band 8 == 8 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 8 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{flowLblData = FlowLabel}, decode_ie_create(Rest,PresentIEs bor 8,UpdatedErrors,UpdatedCreateReq); true -> UpdatedCreateReq = CreateReq#sesT_createReqV0{flowLblData = FlowLabel}, decode_ie_create(Rest,PresentIEs bor 8,Errors,UpdatedCreateReq) end; decode_ie_create(<<17:8, FlowLabel:16, Rest/binary>>,PresentIEs,Errors,CreateReq) -> if PresentIEs band 16 == 16 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 16 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{flowLblSig = FlowLabel}, decode_ie_create(Rest,PresentIEs bor 16,UpdatedErrors,UpdatedCreateReq); true -> UpdatedCreateReq = CreateReq#sesT_createReqV0{flowLblSig = FlowLabel}, decode_ie_create(Rest,PresentIEs bor 16,Errors,UpdatedCreateReq) end; decode_ie_create(<<128:8, Length:16, More/binary>>,PresentIEs,Errors,CreateReq) -> <<PDPElement:Length/binary-unit:8, Rest/binary>> = More, if PresentIEs band 32 == 32 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 32 -> case pdp_addr_internal_storage(PDPElement) of {ok,PDPAddress} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{endUserAdd = PDPAddress}, decode_ie_create(Rest,PresentIEs bor 32,UpdatedErrors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true, outOfSequence = true}, decode_ie_create(Rest,PresentIEs bor 32,UpdatedErrors,CreateReq) end; true -> case pdp_addr_internal_storage(PDPElement) of {ok,PDPAddress} -> UpdatedCreateReq = CreateReq#sesT_createReqV0{endUserAdd = PDPAddress}, decode_ie_create(Rest,PresentIEs bor 32,Errors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true}, decode_ie_create(Rest,PresentIEs bor 32,UpdatedErrors,CreateReq) end end; decode_ie_create(<<131:8, Length:16, More/binary>>,PresentIEs,Errors,CreateReq) -> <<APNElement:Length/binary-unit:8, Rest/binary>> = More, if PresentIEs band 64 == 64 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 64 -> case catch apn_internal_storage(APNElement,[]) of {ok,APN} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{accPointName = APN}, decode_ie_create(Rest,PresentIEs bor 64,UpdatedErrors,UpdatedCreateReq); _ -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true, invalidManIE = true}, decode_ie_create(Rest,PresentIEs bor 64,UpdatedErrors,CreateReq) end; true -> case catch apn_internal_storage(APNElement,[]) of {ok,APN} -> UpdatedCreateReq = CreateReq#sesT_createReqV0{accPointName = APN}, decode_ie_create(Rest,PresentIEs bor 64,Errors,UpdatedCreateReq); _ -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true}, decode_ie_create(Rest,PresentIEs bor 64,UpdatedErrors,CreateReq) end end; decode_ie_create(<<132:8, Length:16, More/binary>>,PresentIEs,Errors,CreateReq) -> <<ConfigurationElement:Length/binary-unit:8, Rest/binary>> = More, if PresentIEs band 128 == 128 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 128 -> case catch pco_internal_storage(ConfigurationElement) of {ok,PCO} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{protConOpt = PCO}, decode_ie_create(Rest,PresentIEs bor 128,UpdatedErrors,UpdatedCreateReq); _ -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true, incorrectOptIE = true}, decode_ie_create(Rest,PresentIEs bor 128,UpdatedErrors,CreateReq) end; true -> case catch pco_internal_storage(ConfigurationElement) of {ok,PCO} -> UpdatedCreateReq = CreateReq#sesT_createReqV0{protConOpt = PCO}, decode_ie_create(Rest,PresentIEs bor 128,Errors,UpdatedCreateReq); _ -> UpdatedErrors = Errors#protocolErrors{incorrectOptIE = true}, decode_ie_create(Rest,PresentIEs bor 128,UpdatedErrors,CreateReq) end end; decode_ie_create(<<133:8, Length:16, More/binary>>,PresentIEs,Errors,CreateReq) -> <<AddressElement:Length/binary-unit:8, Rest/binary>> = More, if PresentIEs band 768 == 768 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 512 -> if PresentIEs band 256 == 0 -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{sgsnAddSig = GSNAddr}, decode_ie_create(Rest,PresentIEs bor 256,UpdatedErrors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true, outOfSequence = true}, decode_ie_create(Rest,PresentIEs bor 256,UpdatedErrors,CreateReq) end; true -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{sgsnAddUser = GSNAddr}, decode_ie_create(Rest,PresentIEs bor 512,UpdatedErrors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true, outOfSequence = true}, decode_ie_create(Rest,PresentIEs bor 512,UpdatedErrors,CreateReq) end end; PresentIEs < 256 -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedCreateReq = CreateReq#sesT_createReqV0{sgsnAddSig = GSNAddr}, decode_ie_create(Rest,PresentIEs bor 256,Errors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true}, decode_ie_create(Rest,PresentIEs bor 256,UpdatedErrors,CreateReq) end; true -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedCreateReq = CreateReq#sesT_createReqV0{sgsnAddUser = GSNAddr}, decode_ie_create(Rest,PresentIEs bor 512,Errors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true}, decode_ie_create(Rest,PresentIEs bor 512,UpdatedErrors,CreateReq) end end; decode_ie_create(<<134:8, Length:16, More/binary>>,PresentIEs,Errors,CreateReq) -> <<MSISDNElement:Length/binary-unit:8, Rest/binary>> = More, if PresentIEs band 1024 == 1024 -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); PresentIEs > 1024 -> case msisdn_internal_storage(MSISDNElement,[]) of {ok,MSISDN} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedCreateReq = CreateReq#sesT_createReqV0{msisdn = MSISDN}, decode_ie_create(Rest,PresentIEs bor 1024,UpdatedErrors,UpdatedCreateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true, invalidManIE = true}, decode_ie_create(Rest,PresentIEs bor 1024,UpdatedErrors,CreateReq) end; true -> UpdatedCreateReq = CreateReq#sesT_createReqV0{msisdn = #mvsT_msisdn{value = MSISDNElement}}, decode_ie_create(Rest,PresentIEs bor 1024,Errors,UpdatedCreateReq) end; decode_ie_create(UnexpectedIE,PresentIEs,Errors,CreateReq) -> case check_ie(UnexpectedIE) of {defined_ie,Rest} -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); {handled_ie,Rest} -> decode_ie_create(Rest,PresentIEs,Errors,CreateReq); {unhandled_ie} -> {fault,193,CreateReq} end. decode_ie_update(<<>>,PresentIEs,Errors,UpdateReq) -> if PresentIEs band 61 /= 61 -> {fault,202,UpdateReq}; true -> case Errors of #protocolErrors{invalidManIE = true} -> {fault,201,UpdateReq}; #protocolErrors{outOfSequence = true} -> {fault,193,UpdateReq}; #protocolErrors{incorrectOptIE = true} -> {fault,203,UpdateReq}; _ -> {ok,UpdateReq} end end; decode_ie_update(<<6:8, QoSElement:3/binary-unit:8, Rest/binary>>,PresentIEs,Errors,UpdateReq) -> if PresentIEs band 1 == 1 -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); PresentIEs > 1 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, <<_:2, DelayClass:3, ReliabilityClass:3, PeakThroughput:4, _:1, PrecedenceClass:3, _:3, MeanThroughput:5>> = QoSElement, QoS = #sesT_qualityOfServiceV0{delayClass = DelayClass, reliabilityClass = ReliabilityClass, peakThroughput = PeakThroughput, precedenceClass = PrecedenceClass, meanThroughput = MeanThroughput}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{qos = QoS}, decode_ie_update(Rest,PresentIEs bor 1,UpdatedErrors,UpdatedUpdateReq); true -> <<_:2, DelayClass:3, ReliabilityClass:3, PeakThroughput:4, _:1, PrecedenceClass:3, _:3, MeanThroughput:5>> = QoSElement, QoS = #sesT_qualityOfServiceV0{delayClass = DelayClass, reliabilityClass = ReliabilityClass, peakThroughput = PeakThroughput, precedenceClass = PrecedenceClass, meanThroughput = MeanThroughput}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{qos = QoS}, decode_ie_update(Rest,PresentIEs bor 1,Errors,UpdatedUpdateReq) end; decode_ie_update(<<14:8, Recovery:8, Rest/binary>>,PresentIEs,Errors,UpdateReq) -> if PresentIEs band 2 == 2 -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); PresentIEs > 2 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{recovery = Recovery}, decode_ie_update(Rest,PresentIEs bor 2,UpdatedErrors,UpdatedUpdateReq); true -> UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{recovery = Recovery}, decode_ie_update(Rest,PresentIEs bor 2,Errors,UpdatedUpdateReq) end; decode_ie_update(<<16:8, FlowLabel:16, Rest/binary>>,PresentIEs,Errors,UpdateReq) -> if PresentIEs band 4 == 4 -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); PresentIEs > 4 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{flowLblData = FlowLabel}, decode_ie_update(Rest,PresentIEs bor 4,UpdatedErrors,UpdatedUpdateReq); true -> UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{flowLblData = FlowLabel}, decode_ie_update(Rest,PresentIEs bor 4,Errors,UpdatedUpdateReq) end; decode_ie_update(<<17:8, FlowLabel:16, Rest/binary>>,PresentIEs,Errors,UpdateReq) -> if PresentIEs band 8 == 8 -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); PresentIEs > 8 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{flowLblSig = FlowLabel}, decode_ie_update(Rest,PresentIEs bor 8,UpdatedErrors,UpdatedUpdateReq); true -> UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{flowLblSig = FlowLabel}, decode_ie_update(Rest,PresentIEs bor 8,Errors,UpdatedUpdateReq) end; decode_ie_update(<<133:8, Length:16, More/binary>>,PresentIEs,Errors,UpdateReq) -> <<AddressElement:Length/binary-unit:8, Rest/binary>> = More, if PresentIEs band 48 == 48 -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); PresentIEs > 32 -> if PresentIEs band 16 == 0 -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{sgsnAddSig = GSNAddr}, decode_ie_update(Rest,PresentIEs bor 16,UpdatedErrors,UpdatedUpdateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true, outOfSequence = true}, decode_ie_update(Rest,PresentIEs bor 16,UpdatedErrors,UpdateReq) end; true -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{sgsnAddUser = GSNAddr}, decode_ie_update(Rest,PresentIEs bor 32,UpdatedErrors,UpdatedUpdateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true, outOfSequence = true}, decode_ie_update(Rest,PresentIEs bor 32,UpdatedErrors,UpdateReq) end end; PresentIEs < 16 -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{sgsnAddSig = GSNAddr}, decode_ie_update(Rest,PresentIEs bor 16,Errors,UpdatedUpdateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true}, decode_ie_update(Rest,PresentIEs bor 16,UpdatedErrors,UpdateReq) end; true -> case gsn_addr_internal_storage(AddressElement) of {ok,GSNAddr} -> UpdatedUpdateReq = UpdateReq#sesT_updateReqV0{sgsnAddUser = GSNAddr}, decode_ie_update(Rest,PresentIEs bor 32,Errors,UpdatedUpdateReq); {fault} -> UpdatedErrors = Errors#protocolErrors{invalidManIE = true}, decode_ie_update(Rest,PresentIEs bor 32,UpdatedErrors,UpdateReq) end end; decode_ie_update(UnexpectedIE,PresentIEs,Errors,UpdateReq) -> case check_ie(UnexpectedIE) of {defined_ie,Rest} -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); {handled_ie,Rest} -> decode_ie_update(Rest,PresentIEs,Errors,UpdateReq); {unhandled_ie} -> {fault,193,UpdateReq} end. decode_ie_delete_res(<<>>,PresentIEs,Errors,DeleteRes) -> if PresentIEs band 1 /= 1 -> {fault,202,DeleteRes}; true -> case Errors of #protocolErrors{invalidManIE = true} -> {fault,201,DeleteRes}; #protocolErrors{outOfSequence = true} -> {fault,193,DeleteRes}; #protocolErrors{incorrectOptIE = true} -> {fault,203,DeleteRes}; _ -> {ok,DeleteRes} end end; decode_ie_delete_res(<<1:8, Cause:8, Rest/binary>>,PresentIEs,Errors,DeleteRes) -> if PresentIEs band 1 == 1 -> decode_ie_delete_res(Rest,PresentIEs,Errors,DeleteRes); PresentIEs > 1 -> UpdatedErrors = Errors#protocolErrors{outOfSequence = true}, UpdatedDeleteRes = DeleteRes#sesT_deleteResV0{cause = Cause}, decode_ie_delete_res(Rest,PresentIEs bor 1,UpdatedErrors,UpdatedDeleteRes); true -> UpdatedDeleteRes = DeleteRes#sesT_deleteResV0{cause = Cause}, decode_ie_delete_res(Rest,PresentIEs bor 1,Errors,UpdatedDeleteRes) end; decode_ie_delete_res(UnexpectedIE,PresentIEs,Errors,DeleteRes) -> case check_ie(UnexpectedIE) of {defined_ie,Rest} -> decode_ie_delete_res(Rest,PresentIEs,Errors,DeleteRes); {handled_ie,Rest} -> decode_ie_delete_res(Rest,PresentIEs,Errors,DeleteRes); {unhandled_ie} -> {fault,193,DeleteRes} end. check_ie(<<1:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<2:8, _:8/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<3:8, _:6/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<4:8, _:4/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<5:8, _:4/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<6:8, _:3/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<8:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<9:8, _:28/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<11:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<12:8, _:3/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<13:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<14:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<15:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<16:8, _:16, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<17:8, _:16, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<18:8, _:32, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<19:8, _:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<127:8, _:4/binary-unit:8, Rest/binary>>) -> {defined_ie,Rest}; check_ie(<<1:1, _:7, Length:16, More/binary>>) -> if Length > size(More) -> {unhandled_ie}; true -> <<_:Length/binary-unit:8, Rest/binary>> = More, {handled_ie,Rest} end; check_ie(_UnhandledIE) -> {unhandled_ie}. tid_internal_storage(Bin,_) -> Size = size(Bin) - 1, <<Front:Size/binary, NSAPI:4, DigitN:4>> = Bin, Result = case DigitN of 15 -> #mvsgT_tid{imsi = #mvsgT_imsi{value = Front}, nsapi = NSAPI}; _ -> #mvsgT_tid{imsi = #mvsgT_imsi{value = <<Front/binary, 15:4, DigitN:4>>}, nsapi = NSAPI} end, {ok,Result}. selection_mode_internal_storage(0) -> subscribed; selection_mode_internal_storage(1) -> msRequested; selection_mode_internal_storage(2) -> sgsnSelected; selection_mode_internal_storage(3) -> sgsnSelected. pdp_addr_internal_storage(<<_:4, 0:4, 1:8>>) -> {ok,#mvsgT_pdpAddressType{pdpTypeNbr = etsi_ppp, address = []}}; pdp_addr_internal_storage(<<_:4, 0:4, 2:8>>) -> {ok,#mvsgT_pdpAddressType{pdpTypeNbr = etsi_osp_ihoss, address = []}}; pdp_addr_internal_storage(<<_:4, 1:4, 33:8>>) -> {ok,#mvsgT_pdpAddressType{pdpTypeNbr = ietf_ipv4, address = []}}; pdp_addr_internal_storage(<<_:4, 1:4, 33:8, IP_A:8, IP_B:8, IP_C:8, IP_D:8>>) -> {ok,#mvsgT_pdpAddressType{pdpTypeNbr = ietf_ipv4, address = [IP_A,IP_B,IP_C,IP_D]}}; pdp_addr_internal_storage(<<_:4, 1:4, 87:8, IP_A:16, IP_B:16, IP_C:16, IP_D:16, IP_E:16, IP_F:16, IP_G:16, IP_H:16>>) -> {ok,#mvsgT_pdpAddressType{pdpTypeNbr = ietf_ipv6, address = [IP_A,IP_B,IP_C,IP_D,IP_E,IP_F,IP_G,IP_H]}}; pdp_addr_internal_storage(_PDP_ADDR) -> {fault}. apn_internal_storage(<<>>,APN) -> {ok,lists:reverse(APN)}; apn_internal_storage(<<Length:8, Rest/binary>>,APN) -> <<Label:Length/binary-unit:8, MoreAPNLabels/binary>> = Rest, apn_internal_storage(MoreAPNLabels,[Label|APN]). pco_internal_storage(<<1:1, _:4, 0:3, PPPConfigurationOptions/binary>>) -> case ppp_configuration_options(PPPConfigurationOptions,#masT_pap{exists = false},[],[]) of {ok,PAP,CHAP,IPCP} -> {ok,#masT_protocolConfigOptions{pap = PAP, chap = CHAP, ipcp = IPCP}}; {fault} -> {fault} end; pco_internal_storage(<<1:1, _:4, 1:3, _OSP_IHOSSConfigurationOptions/binary>>) -> {ok,osp_ihoss}; pco_internal_storage(_UnknownConfigurationOptions) -> {fault}. ppp_configuration_options(<<>>,PAP,CHAP,IPCP) -> {ok,PAP,CHAP,IPCP}; ppp_configuration_options(<<49185:16, Length:8, More/binary>>,PAP,CHAP,IPCP) -> <<_LCP:Length/binary-unit:8, Rest/binary>> = More, ppp_configuration_options(Rest,PAP,CHAP,IPCP); ppp_configuration_options(<<49187:16, _Length:8, 1:8, Identifier:8, DataLength:16, More/binary>>,_PAP,CHAP,IPCP) -> ActualDataLength = DataLength - 4, <<Data:ActualDataLength/binary-unit:8, Rest/binary>> = More, <<PeerIDLength:8, PeerData/binary>> = Data, <<PeerID:PeerIDLength/binary-unit:8, PasswdLength:8, PasswordData/binary>> = PeerData, <<Password:PasswdLength/binary, _Padding/binary>> = PasswordData, ppp_configuration_options(Rest,#masT_pap{exists = true, code = 1, id = Identifier, username = binary_to_list(PeerID), password = binary_to_list(Password)},CHAP,IPCP); ppp_configuration_options(<<49187:16, Length:8, More/binary>>,PAP,CHAP,IPCP) -> <<PAP:Length/binary-unit:8, Rest/binary>> = More, ppp_configuration_options(Rest,PAP,CHAP,IPCP); ppp_configuration_options(<<49699:16, _Length:8, 1:8, Identifier:8, DataLength:16, More/binary>>,PAP,CHAP,IPCP) -> ActualDataLength = DataLength - 4, <<Data:ActualDataLength/binary-unit:8, Rest/binary>> = More, <<ValueSize:8, ValueAndName/binary>> = Data, <<Value:ValueSize/binary-unit:8, Name/binary>> = ValueAndName, ppp_configuration_options(Rest,PAP,[#masT_chap{code = 1, id = Identifier, value = binary_to_list(Value), name = binary_to_list(Name)}|CHAP],IPCP); ppp_configuration_options(<<49699:16, _Length:8, 2:8, Identifier:8, DataLength:16, More/binary>>,PAP,CHAP,IPCP) -> ActualDataLength = DataLength - 4, <<Data:ActualDataLength/binary-unit:8, Rest/binary>> = More, <<ValueSize:8, ValueAndName/binary>> = Data, <<Value:ValueSize/binary-unit:8, Name/binary>> = ValueAndName, ppp_configuration_options(Rest,PAP,[#masT_chap{code = 2, id = Identifier, value = binary_to_list(Value), name = binary_to_list(Name)}|CHAP],IPCP); ppp_configuration_options(<<49699:16, Length:8, More/binary>>,PAP,CHAP,IPCP) -> <<CHAP:Length/binary-unit:8, Rest/binary>> = More, ppp_configuration_options(Rest,PAP,CHAP,IPCP); ppp_configuration_options(<<32801:16, _Length:8, 1:8, Identifier:8, OptionsLength:16, More/binary>>,PAP,CHAP,IPCP) -> ActualOptionsLength = OptionsLength - 4, <<Options:ActualOptionsLength/binary-unit:8, Rest/binary>> = More, case Options of <<3:8, 6:8, A1:8, A2:8, A3:8, A4:8>> -> ppp_configuration_options(Rest,PAP,CHAP,[#masT_ipcp{exists = true, code = 1, id = Identifier, ipcpList = [#masT_ipcpData{type = 3, ipAddress = #mvsgT_ipAddress{version = ipv4, a1 = A1, a2 = A2, a3 = A3, a4 = A4, a5 = 0, a6 = 0, a7 = 0, a8 = 0}, rawMessage = binary_to_list(Options)}]}|IPCP]); <<129:8, 6:8, B1:8, B2:8, B3:8, B4:8>> -> ppp_configuration_options(Rest,PAP,CHAP,[#masT_ipcp{exists = true, code = 1, id = Identifier, ipcpList = [#masT_ipcpData{type = 129, ipAddress = #mvsgT_ipAddress{version = ipv4, a1 = B1, a2 = B2, a3 = B3, a4 = B4}, rawMessage = binary_to_list(Options)}]}|IPCP]); <<131:8, 6:8, C1:8, C2:8, C3:8, C4:8>> -> ppp_configuration_options(Rest,PAP,CHAP,[#masT_ipcp{exists = true, code = 1, id = Identifier, ipcpList = [#masT_ipcpData{type = 131, ipAddress = #mvsgT_ipAddress{version = ipv4, a1 = C1, a2 = C2, a3 = C3, a4 = C4}, rawMessage = binary_to_list(Options)}]}|IPCP]); _ -> ppp_configuration_options(Rest,PAP,CHAP,IPCP) end; ppp_configuration_options(<<_UnknownProtocolID:16, Length:8, More/binary>>,PAP,CHAP,IPCP) -> <<_Skipped:Length/binary-unit:8, Rest/binary>> = More, ppp_configuration_options(Rest,PAP,CHAP,IPCP); ppp_configuration_options(_Unhandled,_PAP,_CHAP,_IPCP) -> {fault}. gsn_addr_internal_storage(<<IP_A:8, IP_B:8, IP_C:8, IP_D:8>>) -> {ok,#mvsgT_ipAddress{version = ipv4, a1 = IP_A, a2 = IP_B, a3 = IP_C, a4 = IP_D, a5 = 0, a6 = 0, a7 = 0, a8 = 0}}; gsn_addr_internal_storage(<<IP_A:16, IP_B:16, IP_C:16, IP_D:16, IP_E:16, IP_F:16, IP_G:16, IP_H:16>>) -> {ok,#mvsgT_ipAddress{version = ipv6, a1 = IP_A, a2 = IP_B, a3 = IP_C, a4 = IP_D, a5 = IP_E, a6 = IP_F, a7 = IP_G, a8 = IP_H}}; gsn_addr_internal_storage(_GSN_ADDR) -> {fault}. msisdn_internal_storage(<<>>,MSISDN) -> {ok,#mvsT_msisdn{value = lists:reverse(MSISDN)}}; msisdn_internal_storage(<<255:8, _Rest/binary>>,MSISDN) -> {ok,#mvsT_msisdn{value = lists:reverse(MSISDN)}}; msisdn_internal_storage(<<15:4, DigitN:4, _Rest/binary>>,MSISDN) when DigitN < 10 -> {ok,#mvsT_msisdn{value = lists:reverse([DigitN bor 240|MSISDN])}}; msisdn_internal_storage(<<DigitNplus1:4, DigitN:4, Rest/binary>>,MSISDN) when DigitNplus1 < 10, DigitN < 10 -> NewMSISDN = [DigitNplus1 bsl 4 bor DigitN|MSISDN], msisdn_internal_storage(Rest,NewMSISDN); msisdn_internal_storage(_Rest,_MSISDN) -> {fault}.