Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
O
OpenXG-RAN
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wangjie
OpenXG-RAN
Commits
d38501c9
Commit
d38501c9
authored
Jul 11, 2018
by
Matthieu Kanj
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
test version
parent
008f825c
Changes
7
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
195 additions
and
80 deletions
+195
-80
openair1/PHY/CODING/ccoding_byte_NB_IoT.c
openair1/PHY/CODING/ccoding_byte_NB_IoT.c
+20
-2
openair1/PHY/LTE_TRANSPORT/defs_NB_IoT.h
openair1/PHY/LTE_TRANSPORT/defs_NB_IoT.h
+4
-5
openair1/PHY/LTE_TRANSPORT/dlsch_coding_NB_IoT.c
openair1/PHY/LTE_TRANSPORT/dlsch_coding_NB_IoT.c
+14
-9
openair1/PHY/LTE_TRANSPORT/dlsch_modulation_NB_IoT.c
openair1/PHY/LTE_TRANSPORT/dlsch_modulation_NB_IoT.c
+17
-3
openair1/PHY/LTE_TRANSPORT/proto_NB_IoT.h
openair1/PHY/LTE_TRANSPORT/proto_NB_IoT.h
+2
-1
openair1/PHY/defs_common.h
openair1/PHY/defs_common.h
+6
-1
openair1/SCHED/phy_procedures_lte_eNb.c
openair1/SCHED/phy_procedures_lte_eNb.c
+132
-59
No files found.
openair1/PHY/CODING/ccoding_byte_NB_IoT.c
View file @
d38501c9
/***********************************************************************
/*
* Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The OpenAirInterface Software Alliance licenses this file to You under
* the OAI Public License, Version 1.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.openairinterface.org/?page_id=698
*
* 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.
*-------------------------------------------------------------------------------
* For more information about the OpenAirInterface (OAI) Software Alliance:
* contact@openairinterface.org
*/
**********************************************************************/
/*! \file PHY/LTE_CODING/ccoding_byte_NB_IoT.c
* \Fucntions for CRC attachment and tail-biting convolutional coding for NPBCH channel, TS 36-212, V13.4.0 2017-02
* \author M. KANJ
...
...
openair1/PHY/LTE_TRANSPORT/defs_NB_IoT.h
View file @
d38501c9
...
...
@@ -188,19 +188,18 @@ typedef struct {
uint8_t
d
[
96
+
(
3
*
(
24
+
56
))];
// new parameter
/// Sub-block interleaver outputs
uint8_t
w
[
3
*
3
*
(
56
+
24
)];
// new parameter
/////////////////////////////////
uint16_t
si_rnti_x
;
/// Concatenated "e"-sequences (for definition see 36-212 V8.6 2009-03, p.17-18)
uint8_t
e_x
[
236
];
uint8_t
e_x
[
472
];
/// data after scrambling
uint8_t
s_e_x
[
236
];
uint8_t
s_e_x
[
472
];
//length of the table e
uint16_t
length_e_x
;
// new parameter
/// Tail-biting convolutional coding outputs
uint8_t
d_x
[
96
+
(
3
*
(
24
+
120
))];
// new parameter
uint8_t
d_x
[
96
+
(
3
*
(
24
+
256
))];
// new parameter
/// Sub-block interleaver outputs
uint8_t
w_x
[
3
*
3
*
(
120
+
24
)];
// new parameter
uint8_t
w_x
[
3
*
3
*
(
256
+
24
)];
// new parameter
////////////////////////////////
/// Status Flag indicating for this DLSCH (idle,active,disabled)
...
...
openair1/PHY/LTE_TRANSPORT/dlsch_coding_NB_IoT.c
View file @
d38501c9
...
...
@@ -163,14 +163,19 @@ int dlsch_encoding_rar_NB_IoT(unsigned char *a,
bzero
(
npbch_a
,
7
);
bzero
(
npbch_a_crc
,
10
);
uint8_t
npbch_a_x
[
15
];
/*
uint8_t npbch_a_x[15];
uint8_t npbch_a_crc_x[18];
bzero(npbch_a_x,15);
bzero
(
npbch_a_crc_x
,
18
);
bzero(npbch_a_crc_x,18);*/
uint8_t
npbch_a_x
[
32
];
uint8_t
npbch_a_crc_x
[
35
];
bzero
(
npbch_a_x
,
32
);
bzero
(
npbch_a_crc_x
,
35
);
dlsch
->
length_e
=
G
;
// G*Nsf (number_of_subframes) = total number of bits to transmit G=236
dlsch
->
length_e
=
G
*
Nsf
;
// G*Nsf (number_of_subframes) = total number of bits to transmit G=236
...
...
@@ -182,8 +187,8 @@ if(option ==1)
npbch_a
[
i
]
=
a
[
i
];
}
}
else
{
A
=
120
;
for
(
int
i
=
0
;
i
<
15
;
i
++
)
A
=
256
;
for
(
int
i
=
0
;
i
<
32
;
i
++
)
{
npbch_a_x
[
i
]
=
a
[
i
];
}
...
...
@@ -218,14 +223,14 @@ if(option ==1)
crc
=
crc24a_NB_IoT
(
npbch_a_x
,
A
)
>>
8
;
for
(
int
j
=
0
;
j
<
15
;
j
++
)
for
(
int
j
=
0
;
j
<
32
;
j
++
)
{
npbch_a_crc_x
[
j
]
=
npbch_a_x
[
j
];
}
npbch_a_crc_x
[
15
]
=
((
uint8_t
*
)
&
crc
)[
2
];
npbch_a_crc_x
[
16
]
=
((
uint8_t
*
)
&
crc
)[
1
];
npbch_a_crc_x
[
17
]
=
((
uint8_t
*
)
&
crc
)[
0
];
npbch_a_crc_x
[
32
]
=
((
uint8_t
*
)
&
crc
)[
2
];
npbch_a_crc_x
[
33
]
=
((
uint8_t
*
)
&
crc
)[
1
];
npbch_a_crc_x
[
34
]
=
((
uint8_t
*
)
&
crc
)[
0
];
dlsch
->
B
=
numbits
;
// The length of table b in bits
//memcpy(dlsch->b,a,numbits/8); // comment if option 2
...
...
openair1/PHY/LTE_TRANSPORT/dlsch_modulation_NB_IoT.c
View file @
d38501c9
...
...
@@ -207,7 +207,8 @@ int dlsch_modulation_rar_NB_IoT(int32_t **txdataF,
int
G
,
// number of bits per subframe
unsigned
int
npdsch_data_subframe
,
// subframe index of the data table of npdsch channel (G*Nsf) , values are between 0..Nsf
unsigned
int
subframe
,
unsigned
short
NB_IoT_RB_ID
)
unsigned
short
NB_IoT_RB_ID
,
uint8_t
option
)
{
//uint8_t harq_pid = dlsch0->current_harq_pid;
//NB_IoT_DL_eNB_HARQ_t *dlsch0_harq = dlsch0->harq_processes[harq_pid];
...
...
@@ -245,8 +246,20 @@ int dlsch_modulation_rar_NB_IoT(int32_t **txdataF,
}
symbol_offset
=
(
14
*
subframe
*
frame_parms
->
ofdm_symbol_size
)
+
frame_parms
->
ofdm_symbol_size
*
l
+
NB_IoT_start
;
// symbol_offset = 512 * L + NB_IOT_RB start
allocate_REs_in_RB_NB_IoT
(
frame_parms
,
if
(
option
==
2
)
{
allocate_REs_in_RB_NB_IoT
(
frame_parms
,
txdataF
,
&
jj
,
symbol_offset
,
&
dlsch0
->
s_e
[
236
],
pilots
,
amp
,
id_offset
,
pilot_shift
,
&
re_allocated
);
}
else
{
allocate_REs_in_RB_NB_IoT
(
frame_parms
,
txdataF
,
&
jj
,
symbol_offset
,
...
...
@@ -256,6 +269,7 @@ int dlsch_modulation_rar_NB_IoT(int32_t **txdataF,
id_offset
,
pilot_shift
,
&
re_allocated
);
}
}
// VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_ENB_DLSCH_MODULATION, VCD_FUNCTION_OUT);
...
...
openair1/PHY/LTE_TRANSPORT/proto_NB_IoT.h
View file @
d38501c9
...
...
@@ -303,7 +303,8 @@ int dlsch_modulation_rar_NB_IoT(int32_t **txdataF,
int
G
,
// number of bits per subframe
unsigned
int
npdsch_data_subframe
,
// subframe index of the data table of npdsch channel (G*Nsf) , values are between 0..Nsf
unsigned
int
subframe
,
unsigned
short
NB_IoT_RB_ID
);
unsigned
short
NB_IoT_RB_ID
,
uint8_t
option
);
int32_t
dlsch_encoding_NB_IoT
(
unsigned
char
*
a
,
NB_IoT_DL_eNB_SIB_t
*
dlsch
,
// NB_IoT_eNB_NDLSCH_t
...
...
openair1/PHY/defs_common.h
View file @
d38501c9
...
...
@@ -79,6 +79,9 @@ typedef struct {
uint32_t
frame_dscr_msg3
;
uint32_t
subframe_dscr_msg3
;
uint32_t
frame_msg5
;
uint32_t
subframe_msg5
;
uint8_t
rar_to_transmit
;
uint8_t
subframe_SP
;
uint8_t
subframe_SP2
;
...
...
@@ -108,11 +111,13 @@ typedef struct {
uint32_t
subframe_msg4
;
uint8_t
guard
;
uint8_t
counter_msg5
;
uint8_t
remaining_dci
;
uint8_t
remaining_rar
;
uint8_t
flag_msg5
;
}
eNB_rxtx_proc_t
;
/// Context data structure for eNB subframe processing
...
...
openair1/SCHED/phy_procedures_lte_eNb.c
View file @
d38501c9
...
...
@@ -537,6 +537,12 @@ void common_signal_procedures (PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc) {
}*/
/////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////// Decoding ACK ////////////////////////////////
if
(
subframe
==
proc
->
subframe_msg5
&&
frame
==
proc
->
frame_msg5
&&
proc
->
flag_msg5
==
1
&&
proc
->
counter_msg5
>
0
)
{
int
x
=
0
;
}
////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////// RX NPUSH //////////////////////////////////////
if
(
subframe
==
proc
->
subframe_real
&&
proc
->
flag_msg3
==
1
&&
frame
==
proc
->
frame_msg3
&&
proc
->
counter_msg3
>
0
)
///&& frame == ????
...
...
@@ -583,48 +589,50 @@ void common_signal_procedures (PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc) {
l
/
(
fp
->
symbols_per_tti
/
2
),
fp
);
/// Channel Estimation
ul_chest_tmp_NB_IoT
(
pusch_vars
->
rxdataF_ext
[
0
],
pusch_vars
->
drs_ch_estimates
[
0
],
l
%
(
fp
->
symbols_per_tti
/
2
),
//symbol within slot
l
/
(
fp
->
symbols_per_tti
/
2
),
proc
->
counter_msg3
,
fp
);
ul_chest_tmp_NB_IoT
(
pusch_vars
->
rxdataF_ext
[
0
],
pusch_vars
->
drs_ch_estimates
[
0
],
l
%
(
fp
->
symbols_per_tti
/
2
),
//symbol within slot
l
/
(
fp
->
symbols_per_tti
/
2
),
proc
->
counter_msg3
,
fp
);
}
for
(
l
=
0
;
l
<
fp
->
symbols_per_tti
;
l
++
)
{
/// Equalization
ul_chequal_tmp_NB_IoT
(
pusch_vars
->
rxdataF_ext
[
0
],
pusch_vars
->
rxdataF_comp
[
0
],
pusch_vars
->
drs_ch_estimates
[
0
],
l
%
(
fp
->
symbols_per_tti
/
2
),
//symbol within slot
l
/
(
fp
->
symbols_per_tti
/
2
),
fp
);
/// Equalization
ul_chequal_tmp_NB_IoT
(
pusch_vars
->
rxdataF_ext
[
0
],
pusch_vars
->
rxdataF_comp
[
0
],
pusch_vars
->
drs_ch_estimates
[
0
],
l
%
(
fp
->
symbols_per_tti
/
2
),
//symbol within slot
l
/
(
fp
->
symbols_per_tti
/
2
),
fp
);
}
for
(
l
=
0
;
l
<
fp
->
symbols_per_tti
;
l
++
)
{
/// In case of 1 subcarrier: BPSK and QPSK should be rotated by pi/2 and pi/4, respectively
rotate_single_carrier_NB_IoT
(
eNB
,
fp
,
pusch_vars
->
rxdataF_comp
[
0
],
0
,
// UE ID
l
,
proc
->
counter_msg3
,
2
);
// Qm
/// In case of 1 subcarrier: BPSK and QPSK should be rotated by pi/2 and pi/4, respectively
rotate_single_carrier_NB_IoT
(
eNB
,
fp
,
pusch_vars
->
rxdataF_comp
[
0
],
0
,
// UE ID
l
,
proc
->
counter_msg3
,
2
);
// Qm
}
proc
->
subframe_delay
++
;
if
(
proc
->
subframe_delay
==
10
)
///&& frame == ????
proc
->
subframe_delay
++
;
if
(
proc
->
subframe_delay
==
10
)
///&& frame == ????
{
proc
->
subframe_delay
=
0
;
proc
->
subframe_delay
=
0
;
}
proc
->
subframe_delay
++
;
proc
->
subframe_delay
++
;
///////// IDFT inverse precoding is done over the whole subframe of 14 - 2 (pilots) symbols
...
...
@@ -633,7 +641,7 @@ void common_signal_procedures (PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc) {
// 12); // IDFT size
llrp
=
(
int16_t
*
)
&
pusch_vars
->
llr
[
0
+
(
8
-
proc
->
counter_msg3
)
*
24
];
int
ii
=
0
;
int
ii
=
0
;
for
(
l
=
0
;
l
<
fp
->
symbols_per_tti
;
l
++
)
{
if
(
l
==
pilot_pos1
||
l
==
pilot_pos2
)
// skip pilots
...
...
@@ -648,7 +656,7 @@ void common_signal_procedures (PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc) {
l
,
0
,
// UE ID
&
llrp
[
ii
*
2
]);
//// !!! Pensez à créer un buffer de longueur 8 subframes
ii
++
;
ii
++
;
}
/*printf("\n");
...
...
@@ -1009,23 +1017,17 @@ void common_signal_procedures (PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc) {
//////////////////////////////////////////////////////////////////////////////////////////////////
if
(
proc
->
flag_msg4
==
1
&&
proc
->
counter_msg4
>
0
)
{
if
(
subframe
==
9
)
{
proc
->
subframe_msg4
=
1
;
proc
->
frame_msg4
=
frame
+
1
;
}
else
{
if
(
frame
==
proc
->
frame_msg4
&&
subframe
==
proc
->
subframe_msg4
)
{
NB_IoT_DL_eNB_RAR_t
*
rar
=
&
eNB
->
ndlsch_rar
.
content_rar
;
uint8_t
tab_rar
[
15
];
//uint8_t tab_rar[7];
//uint8_t tab_rar[15];
//uint8_t tab_rar[18];
uint8_t
tab_rar
[
7
];
uint8_t
*
nas_id
=
&
eNB
->
msg3_pdu
[
0
];
//uint8_t *NAS_tab = &eNB->tab_nas;
// avoid subframe 9 and subframe 0 of next frame
/*
tab_rar[0]=63;
/*
tab_rar[0]=63;
tab_rar[1]=60;
tab_rar[2]=0;
tab_rar[3]=nas_id[0]; // NAS part 1
...
...
@@ -1041,7 +1043,27 @@ if(proc->flag_msg4 == 1 && proc->counter_msg4 > 0)
tab_rar[13]=145;
tab_rar[14]=8;*/
/*
tab_rar[0]=63;
tab_rar[1]=60;
tab_rar[2]=0;
tab_rar[3]=nas_id[0]; // NAS part 1
tab_rar[4]=nas_id[1]; // NAS part 2
tab_rar[5]=nas_id[2]; // NAS part 3
tab_rar[6]=nas_id[3]; // NAS part 4
tab_rar[7]=nas_id[4]; // NAS part 5
tab_rar[8]=nas_id[5]; // NAS part 6
tab_rar[9]=48;
tab_rar[10]=19;
tab_rar[11]=176;
tab_rar[12]=216;
tab_rar[13]=134;
tab_rar[14]=114;
tab_rar[15]=68;
tab_rar[16]=32;
tab_rar[17]=32;
*/
tab_rar
[
0
]
=
28
;
tab_rar
[
1
]
=
nas_id
[
0
];
// NAS part 1
...
...
@@ -1052,14 +1074,11 @@ if(proc->flag_msg4 == 1 && proc->counter_msg4 > 0)
tab_rar
[
6
]
=
nas_id
[
5
];
// NAS part 5
printf
(
"ms4pdu[0] = %d
\n
"
,
nas_id
[
0
]);
printf
(
"ms4pdu[1] = %d
\n
"
,
nas_id
[
1
]);
printf
(
"ms4pdu[2] = %d
\n
"
,
nas_id
[
2
]);
printf
(
"ms4pdu[3] = %d
\n
"
,
nas_id
[
3
]);
printf
(
"ms4pdu[2] = %d
\n
"
,
nas_id
[
4
]);
printf
(
"ms4pdu[0] = %d
\n
"
,
nas_id
[
0
]);
printf
(
"ms4pdu[1] = %d
\n
"
,
nas_id
[
1
]);
printf
(
"ms4pdu[2] = %d
\n
"
,
nas_id
[
2
]);
printf
(
"ms4pdu[3] = %d
\n
"
,
nas_id
[
3
]);
printf
(
"ms4pdu[2] = %d
\n
"
,
nas_id
[
4
]);
if
(
proc
->
flag_scrambling
==
0
)
...
...
@@ -1067,7 +1086,7 @@ if(proc->flag_msg4 == 1 && proc->counter_msg4 > 0)
dlsch_encoding_rar_NB_IoT
(
tab_rar
,
rar
,
8
,
///// number_of_subframes_required
2
,
///// number_of_subframes_required
236
,
1
);
//////////// G*2 // option =2 for msg4
...
...
@@ -1081,7 +1100,9 @@ if(proc->flag_msg4 == 1 && proc->counter_msg4 > 0)
}
proc
->
flag_scrambling
=
1
;
printf
(
"
\n
RAR sentttttt frame %d, subframe %d"
,
frame
,
subframe
);
dlsch_modulation_rar_NB_IoT
(
txdataF
,
/* if(proc->counter_msg4 == 3 || proc->counter_msg4 ==1)
{
dlsch_modulation_rar_NB_IoT(txdataF,
AMP,
fp,
3, // control region size for LTE , values between 0..3, (0 for stand-alone / 1, 2 or 3 for in-band)
...
...
@@ -1089,13 +1110,47 @@ if(proc->flag_msg4 == 1 && proc->counter_msg4 > 0)
236, // number of bits per subframe
frame, // unrequired
subframe,
22
);
22,
2);
} else {*/
dlsch_modulation_rar_NB_IoT
(
txdataF
,
AMP
,
fp
,
3
,
// control region size for LTE , values between 0..3, (0 for stand-alone / 1, 2 or 3 for in-band)
rar
,
236
,
// number of bits per subframe
frame
,
// unrequired
subframe
,
22
,
0
);
// }
proc
->
counter_msg4
--
;
proc
->
subframe_msg4
=
subframe
+
1
;
if
(
proc
->
counter_msg4
==
0
)
{
proc
->
flag_msg5
=
1
;
proc
->
counter_msg5
=
2
;
proc
->
subframe_msg5
=
(
subframe
+
12
+
1
+
4
)
%
10
;
if
(
((
subframe
+
12
+
1
+
4
)
%
10
)
<
20
)
{
proc
->
frame_msg5
=
frame
+
2
;
}
else
{
proc
->
frame_msg5
=
frame
+
3
;
}
}
}
}
if
(
subframe
==
9
)
{
proc
->
subframe_msg4
=
1
;
proc
->
frame_msg4
=
frame
+
1
;
}
}
...
...
@@ -1154,15 +1209,29 @@ if(proc->flag_DCI_msg4 ==1 && proc->counter_DCI_msg4>0)
proc
->
flag_scrambling
=
0
;
/// for msg4 PDU
int32_t
numbits
=
23
;
uint8_t
RCC
;
uint16_t
npdcch_D
=
16
+
23
;
uint8_t
RCC
;
uint16_t
npdcch_D
=
16
+
23
;
uint16_t
npdcch_E
=
236
;
uint8_t
agr_level
=
2
;
// uint8_t dci_number=1;
// uint8_t dci_number=1;
uint8_t
tab_a
[
3
];
tab_a
[
0
]
=
128
;
/*
tab_a[0]= 128;
tab_a[1]= 66;
tab_a[2]= 4;
*/
// TBS =120
tab_a
[
0
]
=
129
;
tab_a
[
1
]
=
130
;
tab_a
[
2
]
=
2
;
/* // TBS =144
tab_a[0]= 128;
tab_a[1]= 130;
tab_a[2]= 4;
*/
// int RB_ID_IoT=22 ;
bzero
(
npdcch_struct_x
->
npdcch_e
[
0
],
npdcch_E
);
...
...
@@ -1214,7 +1283,7 @@ if(proc->flag_DCI_msg4 ==1 && proc->counter_DCI_msg4>0)
proc
->
subframe_real
=-
1
;
proc
->
subframe_delay
=-
1
;
proc
->
flag_msg5
=
0
;
proc
->
flag_msg4
=
0
;
proc
->
subframe_SP2
=
0
;
proc
->
subframe_SP
=
0
;
...
...
@@ -1227,7 +1296,9 @@ if(proc->flag_DCI_msg4 ==1 && proc->counter_DCI_msg4>0)
proc
->
guard
=
5
;
proc
->
subframe_DCI_msg4
=
0
;
proc
->
subframe_msg4
=
0
;
proc
->
counter_msg5
=
0
;
proc
->
frame_msg5
=
0
;
proc
->
subframe_msg5
=
0
;
//printf("detection ok in TX !!!!!!!!!!!!!!!!!");
//RA_template[0].generate_rar = 0;
//RA_template[0].RA_active = FALSE;
...
...
@@ -1588,7 +1659,8 @@ if(subframe !=5 && subframe !=0)
236
,
// number of bits per subframe
frame
,
// unrequired
subframe
,
22
);
22
,
0
);
proc
->
remaining_rar
--
;
proc
->
next_subframe_tx
=
subframe
+
2
;
...
...
@@ -1637,7 +1709,8 @@ if(subframe !=5 && subframe !=0)
236
,
// number of bits per subframe
frame
,
// unrequired
subframe
,
22
);
22
,
0
);
proc
->
remaining_rar
--
;
proc
->
next_subframe_tx
=
subframe
+
1
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment