%%
%% %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}.