Adding initial code jy.oak@samsung.com
[ric-app/kpimon.git] / src / E2SM / e2sm.cc
diff --git a/src/E2SM/e2sm.cc b/src/E2SM/e2sm.cc
new file mode 100755 (executable)
index 0000000..c6a3159
--- /dev/null
@@ -0,0 +1,632 @@
+/*\r
+  ==================================================================================\r
+\r
+  Copyright (c) 2018-2019 AT&T Intellectual Property.\r
+  \r
+  Licensed under the Apache License, Version 2.0 (the "License");\r
+  you may not use this file except in compliance with the License.\r
+  You may obtain a copy of the License at\r
+  \r
+  http://www.apache.org/licenses/LICENSE-2.0\r
+  \r
+  Unless required by applicable law or agreed to in writing, softwares\r
+  distributed under the License is distributed on an "AS IS" BASIS,\r
+  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r
+  See the License for the specific language governing permissions and\r
+  limitations under the License.\r
+  ==================================================================================\r
+*/\r
+\r
+/* Classes to handle E2 service model based on e2sm-gNB-X2-release-1-v040.asn */\r
+\r
+#include "e2sm.hpp"\r
+\r
+\r
+\r
+  //initialize\r
+  e2sm_event_trigger::e2sm_event_trigger(void){\r
+\r
+    memset(&gNodeB_ID, 0, sizeof(GlobalGNB_ID_t));\r
+\r
+    event_trigger = 0;\r
+    event_trigger = ( E2SM_gNB_X2_eventTriggerDefinition_t *)calloc(1, sizeof( E2SM_gNB_X2_eventTriggerDefinition_t));\r
+    assert(event_trigger != 0);\r
+    \r
+    // allocate space for gNodeB id  (used for encoding)\r
+    gNodeB_ID.gNB_ID.choice.gNB_ID.buf = 0;\r
+    gNodeB_ID.gNB_ID.choice.gNB_ID.buf = (uint8_t *)calloc(4, sizeof(uint8_t));\r
+    assert(gNodeB_ID.gNB_ID.choice.gNB_ID.buf != 0);\r
+    \r
+    // allocate space for plmn identity  (used for encoding)\r
+    gNodeB_ID.pLMN_Identity.buf = 0;\r
+    gNodeB_ID.pLMN_Identity.buf = (uint8_t *) calloc(4, sizeof(uint8_t));\r
+    assert(gNodeB_ID.pLMN_Identity.buf != 0);\r
+\r
+    ie_list = 0;\r
+    ie_list = ( struct InterfaceProtocolIE_Item *) calloc(INITIAL_LIST_SIZE, sizeof( struct InterfaceProtocolIE_Item));\r
+    assert(ie_list != 0);\r
+    ie_list_size = INITIAL_LIST_SIZE;\r
+\r
+    condition_list = 0;\r
+    condition_list = (E2SM_gNB_X2_eventTriggerDefinition::E2SM_gNB_X2_eventTriggerDefinition__interfaceProtocolIE_List *) calloc(1, sizeof(E2SM_gNB_X2_eventTriggerDefinition::E2SM_gNB_X2_eventTriggerDefinition__interfaceProtocolIE_List ));\r
+    assert(condition_list != 0);\r
+\r
\r
+    \r
+  };\r
+  \r
+e2sm_event_trigger::~e2sm_event_trigger(void){\r
+\r
+  mdclog_write(MDCLOG_INFO, "Freeing event trigger object memory");\r
+  for(int i = 0; i < condition_list->list.size; i++){\r
+    condition_list->list.array[i] = 0;\r
+  }\r
+\r
+  if (condition_list->list.size > 0){\r
+    free(condition_list->list.array);\r
+    condition_list->list.array = 0;\r
+    condition_list->list.size = 0;\r
+    condition_list->list.count = 0;\r
+  }\r
+\r
+  free(condition_list);\r
+  condition_list = 0;\r
+  \r
+  free(gNodeB_ID.gNB_ID.choice.gNB_ID.buf);\r
+  gNodeB_ID.gNB_ID.choice.gNB_ID.buf = 0;\r
+  \r
+  free(gNodeB_ID.pLMN_Identity.buf);\r
+  gNodeB_ID.pLMN_Identity.buf = 0;\r
+  \r
+  free(ie_list);\r
+  ie_list = 0;\r
+  \r
+  event_trigger->interface_ID.choice.global_gNB_ID = 0;\r
+  event_trigger->interfaceProtocolIE_List = 0;\r
+  \r
+  ASN_STRUCT_FREE(asn_DEF_E2SM_gNB_X2_eventTriggerDefinition, event_trigger);\r
+  mdclog_write(MDCLOG_INFO, "Freed event trigger object memory");\r
+\r
\r
+};\r
+\r
+bool e2sm_event_trigger::encode_event_trigger(unsigned char *buf, size_t *size, e2sm_event_trigger_helper &helper){\r
+  \r
+  bool res;\r
+  res = set_fields(event_trigger, helper);\r
+  if (!res){\r
+    return false;\r
+  }\r
+  \r
+  int ret_constr = asn_check_constraints(&asn_DEF_E2SM_gNB_X2_eventTriggerDefinition, event_trigger, errbuf, &errbuf_len);\r
+  if(ret_constr){\r
+    error_string.assign(&errbuf[0], errbuf_len);\r
+    return false;\r
+  }\r
+\r
+  //xer_fprint(stdout, &asn_DEF_E2SM_gNB_X2_eventTriggerDefinition, event_trigger);\r
+  \r
+  asn_enc_rval_t retval = asn_encode_to_buffer(0, ATS_ALIGNED_BASIC_PER, &asn_DEF_E2SM_gNB_X2_eventTriggerDefinition, event_trigger, buf, *size);\r
+  \r
+  if(retval.encoded == -1){\r
+    error_string.assign(strerror(errno));\r
+    return false;\r
+  }\r
+  else if (retval.encoded > *size){\r
+    std::stringstream ss;\r
+    ss  <<"Error encoding event trigger definition. Reason =  encoded pdu size " << retval.encoded << " exceeds buffer size " << *size << std::endl;\r
+    error_string = ss.str();\r
+    return false;\r
+  }\r
+  else{\r
+    *size = retval.encoded;\r
+  }\r
+  \r
+  return true;\r
+}\r
+\r
+\r
+bool e2sm_event_trigger::set_fields(E2SM_gNB_X2_eventTriggerDefinition_t * ref_event_trigger, e2sm_event_trigger_helper & helper){\r
+  if(ref_event_trigger == 0){\r
+    error_string = "Invalid reference for Event Trigger Definition set fields";\r
+    return false;\r
+  }\r
+      \r
+  // set the message type\r
+  ref_event_trigger->interfaceMessageType.procedureCode = helper.procedure_code;\r
+  ref_event_trigger->interfaceMessageType.typeOfMessage = helper.message_type;\r
+  \r
+  ref_event_trigger->interfaceDirection = helper.interface_direction; \r
+  ref_event_trigger->interface_ID.present = Interface_ID_PR_global_gNB_ID;\r
+  \r
+  ref_event_trigger->interface_ID.choice.global_gNB_ID = &gNodeB_ID;\r
+\r
+  // to do : need to put correct code here for upding plmn id and gNodeB\r
+  // for now just place holders :\r
+  //================================================================\r
+  memcpy(gNodeB_ID.pLMN_Identity.buf, helper.plmn_id.c_str(), 3);\r
+  gNodeB_ID.pLMN_Identity.size = 3;\r
+  \r
+  memcpy(gNodeB_ID.gNB_ID.choice.gNB_ID.buf, helper.egNB_id.c_str(), 3);\r
+  gNodeB_ID.gNB_ID.choice.gNB_ID.size = 3;\r
+  \r
+  // we only do global gNodeB id for now, not eNodeB\r
+  gNodeB_ID.gNB_ID.present = GNB_ID_PR_gNB_ID;\r
+  //================================================================\r
+  \r
+  \r
+  // Add in any requested IE items\r
+  std::vector<Item> * ref_ie_array = helper.get_list();\r
+\r
+  if (ref_ie_array->size() == 0){\r
+    event_trigger->interfaceProtocolIE_List = 0;\r
+    \r
+  }\r
+  else{\r
+    event_trigger->interfaceProtocolIE_List = condition_list;\r
+    \r
+    //reallocate ? \r
+    if(ref_ie_array->size() > ie_list_size){\r
+      struct InterfaceProtocolIE_Item *new_ref = 0;\r
+      ie_list_size = 2 * ref_ie_array->size();\r
+      new_ref = (struct InterfaceProtocolIE_Item *)realloc(ie_list, ie_list_size);\r
+      assert(new_ref != 0);\r
+      ie_list = new_ref;\r
+    }\r
+    \r
+    // reset the count so that adds start from the beginning\r
+    ref_event_trigger->interfaceProtocolIE_List->list.count = 0;\r
+    \r
+    for(unsigned int i = 0; i < ref_ie_array->size(); i++){\r
+\r
+      ie_list[i].interfaceProtocolIE_ID = (*ref_ie_array)[i].interface_id;\r
+      ie_list[i].interfaceProtocolIE_Test = (*ref_ie_array)[i].test;\r
+      \r
+      //switch(ie_list[i].interfaceProtocolIE_Value.present){\r
+      switch((*ref_ie_array)[i].val_type){\r
+       \r
+      case (InterfaceProtocolIE_Value_PR_valueInt):\r
+       ie_list[i].interfaceProtocolIE_Value.present = InterfaceProtocolIE_Value_PR_valueInt;\r
+       ie_list[i].interfaceProtocolIE_Value.choice.valueInt = (*ref_ie_array)[i].value_n;\r
+       break;\r
+       \r
+      case (InterfaceProtocolIE_Value_PR_valueEnum):\r
+       ie_list[i].interfaceProtocolIE_Value.present = InterfaceProtocolIE_Value_PR_valueEnum;\r
+       ie_list[i].interfaceProtocolIE_Value.choice.valueEnum = (*ref_ie_array)[i].value_n;\r
+       break;\r
+       \r
+      case (InterfaceProtocolIE_Value_PR_valueBool):\r
+       ie_list[i].interfaceProtocolIE_Value.present = InterfaceProtocolIE_Value_PR_valueBool;\r
+       ie_list[i].interfaceProtocolIE_Value.choice.valueBool = (*ref_ie_array)[i].value_n;\r
+       break;\r
+       \r
+      case (InterfaceProtocolIE_Value_PR_valueBitS):\r
+       ie_list[i].interfaceProtocolIE_Value.present = InterfaceProtocolIE_Value_PR_valueBitS;\r
+       ie_list[i].interfaceProtocolIE_Value.choice.valueBitS.buf = (uint8_t *)(*ref_ie_array)[i].value_s.c_str();\r
+       ie_list[i].interfaceProtocolIE_Value.choice.valueBitS.size = (*ref_ie_array)[i].value_s.length();\r
+       break;\r
+       \r
+      default:\r
+       {\r
+         std::stringstream ss;\r
+         ss <<"Error ! " << __FILE__ << "," << __LINE__ << " illegal enum " << (*ref_ie_array)[i].val_type << " for interface Protocol IE value" << std::endl;\r
+         std::string error_string = ss.str();\r
+         return false;\r
+       }\r
+      }\r
+      \r
+      ASN_SEQUENCE_ADD(ref_event_trigger->interfaceProtocolIE_List, &ie_list[i]);\r
+    }\r
+  }\r
+\r
+  return true;\r
+};\r
+  \r
+\r
+bool e2sm_event_trigger::get_fields(E2SM_gNB_X2_eventTriggerDefinition_t * ref_event_trigger, e2sm_event_trigger_helper & helper){\r
+\r
+  if (ref_event_trigger == 0){\r
+    error_string = "Invalid reference for Event Trigger definition get fields";\r
+    return false;\r
+  }\r
+  \r
+  helper.procedure_code = ref_event_trigger->interfaceMessageType.procedureCode;\r
+  helper.message_type   = ref_event_trigger->interfaceMessageType.typeOfMessage;\r
+  helper.interface_direction = ref_event_trigger->interfaceDirection;\r
+  \r
+  helper.plmn_id.assign((const char *)ref_event_trigger->interface_ID.choice.global_gNB_ID->pLMN_Identity.buf, ref_event_trigger->interface_ID.choice.global_gNB_ID->pLMN_Identity.size);\r
+  helper.egNB_id.assign((const char *)ref_event_trigger->interface_ID.choice.global_gNB_ID->gNB_ID.choice.gNB_ID.buf, ref_event_trigger->interface_ID.choice.global_gNB_ID->gNB_ID.choice.gNB_ID.size);\r
+\r
+  return true;\r
+};\r
+    \r
+\r
+  \r
+   \r
+// initialize\r
+e2sm_indication::e2sm_indication(void) {\r
+  \r
+  memset(&gNodeB_ID, 0, sizeof(GlobalGNB_ID_t));\r
+    \r
+  // allocate space for gNodeB id  (used for encoding)\r
+  gNodeB_ID.gNB_ID.choice.gNB_ID.buf = (uint8_t *)calloc(4, sizeof(uint8_t));\r
+  assert(gNodeB_ID.gNB_ID.choice.gNB_ID.buf != 0);\r
+    \r
+  // allocate space for plmn identity  (used for encoding)\r
+  gNodeB_ID.pLMN_Identity.buf = (uint8_t *) calloc(4, sizeof(uint8_t));\r
+  assert(gNodeB_ID.pLMN_Identity.buf != 0);\r
+\r
+  header = 0;\r
+  header = (E2SM_gNB_X2_indicationHeader_t *)calloc(1, sizeof(E2SM_gNB_X2_indicationHeader_t));\r
+  assert(header != 0);\r
+\r
+  message = 0;\r
+  message = (E2SM_gNB_X2_indicationMessage_t *)calloc(1, sizeof(E2SM_gNB_X2_indicationMessage_t));\r
+  assert(message != 0);\r
+}\r
+  \r
+e2sm_indication::~e2sm_indication(void){\r
+  mdclog_write(MDCLOG_INFO, "Freeing E2SM Indication  object memory");\r
+\r
+  free(gNodeB_ID.gNB_ID.choice.gNB_ID.buf);\r
+  free(gNodeB_ID.pLMN_Identity.buf);\r
+  \r
+  header->interface_ID.choice.global_gNB_ID = 0;\r
+\r
+  ASN_STRUCT_FREE(asn_DEF_E2SM_gNB_X2_indicationHeader, header);\r
+\r
+  message->interfaceMessage.buf = 0;\r
+  message->interfaceMessage.size = 0;\r
+\r
+  ASN_STRUCT_FREE(asn_DEF_E2SM_gNB_X2_indicationMessage, message);\r
+  mdclog_write(MDCLOG_INFO, "Freed E2SM Indication  object memory");\r
+    \r
+}\r
+  \r
+  \r
+\r
+bool e2sm_indication::encode_indication_header(unsigned char *buf, size_t *size, e2sm_header_helper &helper){\r
+    \r
+  bool res;\r
+  res = set_header_fields(header, helper);\r
+  if (!res){\r
+    return false;\r
+  }\r
+\r
+  int ret_constr = asn_check_constraints(&asn_DEF_E2SM_gNB_X2_indicationHeader, header, errbuf, &errbuf_len);\r
+  if(ret_constr){\r
+    error_string.assign(&errbuf[0], errbuf_len);\r
+    error_string = "E2SM Indication Header Constraint failed : " + error_string;\r
+\r
+    return false;\r
+  }\r
+\r
+  asn_enc_rval_t retval = asn_encode_to_buffer(0, ATS_ALIGNED_BASIC_PER, &asn_DEF_E2SM_gNB_X2_indicationHeader, header, buf, *size);\r
+\r
+  if(retval.encoded == -1){\r
+    error_string.assign(strerror(errno));\r
+    error_string = "Error encoding E2SM Indication Header. Reason = " + error_string;\r
+    return false;\r
+  }\r
+  else if (retval.encoded > *size){\r
+    std::stringstream ss;\r
+    ss  <<"Error encoding E2SM Indication Header . Reason =  encoded pdu size " << retval.encoded << " exceeds buffer size " << *size << std::endl;\r
+    error_string = ss.str();\r
+    return false;\r
+  }\r
+  else{\r
+    *size = retval.encoded;\r
+  }\r
+    \r
+  return true;\r
+}\r
+\r
+\r
+bool e2sm_indication::encode_indication_message(unsigned char *buf, size_t *size, e2sm_message_helper &helper){\r
+\r
+  set_message_fields(message, helper); \r
+\r
+  int ret_constr = asn_check_constraints(&asn_DEF_E2SM_gNB_X2_indicationMessage, message, errbuf, &errbuf_len);\r
+  if(ret_constr){\r
+    error_string.assign(&errbuf[0], errbuf_len);\r
+    error_string = "E2SM Indication Message Constraint failed : " + error_string;\r
+    return false;\r
+  }\r
+\r
+  asn_enc_rval_t retval = asn_encode_to_buffer(0, ATS_ALIGNED_BASIC_PER, &asn_DEF_E2SM_gNB_X2_indicationMessage, message, buf, *size);\r
+  if(retval.encoded == -1){\r
+    error_string.assign(strerror(errno));\r
+    error_string = "Error encoding E2SM Indication Header. Reason = " + error_string;\r
+    return false;\r
+  }\r
+  else if (retval.encoded > *size){\r
+    std::stringstream ss;\r
+    ss  <<"Error encoding E2SM Indication Message . Reason =  encoded pdu size " << retval.encoded << " exceeds buffer size " << *size << std::endl;\r
+    error_string = ss.str();\r
+    \r
+    return false;\r
+  }\r
+  else{\r
+    *size = retval.encoded;\r
+  }\r
+  \r
+  return true;\r
+}\r
+\r
+\r
+\r
+// Used when generating an indication header \r
+bool e2sm_indication::set_header_fields(E2SM_gNB_X2_indicationHeader_t *header,  e2sm_header_helper &helper){\r
+\r
+  if (header == 0){\r
+    error_string = "Invalid reference for E2SM Indication Header set fields";\r
+    return false;\r
+  }\r
+  \r
+  \r
+  header->interfaceDirection = helper.interface_direction;\r
+  header->interface_ID.present = Interface_ID_PR_global_gNB_ID;\r
+  header->interface_ID.choice.global_gNB_ID = &gNodeB_ID;\r
+\r
+\r
+  // to do : need to put correct code here for upding plmn id and gNodeB\r
+  // for now just place holders :\r
+  memcpy(gNodeB_ID.pLMN_Identity.buf, helper.plmn_id.c_str(), 3);\r
+  gNodeB_ID.pLMN_Identity.size = 3;\r
+  \r
+  memcpy(gNodeB_ID.gNB_ID.choice.gNB_ID.buf, helper.egNB_id.c_str(), 3);\r
+  gNodeB_ID.gNB_ID.choice.gNB_ID.size = 3;\r
+  \r
+  // we only do global gNodeB id for now, not eNodeB\r
+  gNodeB_ID.gNB_ID.present = GNB_ID_PR_gNB_ID;\r
+\r
+  return true;\r
+  \r
+};\r
+\r
+\r
+// used when decoding an indication header\r
+bool e2sm_indication::get_header_fields(E2SM_gNB_X2_indicationHeader_t *header,  e2sm_header_helper &helper){\r
+\r
+  if (header == 0){\r
+    error_string = "Invalid reference for E2SM Indication header get fields";\r
+    return false;\r
+  }\r
+  \r
+  helper.interface_direction = header->interfaceDirection;\r
+  helper.plmn_id.assign((const char *)header->interface_ID.choice.global_gNB_ID->pLMN_Identity.buf, header->interface_ID.choice.global_gNB_ID->pLMN_Identity.size);\r
+  helper.egNB_id.assign((const char *)header->interface_ID.choice.global_gNB_ID->gNB_ID.choice.gNB_ID.buf, header->interface_ID.choice.global_gNB_ID->gNB_ID.choice.gNB_ID.size);\r
+  \r
+  // to do : add code to decipher plmn and global gnodeb from ints (since that is likely the convention for packing)\r
+\r
+  return true;\r
+}\r
+\r
+\r
+\r
+// Used when generating an indication message \r
+bool   e2sm_indication::set_message_fields(E2SM_gNB_X2_indicationMessage_t *interface_message,  e2sm_message_helper &helper){\r
+\r
+  if(interface_message == 0){\r
+    error_string = "Invalid reference for E2SM Indication Message set fields";\r
+    return false;\r
+  }\r
+\r
+  // interface-message is an octet string. just point it to the buffer\r
+  interface_message->interfaceMessage.buf = &(helper.x2ap_pdu[0]);\r
+  interface_message->interfaceMessage.size = helper.x2ap_pdu_size;\r
+\r
+  return true;\r
+  \r
+};\r
+\r
+// used when decoding an indication message\r
+bool e2sm_indication::get_message_fields( E2SM_gNB_X2_indicationMessage_t *interface_message, e2sm_message_helper &helper){\r
+\r
+  \r
+  if(interface_message == 0){\r
+    error_string = "Invalid reference for E2SM Indication Message get fields";\r
+    return false;\r
+  }\r
+\r
+  // interface message is an octet string\r
+  helper.x2ap_pdu = interface_message->interfaceMessage.buf;;\r
+  helper.x2ap_pdu_size = interface_message->interfaceMessage.size;\r
+\r
+  return true;\r
+  \r
+}\r
+  \r
+\r
+   \r
+// initialize\r
+e2sm_control::e2sm_control(void) {\r
+  \r
+  memset(&gNodeB_ID, 0, sizeof(GlobalGNB_ID_t));\r
+    \r
+  // allocate space for gNodeB id  (used for encoding)\r
+  gNodeB_ID.gNB_ID.choice.gNB_ID.buf = (uint8_t *)calloc(4, sizeof(uint8_t));\r
+  assert(gNodeB_ID.gNB_ID.choice.gNB_ID.buf != 0);\r
+    \r
+  // allocate space for plmn identity  (used for encoding)\r
+  gNodeB_ID.pLMN_Identity.buf = (uint8_t *) calloc(4, sizeof(uint8_t));\r
+  assert(gNodeB_ID.pLMN_Identity.buf != 0);\r
+\r
+  header = 0;\r
+  header = (E2SM_gNB_X2_controlHeader_t *)calloc(1, sizeof(E2SM_gNB_X2_controlHeader_t));\r
+  assert(header != 0);\r
+\r
+  message = 0;\r
+  message = (E2SM_gNB_X2_controlMessage_t *)calloc(1, sizeof(E2SM_gNB_X2_controlMessage_t));\r
+  assert(message != 0);\r
+}\r
+  \r
+e2sm_control::~e2sm_control(void){\r
+  mdclog_write(MDCLOG_INFO, "Freeing E2SM Control  object memory");\r
+\r
+  free(gNodeB_ID.gNB_ID.choice.gNB_ID.buf);\r
+  free(gNodeB_ID.pLMN_Identity.buf);\r
+  header->interface_ID.choice.global_gNB_ID = 0;\r
+  ASN_STRUCT_FREE(asn_DEF_E2SM_gNB_X2_controlHeader, header);\r
+\r
+  message->interfaceMessage.buf = 0;\r
+  ASN_STRUCT_FREE(asn_DEF_E2SM_gNB_X2_controlMessage, message);\r
+\r
+  mdclog_write(MDCLOG_INFO, "Freed E2SM Control  object memory");\r
+    \r
+}\r
+  \r
+  \r
+\r
+bool e2sm_control::encode_control_header(unsigned char *buf, size_t *size, e2sm_header_helper &helper){\r
+    \r
+  bool res;\r
+  res = set_header_fields(header, helper);\r
+  if (!res){\r
+    return false;\r
+  }\r
+\r
+  int ret_constr = asn_check_constraints(&asn_DEF_E2SM_gNB_X2_controlHeader, header, errbuf, &errbuf_len);\r
+  if(ret_constr){\r
+    error_string.assign(&errbuf[0], errbuf_len);\r
+    error_string = "E2SM Control Header Constraint failed : " + error_string;\r
+\r
+    return false;\r
+  }\r
+\r
+  asn_enc_rval_t retval = asn_encode_to_buffer(0, ATS_ALIGNED_BASIC_PER, &asn_DEF_E2SM_gNB_X2_controlHeader, header, buf, *size);\r
+\r
+  if(retval.encoded == -1){\r
+    error_string.assign(strerror(errno));\r
+    error_string = "Error encoding E2SM Control Header. Reason = " + error_string;\r
+    return false;\r
+  }\r
+  else if (retval.encoded > *size){\r
+    std::stringstream ss;\r
+    ss  <<"Error encoding E2SM Control Header . Reason =  encoded pdu size " << retval.encoded << " exceeds buffer size " << *size << std::endl;\r
+    error_string = ss.str();\r
+    return false;\r
+  }\r
+  else{\r
+    *size = retval.encoded;\r
+  }\r
+    \r
+  return true;\r
+}\r
+\r
+\r
+bool e2sm_control::encode_control_message(unsigned char *buf, size_t *size, e2sm_message_helper &helper){\r
+\r
+  set_message_fields(message, helper); \r
+\r
+  int ret_constr = asn_check_constraints(&asn_DEF_E2SM_gNB_X2_controlMessage, message, errbuf, &errbuf_len);\r
+  if(ret_constr){\r
+    error_string.assign(&errbuf[0], errbuf_len);\r
+    error_string = "E2SM Control Message Constraint failed : " + error_string;\r
+    return false;\r
+  }\r
+\r
+  asn_enc_rval_t retval = asn_encode_to_buffer(0, ATS_ALIGNED_BASIC_PER, &asn_DEF_E2SM_gNB_X2_controlMessage, message, buf, *size);\r
+  if(retval.encoded == -1){\r
+    error_string.assign(strerror(errno));\r
+    error_string = "Error encoding E2SM Control Message. Reason = " + error_string;\r
+    return false;\r
+  }\r
+  else if (retval.encoded > *size){\r
+    std::stringstream ss;\r
+    ss  <<"Error encoding E2SM Control Message . Reason =  encoded pdu size " << retval.encoded << " exceeds buffer size " << *size << std::endl;\r
+    error_string = ss.str();\r
+    \r
+    return false;\r
+  }\r
+  else{\r
+    *size = retval.encoded;\r
+  }\r
+  \r
+  return true;\r
+}\r
+\r
+\r
+\r
+// Used when generating an indication header \r
+bool e2sm_control::set_header_fields(E2SM_gNB_X2_controlHeader_t *header,  e2sm_header_helper &helper){\r
+\r
+  if (header == 0){\r
+    error_string = "Invalid reference for E2SM Control Header set fields";\r
+    return false;\r
+  }\r
+  \r
+  \r
+  header->interfaceDirection = helper.interface_direction;\r
+  header->interface_ID.present = Interface_ID_PR_global_gNB_ID;\r
+  header->interface_ID.choice.global_gNB_ID = &gNodeB_ID;\r
+\r
+\r
+  // to do : need to put correct code here for upding plmn id and gNodeB\r
+  // for now just place holders :\r
+  memcpy(gNodeB_ID.pLMN_Identity.buf, helper.plmn_id.c_str(), 3);\r
+  gNodeB_ID.pLMN_Identity.size = 3;\r
+  \r
+  memcpy(gNodeB_ID.gNB_ID.choice.gNB_ID.buf, helper.egNB_id.c_str(), 3);\r
+  gNodeB_ID.gNB_ID.choice.gNB_ID.size = 3;\r
+  \r
+  // we only do global gNodeB id for now, not eNodeB\r
+  gNodeB_ID.gNB_ID.present = GNB_ID_PR_gNB_ID;\r
+\r
+  return true;\r
+  \r
+};\r
+\r
+\r
+// used when decoding an indication header\r
+bool e2sm_control::get_header_fields(E2SM_gNB_X2_controlHeader_t *header,  e2sm_header_helper &helper){\r
+\r
+  if (header == 0){\r
+    error_string = "Invalid reference for E2SM Control header get fields";\r
+    return false;\r
+  }\r
+  \r
+  helper.interface_direction = header->interfaceDirection;\r
+  helper.plmn_id.assign((const char *)header->interface_ID.choice.global_gNB_ID->pLMN_Identity.buf, header->interface_ID.choice.global_gNB_ID->pLMN_Identity.size);\r
+  helper.egNB_id.assign((const char *)header->interface_ID.choice.global_gNB_ID->gNB_ID.choice.gNB_ID.buf, header->interface_ID.choice.global_gNB_ID->gNB_ID.choice.gNB_ID.size);\r
+  \r
+  // to do : add code to decipher plmn and global gnodeb from ints (since that is likely the convention for packing)\r
+\r
+  return true;\r
+}\r
+\r
+\r
+\r
+// Used when generating an indication message \r
+bool   e2sm_control::set_message_fields(E2SM_gNB_X2_controlMessage_t *interface_message,  e2sm_message_helper &helper){\r
+\r
+  if(interface_message == 0){\r
+    error_string = "Invalid reference for E2SM Control Message set fields";\r
+    return false;\r
+  }\r
+\r
+  // interface-message is an octet string. just point it to the buffer\r
+  interface_message->interfaceMessage.buf = &(helper.x2ap_pdu[0]);\r
+  interface_message->interfaceMessage.size = helper.x2ap_pdu_size;\r
+\r
+  return true;\r
+  \r
+};\r
+\r
+// used when decoding an indication message\r
+bool e2sm_control::get_message_fields( E2SM_gNB_X2_controlMessage_t *interface_message, e2sm_message_helper &helper){\r
+\r
+  \r
+  if(interface_message == 0){\r
+    error_string = "Invalid reference for E2SM Control Message get fields";\r
+    return false;\r
+  }\r
+\r
+  // interface message is an octet string\r
+  helper.x2ap_pdu = interface_message->interfaceMessage.buf;;\r
+  helper.x2ap_pdu_size = interface_message->interfaceMessage.size;\r
+\r
+  return true;\r
+  \r
+}\r
+  \r