diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/README.txt b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/README.txt new file mode 100644 index 0000000000000000000000000000000000000000..87e991c1fe53ab91278f646c8a99c82fb09a6d08 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/README.txt @@ -0,0 +1,251 @@ + +######################################## +# +# SPECTRA DEMO INSTALLATION AND RUN +# +# +# +# Rui Costa <ferreira@eurecom.fr> +# +# v1.12 (13 Oct 2014) +# +######################################## + +######################################## +1. REQUIREMENTS + + + To properly run the demo these are the main pieces of software + that are required: + - OPENAIR4G (checkout the main trunk from the EURECOM SVN). + - ODTONE (Open 802.21 protocol from ITAveiro git server). + - Boost C++ Libraries are requested by ODTONE (v1.48 or higher) + - REST Toolkit (Codename "Casablanca") + - Adequate shell/bash start-up scripts for both eNB and UE. + + BEFORE attempting any installation YOU MUST check your system for + any required software that these components require, as for + instance, a compatible GCC/G++ compiler, build tools, etc. + + NOTES BEFORE INSTALLATION: + + N1. Installation SHOULD NOT be done as ROOT. IF and WHEN super + user permissions are required you will be prompted for them. + + N2. You MUST have a valid EURECOM SVN-enabled account for the + Openair4G code. + + N3. For ODTONE software, you MUST have installed the ITAveiro + SSL/TLS certificates available at their webserver. + + N4. You MUST have Internet access in order to properly build and + deploy the software, as some of the software is fetched online. + + +######################################## +2. BUILD & INSTALL + + 2.1. PREPARATION + Start by creating a folder on a non-root directory, preferably + your home folder. Copy the <spectra_demo_prepare.sh> script into + the newly created folder, henceforth referred to as + <spectra_root_folder>. + + + 2.2. INSTALLATION + There are two ways of processing the installation: + - If you already have performed a checkout of the OpenAir svn. + (Aimed at single installations for OAI developers.) + - If you only have the demonstration source code & scripts. + (To allow easier deployment over several machines, just copy + the demo's source folder, including the prepare and install + scripts to which ever machine you need and let the scripts do + the work for you) + + + 2.2.1 From the OPENAIR SVN + Go to the folder: + <openair_root_folder>/trunk/targets/PROJECTS/SPECTRA/DEMO_SPECTRA + and run the installation script with the node type to install: + # ./spectra_demo_install.sh -oai_svn <node_type> + + 2.2.2 From the demo's source code and scripts. + Go to the <spectra_root_folder> and run: + # ./spectra_demo_install.sh -full <node_type> + + + + If the previous step has been well performed, you should now have + a local working copy of the OpenAir SVN on your local machine, along + with additional code required for the demo. + + + 2.3. CRM + This component of the demo MUST be compiled and installed separately. + Check <spectra_root_folder>/CRM/ for more details. + + 2.4. TVWS Sensing Measurements + There is a server and client for simulating the TVWS measurements and + run the cognitive algorithm to activate the required actions on the + LTE link. (this is the goal of the demo) + Copy the client & server located at <spectra_root_folder>/clientSensing/ + into the ENB2 startup folder (<openair_root_folder>/targets/PROJECTS/SPECTRA) + + + 2.5. TROUBLESHOOTING + If for some reason the installation does not run as expected, check + for permission-related issues or missing software packages. + Otherwise, try to install the components one by one, as defined in + the installation scripts. + (see <spectra_demo_prepare.sh> and <spectra_demo_install.sh> for + compilation and configuration details). + + The preparation script will automatically call the installation + script, but in case it does not, run it manually, it is located at: + <spectra_root_folder>/openair4G/targets/PROJECTS/SPECTRA/ + + NOTE: During installation we apply a patch to ODTONE, due to the + lack of a definition for LTE links and their attributes. + However, as new versions are released the patch may need to + be adapted. Therefore, be aware of compilation issues + originating from the compilation of the ODTONE core functions. + + +######################################## +3. CONFIGURATION + + You MUST make sure that the configuration of your physical and + logical interfaces matches with your Open Air and ODTONE + configuration files. Take all parameters into account while + checking the configuration files and launch scripts, located in: + + <spectra_root_folder>/openair4G/targets/PROJECTS/SPECTRA/ + + <spectra_root_folder>/ODTONE/dist/ + + 3.1 Base Configuration + This is the network configuration on the config files: + + UE1 + eth0 (192.168.13.2) - oai0 (10.0.0.2) (LTE emulated link) + + ENB1 + eth0 (192.168.12.122) internet access gw (192.168.12.100) + eth1 (192.168.13.1) - oai0 (10.0.0.1) (LTE emulated link) + eth2 (192.168.14.3) (Internal Relay Connection) + + UE2 + eth0 (192.168.14.4) (Internal Relay Connection) + eth1 (192.168.15.5) - oai0 (10.0.2.3) (LTE emulated link) + + ENB2 + eth0 (192.168.15.6) - oai0 (10.0.2.4) (LTE emulated link) + + + +######################################## +4. RUNNING THE DEMO + + 1. We advise you to turn off the network manager service in order + for the startup script configuration to be effective and final. + This can be achieved by executing: + + # sudo service network-manager stop + + 2. (On all machines) Go to OpenAir Launch folder: + <spectra_root_folder>/openair4G/targets/PROJECTS/SPECTRA/ + + 3. Launch the eNB1 on eNB1 machine: + # sudo ./start_enb.bash + Wait for eNB1 to reach stand by mode. + + 4. Launch the UE1 on UE1 machine: + # sudo ./start_ue.bash + + 5. Launch the eNB2 on eNB2 machine: + # sudo ./start_enb.bash + Wait for eNB2 to reach stand by mode. + + 6. Launch the UE2 on UE2 machine: + # sudo ./start_ue.bash + + 7. Any other modules will be launched automatically + if present. (CRM client, TVWS sensing, MIHF, MIH Users, etc.) + + Enjoy the demo! + + + + + + + +######################################## +CHANGELOG + +v0.99 (03/09/2014) + - First <spectra_demo_prepare.sh> script and <README> file. + +v1.00 (04/09/2014) + - Added compartmentalization for each sw component + - Added full installation procedure from scratch + - Fetching sw from repositories, compilation & installation + - Revisioned output + +v1.01 (05/09/2014) + - Added OpenAir conf files <spectra_demo_src/openair_conf> + - Added MIHF Users <spectra_demo_src/mih_users> + - Reduced printed output + +v1.02 (08/09/2014) + - Bug correcting on installation and configuration paths & scripts + - Patched Open Air script <targets/PROJECTS/SPECTRA/build_all.sh> + +v1.03 (09/09/2014) + - Patched MIHF to support LTE link based on previous code + - Added <spectra_demo_src/mihf/> + with patch files: <link.hpp> <bin_query.hpp> + - Revisioned <spectra_demo_prepare.sh> + +v1.04 (11/09/2014) + - Corrected MIHF patch bugs + - Added patch file <archive.hpp> + - Corrected path and filename bugs on <spectra_demo_prepare.sh> + +v1.05 (15/09/2014) + - Corrected MIHF patch bugs + - Added a number of patch files + - Adjusted patch execution on <spectra_demo_prepare.sh> + +v1.06 (17/09/2014) + - Working demo 1 eNB 1 UE + - Modified configuration files for UE and eNB + - Substituted overwrite of files into a Git patch for SPECTRA + +v1.07 (19/09/2014) + - Expanded demo to 4 machines: + - 2 real: ue1, eNB1 + - 2 virtual: eNB2, ue2 (both hosted by eNB1) + +v1.08 (23/09/2014) + - Included CRM client and TVWS sensing modules. + +v1.09 (24/09/2014) + - Final Demo setup ready. + +v1.10 (29/09/2014) + - Updated RRC code for OpenAir. + - Code commit to OpenAir SVN trunk. + +v1.11 (01/10/2014) + - Restructure of source code tree and installation files to + include the full demo (2 eNB and 2 ue) + - Splitted prepare and install scripts. + - Added colour coded steps and error stoppage on install script. + +v1.12 (13/10/2014) + - Added bug fixes on launch and install scripts + - Added final version to openair svn + - Reworked README file. + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_install.sh b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_install.sh new file mode 100755 index 0000000000000000000000000000000000000000..089b853cf1b2cc7a65661ccc188358aa057b2fbb --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_install.sh @@ -0,0 +1,290 @@ +#!/bin/bash + +######################################## +# +# SPECTRA DEMO INSTALLATION AND RUN +# +# +# +SCRIPT_VERSION="v1.11" +DATE="01 Oct 2014" +AUTHOR="Rui Costa" +EMAIL="ferreira@eurecom.fr" +######################################## + +OWN_TAG=$0 + +INSTALL_PATH=$PWD +OPENAIR_PATH=$INSTALL_PATH/openair4G +ODTONE_PATH=$INSTALL_PATH/ODTONE +BOOST_PATH=$INSTALL_PATH/boost_1_49_0 +SPECTRA_SRC_PATH=$INSTALL_PATH/spectra_demo_src + +NODE_TYPE=NONE +INSTALL_TYPE=NONE + +########################################################################################### UTILS +function util_echo { +# return value passed in $1 (usually $?) +# return value = 0 - OK - Light green +if [ $1 -eq 0 ] ; then + echo -e "\e[92m\e[1m[$OWN_TAG]\e[0m $2" +# return value != 0 - ERROR +else +# return value -gt 255 - WARNING - Light Yellow + if [ $1 -gt 255 ] ; then + echo -e "\e[93m\e[1m[$OWN_TAG]\e[0m $2" + else +# return value 1 to 255 - ERROR - Light RED + echo -e "\e[91m\e[1m[$OWN_TAG]\e[0m $2" + exit + fi +fi +} + +########################################################################################### BOOST C++ +function fetch_boost { +cd $INSTALL_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "BOOST C++ Library ..." +# Although new versions of boost exist, when compiling ODTONE he asks for v1.49.0 +# (we humour it so to not have linkage problems later) +wget --trust-server-names "http://downloads.sourceforge.net/project/boost/boost/1.49.0/boost_1_49_0.tar.gz?r=&ts=1412338583&use_mirror=optimate" +util_echo $? "Fetched BOOST C++ Library ..." +sleep 2 +tar -xzf boost_1_49_0.tar.gz +util_echo $? "Extracted BOOST C++ Library ..." +} + +function build_boost { +cd $BOOST_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "BOOST C++ Library ..." +sleep 2 +# Configuring Boost +./bootstrap.sh +util_echo $? "Bootstrapped BOOST C++ Library ..." +sleep 2 +# Compiling Boost (pthread link flag is vital) +./b2 --with-date_time --with-thread linkflags=-lpthread +util_echo $? "Built BOOST C++ Library ..." +} + +########################################################################################### ODTONE +function fetch_odtone { +cd $INSTALL_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "ODTONE ..." +sleep 2 +# Getting newest version of Odtone +git clone https://github.com/ATNoG/ODTONE.git +util_echo $? "Checked Out ODTONE from ITAveiro GIT..." +cd $ODTONE_PATH +git submodule update --init +util_echo $? "Updated ODTONE submodules..." +} + +function build_odtone { +cd $ODTONE_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "ODTONE ..." + +# /***** REPLACED WITH Git spectra PATCH +# +# Establishing the pointer to a local compilation of boost instead of the usual in /usr/local/src/ +#touch boost-build.jam +#echo "boost-build $BOOST_PATH/tools/build/v2 ;" > boost-build.jam +# Copying our new MIH Users +#cp -r $SPECTRA_SRC_PATH/mih_users/* $ODTONE_PATH/app/ +# we MUST patch these files! ODTONE by default has no knowledge of LTE links +# this overwrite of files ensures that full LTE link parameters exist. +# A better way of paching the files should be implemented later!! +#cp -r $SPECTRA_SRC_PATH/mihf/link.hpp $ODTONE_PATH/inc/odtone/mih/types/ +#cp -r $SPECTRA_SRC_PATH/mihf/archive.hpp $ODTONE_PATH/inc/odtone/mih/detail/ +#cp -r $SPECTRA_SRC_PATH/mihf/address.hpp $ODTONE_PATH/inc/odtone/mih/types/ +#cp -r $SPECTRA_SRC_PATH/mihf/bin_query.hpp $ODTONE_PATH/inc/odtone/mih/types/ +#cp -r $SPECTRA_SRC_PATH/mihf/CMakeLists.txt $ODTONE_PATH/lib/odtone/ +#cp -r $SPECTRA_SRC_PATH/mihf/conf.cpp $ODTONE_PATH/lib/odtone/ +#cp -r $SPECTRA_SRC_PATH/mihf/conf.hpp $ODTONE_PATH/inc/odtone/ +#cp -r $SPECTRA_SRC_PATH/mihf/Jamfile $ODTONE_PATH/lib/odtone/ +#cp -r $SPECTRA_SRC_PATH/mihf/command_service.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/dst_transaction.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/event_service.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/link_book.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/main.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/service_access_controller.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/service_management.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/src_transaction.cpp $ODTONE_PATH/src/mihf/ +#cp -r $SPECTRA_SRC_PATH/mihf/transaction_pool.cpp $ODTONE_PATH/src/mihf/ +# +# *****/ +# Applying SPECTRA patch to ODTONE source code +git apply --check --stat $SPECTRA_SRC_PATH/common/mihf_src/0001-SPECTRA-patch-for-ODTONE-v0.1.patch +util_echo $? "Checking if SPECTRA patch can be aplied..." +git apply $SPECTRA_SRC_PATH/common/mihf_src/0001-SPECTRA-patch-for-ODTONE-v0.1.patch +util_echo $? "Applying SPECTRA patch to ODTONE..." +# Changing hostname on the boost build +sed -i "s%nikaia%$USER%g" $ODTONE_PATH/boost-build.jam +util_echo $? "Modifying boost-build.jam..." +# Compiling Odtone core +$BOOST_PATH/b2 linkflags=-lpthread +util_echo $? "Compiled ODTONE..." +# Copying lte_test_user for this node +rm -rf $ODTONE_PATH/app/lte_test_user +cp -r $SPECTRA_SRC_PATH/$NODE_TYPE/mih_user/* $ODTONE_PATH/app/ +util_echo $? "Installed ODTONE mih user for: $NODE_TYPE..." +# Compiling lte_test_user +$BOOST_PATH/b2 app/lte_test_user/ linkflags=-lpthread linkflags=-lrt +util_echo $? "Compiled ODTONE apps..." +# Copying odtone confs +cp $SPECTRA_SRC_PATH/$NODE_TYPE/mih_conf/*.conf $ODTONE_PATH/dist/ +util_echo $? "Installed ODTONE configuration files for: $NODE_TYPE..." +} + +########################################################################################### OPENAIR +function fetch_openair { +mkdir $OPENAIR_PATH +cd $INSTALL_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "OPENAIR EURECOM ..." +sleep 2 +svn co http://svn.eurecom.fr/openair/openair4G/trunk $OPENAIR_PATH +util_echo $? "Checked Out openair4G/trunk from EURECOM SVN..." +} + +function build_openair { +cd $OPENAIR_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "OPENAIR ..." +sleep 2 +# Putting in place pre-made scripts for the build of openair +cp $SPECTRA_SRC_PATH/common/openair_scripts/* $OPENAIR_PATH/targets/PROJECTS/SPECTRA/ +util_echo $? "Installed OpenAir common build scripts ..." +# Putting in place pre-made scripts for the conf of openair +cp $SPECTRA_SRC_PATH/$NODE_TYPE/oai_conf/* $OPENAIR_PATH/targets/PROJECTS/SPECTRA/ +util_echo $? "Installed OpenAir startup scripts and conf for $NODE_TYPE..." +cd $OPENAIR_PATH/targets/PROJECTS/SPECTRA/ +# Updating the path location of ODTONE and BOOST on the script, according to the local machine. +sed -i "s%path_to_boost_folder%$BOOST_PATH%g" env_802dot21.bash +util_echo $? " Fixed BOOST path in env_802dot21.bash... $BOOST_PATH" +sed -i "s%path_to_odtone_folder%$ODTONE_PATH%g" env_802dot21.bash +util_echo $? " Fixed ODTONE path in env_802dot21.bash... $ODTONE_PATH" +if [ $NODE_TYPE == "enb2" ] ; then + sed -i "s%enb_lte_user%enb2_lte_user%g" env_802dot21.bash + util_echo $? " Fixed exec name (enb2_lte_user) for $NODE_TYPE..." +# sed -i "s%ENB_MIH_USER_CONF_FILE=enb_lte_user.conf%ENB_MIH_USER_CONF_FILE=enb2_lte_user.conf%g" env_802dot21.bash + util_echo $? " Fixed conf name (enb2_lte_user.conf) for path $NODE_TYPE..." +fi +if [ $NODE_TYPE == "ue2" ] ; then + sed -i "s%ue_lte_user%ue2_user%g" env_802dot21.bash + util_echo $? " Fixed exec name (enb2_lte_user) for $NODE_TYPE..." + util_echo $? " Fixed conf name (enb2_lte_user.conf) for path $NODE_TYPE..." +fi +# executing the (re)build of all components +# superuser necessary for installing the ITTY tool of openair +./build_all.bash +util_echo $? "Finished OpenAir build..." +} + +########################################################################################### PRINT_HELP +function print_help { +util_echo 333 " -------------------------------------------------" +util_echo 333 " USAGE: " +util_echo 333 " $ $OWN_TAG <install_type> <node_type>" +util_echo 333 " " +util_echo 333 " <install_type>" +util_echo 333 " -full If you start from scratch" +util_echo 333 " -oai_svn If you got this script after" +util_echo 333 " checking out the Openair SVN" +util_echo 333 " <node_type>" +util_echo 333 " -ue1 For installing ue1 files" +util_echo 333 " -ue2 For installing ue2 files" +util_echo 333 " -enb1 For installing eNB1 files" +util_echo 333 " -enb2 For installing eNB2 files" +util_echo 333 " " +util_echo 333 " -------------------------------------------------" +util_echo 333 " " +exit +} + +########################################################################################### MAIN +clear +# Testing Args +if [ $# -ne 2 ]; then + util_echo 333 " Incorrect Number of Arguments!!" + print_help +fi +if [ $1 == "-full" ]; then + INSTALL_TYPE=FULL +fi +if [ $1 == "-oai_svn" ]; then + INSTALL_TYPE=OAISVN +fi +if [ $2 == "-ue1" ]; then + NODE_TYPE=ue1 +fi +if [ $2 == "-ue2" ]; then + NODE_TYPE=ue2 +fi +if [ $2 == "-enb1" ]; then + NODE_TYPE=enb1 +fi +if [ $2 == "-enb2" ]; then + NODE_TYPE=enb2 +fi +if [ $INSTALL_TYPE == "NONE" ]; then + util_echo 333 " Incorrect Arguments (type of installation) !!" + print_help +fi +if [ $NODE_TYPE == "NONE" ]; then + util_echo 333 " Incorrect Arguments (type of node) !!" + print_help +fi +# Starting... +util_echo $? " -------------------------------------------------" +util_echo $? " SPECTRA DEMO " +util_echo $? " Installation Script" +util_echo $? " " +util_echo $? " $SCRIPT_VERSION $DATE" +util_echo $? " " +util_echo $? " Author: $AUTHOR <$EMAIL> " +util_echo $? " -------------------------------------------------" +util_echo 333 " Performing installation type: $INSTALL_TYPE for $NODE_TYPE" +sleep 3 +util_echo $? " -------------------------------------------------" +util_echo $? " >> Fetching sources" +util_echo $? " -------------------------------------------------" +if [ $INSTALL_TYPE == "OAISVN" ]; then + INSTALL_PATH=$PWD/../../../../ + OPENAIR_PATH=$PWD/../../../ + ODTONE_PATH=$INSTALL_PATH/ODTONE + BOOST_PATH=$INSTALL_PATH/boost_1_49_0 + SPECTRA_SRC_PATH=$PWD/spectra_demo_src +else + fetch_openair +fi +fetch_odtone +fetch_boost +sleep 1 +util_echo $? " -------------------------------------------------" +util_echo $? " >> Building" +util_echo $? " -------------------------------------------------" +sleep 2 +build_boost +build_odtone +build_openair +util_echo 333 " Open Air Configuration files at:" +util_echo 333 " $OPENAIR_PATH/targets/PROJECTS/SPECTRA/" +util_echo 333 " " +util_echo 333 " ODTONE Configuration files at:" +util_echo 333 " $ODTONE_PATH/dist/" +util_echo $? " " +util_echo $? " " +util_echo $? " -------------------------------------------------" +util_echo $? " SPECTRA DEMO preparation script ... finished!" +util_echo $? " -------------------------------------------------" +cd $INSTALL_PATH +exit 0 +########################################################################################### EOF + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_prepare.sh b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_prepare.sh new file mode 100755 index 0000000000000000000000000000000000000000..8ce29f46a8d3c0040b706b18e02a5df626219e2a --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_prepare.sh @@ -0,0 +1,80 @@ +#!/bin/bash + +######################################## +# +# SPECTRA DEMO PREPARATION +# +# +# +SCRIPT_VERSION="v1.11" +DATE="01 Oct 2014" +AUTHOR="Rui Costa" +EMAIL="ferreira@eurecom.fr" +######################################## + +OWN_TAG=$0 + +INSTALL_PATH=$PWD +OPENAIR_PATH=$INSTALL_PATH/openair4G +ODTONE_PATH=$INSTALL_PATH/ODTONE +BOOST_PATH=$INSTALL_PATH/boost_1_49_0 +SPECTRA_SRC_PATH=$INSTALL_PATH/spectra_demo_src +SPECTRA_OAI_PATH=$OPENAIR_PATH/targets/PROJECTS/SPECTRA/ + +########################################################################################### UTILS +function util_echo { +# return value passed in $1 (usually $?) +# return value = 0 - OK - Light green +if [ $1 -eq 0 ] ; then + echo -e "\e[92m\e[1m[$OWN_TAG]\e[0m $2" +# return value != 0 - ERROR +else +# return value -gt 255 - WARNING - Light Yellow + if [ $1 -gt 255 ] ; then + echo -e "\e[93m\e[1m[$OWN_TAG]\e[0m $2" + else +# return value 1 to 255 - ERROR - Light RED + echo -e "\e[91m\e[1m[$OWN_TAG]\e[0m $2" + exit + fi +fi +} + +########################################################################################### OPENAIR +function fetch_openair { +mkdir $OPENAIR_PATH +cd $INSTALL_PATH +util_echo $? "-------------------------------------------------" +util_echo $? "OPENAIR EURECOM ..." +sleep 2 +svn co http://svn.eurecom.fr/openair/openair4G/trunk $OPENAIR_PATH +util_echo $? "Checked Out openair4G/trunk from EURECOM SVN..." +} + +########################################################################################### MAIN +clear +# Starting... +util_echo $? " -------------------------------------------------" +util_echo $? " SPECTRA DEMO " +util_echo $? " Prepararation Script" +util_echo $? " " +util_echo $? " $SCRIPT_VERSION $DATE" +util_echo $? " " +util_echo $? " Author: $AUTHOR <$EMAIL> " +util_echo $? " -------------------------------------------------" +util_echo $? " " +sleep 3 +util_echo $? " -------------------------------------------------" +util_echo $? " >> Fetching sources" +util_echo $? " -------------------------------------------------" + +fetch_openair + +cd $SPECTRA_OAI_PATH/ +util_echo $? " >> Finished Prepare Script!!" +util_echo $? " >> Entering Install Script!!" +./spectra_demo_install.sh -oai_svn $1 + +########################################################################################### EOF + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient new file mode 100644 index 0000000000000000000000000000000000000000..29465e592af164bb52a3340f0900e1acead09fa9 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient.cpp new file mode 100644 index 0000000000000000000000000000000000000000..f079c0594ec79e13f4b0d9d2fdd80c6c57561734 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient.cpp @@ -0,0 +1,249 @@ +#include <cpprest/http_client.h> +#include <cpprest/json.h> +#include <iostream> +#include <ostream> +#include <sstream> +#include <fstream> +#include "cpprest/basic_types.h" +#include "cpprest/asyncrt_utils.h" +#include "cpprest/uri.h" +#include <string> + +#include "CRMClient.hpp" + +using namespace std; +using namespace web; +using namespace web::http; +using namespace web::http::client; +using namespace utility; + +// namespace odtone { + +CRMClient::CRMClient(char* url) +{ + m_url = (char*) (malloc (200* sizeof (char))); + m_url = url; +} + +CRMClient::CRMClient() +{ +} + +CRMClient::~CRMClient() +{ +} + +void CRMClient::SetUrl(char* url) +{ + m_url = url; +} + +// Retrieves a JSON value from an HTTP request. +pplx::task<void> CRMClient::RequestJSONValueAsync() +{ + http_client client (U(m_url)); + return client.request(methods::GET).then([](http_response response) -> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + ofstream myfile; + myfile.open ("outputGET.txt"); + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + +//Parsing Begin + + cout << U("Start Parsing...") << endl; + for(auto iterArray = v.cbegin(); iterArray != v.cend(); ++iterArray) + { + const json::value &arrayValue = iterArray->second; + + for(auto iterInner = arrayValue.cbegin(); iterInner != arrayValue.cend(); ++iterInner) + { + const json::value &propertyValue = iterInner->second; + for(auto iterlast = propertyValue.cbegin(); iterlast != propertyValue.cend(); ++iterlast) + { + const json::value &Name = iterlast->first; + const json::value &Value = iterlast->second; + cout<< U("Parameter: ") << Name.to_string()<<endl; + cout<< U("Value: ") << Value.to_string()<< std::endl; + } + } + cout << std::endl; + } + +//parsing End + + myfile << jsonString.c_str(); + myfile.close(); + + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +// Stores a JSON value (a Policy) from an HTTP request. +pplx::task<void> CRMClient::StoreJSONValuePolicies(char * param1, char * param2, char * param3, char * param4) +{ + + http_client client (U(m_url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("pid"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("description"), json::value(param3))); + putvalue.push_back(make_pair(json::value("value"), json::value(param4))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); +} + + +// Stores a JSON value (a Measurement) from an HTTP request. +pplx::task<void> CRMClient::StoreJSONValuemeasurements(char * param1, char * param2, char * param3, char * param4, char * param5, char * param6) +{ + + http_client client (U(m_url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("key"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("type"), json::value(param3))); + putvalue.push_back(make_pair(json::value("unit"), json::value(param4))); + putvalue.push_back(make_pair(json::value("value"), json::value(param5))); + putvalue.push_back(make_pair(json::value("time"), json::value(param6))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +// Stores a JSON value (a Decision) from an HTTP request. +pplx::task<void> CRMClient::StoreJSONValuedecisions(char * param1, char * param2, char * param3, char * param4, char * param5) +{ + + http_client client (U(m_url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("did"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("description"), json::value(param3))); + putvalue.push_back(make_pair(json::value("value"), json::value(param4))); + putvalue.push_back(make_pair(json::value("time"), json::value(param5))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +pplx::task<void> CRMClient::Delete() +{ + char* url = m_url; + return pplx::create_task([url] +// return pplx::create_task([m_url] + { + http_client client (U(url)); + + return client.request(methods::DEL); + + }).then([](http_response response) + { + if(response.status_code() == status_codes::OK) + { + auto body = response.extract_string(); + + std::wcout << L"Deleted: " << body.get().c_str() << std::endl; + } + }); +} + + +// } \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient.hpp new file mode 100644 index 0000000000000000000000000000000000000000..29949336f4fefc8ebfc6058c56a89b41eb6d3569 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClient.hpp @@ -0,0 +1,56 @@ +//============================================================================== +// Brief : MIH-User +// Authors : FATMA HRIZI <hrizi@eurecom.fr> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#ifndef ODTONE_CRM_CLIENT_HPP +#define ODTONE_CRM_CLIENT_HPP + + +#include <cpprest/http_client.h> +#include <cpprest/json.h> +#include <iostream> +#include <ostream> +#include <sstream> +#include "cpprest/basic_types.h" +#include "cpprest/asyncrt_utils.h" +#include "cpprest/uri.h" +#include <string> + +// namespace odtone { +class CRMClient { + +public: + + CRMClient(); + ~CRMClient(); + CRMClient(char* url); + + void SetUrl(char* url); + pplx::task<void> RequestJSONValueAsync(); + pplx::task<void> StoreJSONValuePolicies(char * param1, char * param2, char * param3, char * param4); + pplx::task<void> StoreJSONValuemeasurements(char * param1, char * param2, char * param3, char * param4, char * param5, char * param6); + pplx::task<void> StoreJSONValuedecisions(char * param1, char * param2, char * param3, char * param4, char * param5); + pplx::task<void> Delete(); + + +private: + char * m_url; + +}; +// } + + +#endif diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClientmain b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClientmain new file mode 100644 index 0000000000000000000000000000000000000000..be19a635600fc47563ad483372126d486aa6876a Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/CRMClientmain differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/Makefile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..baefb60f6074aeb51be8d556d15e2b8aaf9f4d39 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/Makefile @@ -0,0 +1,26 @@ +# CRMClient.o: CRMClient.cpp CRMClient.hpp +# g++-4.8 CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem +# +# +# install: main.cpp +# g++-4.8 -Wall main.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -L. -std=c++11 -lcrmclient -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem -o CRMClientmain + +main: main.cpp + g++-4.8 -Wall main.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -L. -std=c++11 -lcrmclient -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem -o CRMClientmain + +lib: CRMClient.cpp CRMClient.hpp + +# 64 bits compilation +# g++-4.8 -Wall -fPIC -c CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -m64 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem +# g++-4.8 -shared -m64 -Wl,-soname,libcrmclient.so.1 -o libcrmclient.so.1.0 CRMClient.o +# ln -sf libcrmclient.so.1.0 libcrmclient.so +# ln -sf libcrmclient.so.1.0 libcrmclient.so.1 + +# 32 bits compilation + g++-4.8 -Wall -fPIC -c CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem + g++-4.8 -shared -Wl,-soname,libcrmclient.so.1 -o libcrmclient.so.1.0 CRMClient.o + ln -sf libcrmclient.so.1.0 libcrmclient.so + ln -sf libcrmclient.so.1.0 libcrmclient.so.1 + +clean: + rm *.o *.so* diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/main.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/main.cpp new file mode 100644 index 0000000000000000000000000000000000000000..1bad18a3db64a2cfa85a67b5a68ca0afc0a161ee --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/CRMClient/main.cpp @@ -0,0 +1,21 @@ +#include "CRMClient.hpp" + +int main(int argc,char *argv[]) +{ + char * policiesurl = (char*) malloc( 100 * sizeof(char) ); + strcpy(policiesurl,"http://1.gae-spectra.appspot.com/policies"); + +// strcpy(policiesurl,"http://localhost:8888/policies"); + char * measurementsurl = (char*) malloc( 100* sizeof(char) ); + strcpy(measurementsurl,"http://1.gae-spectra.appspot.com/measurements"); +// strcpy(measurementsurl,"http://localhost:8888/measurements"); + char * decisionsurl = (char*) malloc( 100 * sizeof(char)); + strcpy(decisionsurl,"http://1.gae-spectra.appspot.com/decisions"); + int c; + + CRMClient client (policiesurl); + client.RequestJSONValueAsync().wait(); + + + return 0; +} \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/docs/server-gae-codeoverview.doc b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/docs/server-gae-codeoverview.doc new file mode 100644 index 0000000000000000000000000000000000000000..22386112039d3058408c1492e73e1fde74036cb2 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/docs/server-gae-codeoverview.doc differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.classpath b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.classpath new file mode 100644 index 0000000000000000000000000000000000000000..a47d678fae2ee50b6303499fbad6f4b60ad4a31a --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.classpath @@ -0,0 +1,19 @@ +<?xml version="1.0" encoding="UTF-8"?> +<classpath> + <classpathentry kind="src" path="src"/> + <classpathentry exported="true" kind="con" path="com.google.appengine.eclipse.core.GAE_CONTAINER"/> + <classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER"/> + <classpathentry kind="lib" path="lib/org.restlet.jar"/> + <classpathentry kind="lib" path="lib/org.restlet.ext.servlet.jar"/> + <classpathentry kind="lib" path="lib/org.restlet.ext.wadl.jar"/> + <classpathentry kind="lib" path="lib/org.restlet.ext.xml.jar"/> + <classpathentry kind="lib" path="lib/org.restlet.ext.json.jar"/> + <classpathentry kind="lib" path="lib/org.json.jar"/> + <classpathentry kind="lib" path="lib/org.springframework.context-3.1.1.RELEASE.jar"/> + <classpathentry kind="lib" path="lib/org.springframework.web-3.1.1.RELEASE.jar"/> + <classpathentry kind="lib" path="lib/org.springframework.web.servlet-3.1.1.RELEASE.jar"/> + <classpathentry kind="lib" path="lib/org.springframework.beans-3.1.1.RELEASE.jar"/> + <classpathentry kind="lib" path="lib/json_simple-1.1.jar"/> + <classpathentry kind="lib" path="lib/gcm-server.jar"/> + <classpathentry kind="output" path="war/WEB-INF/classes"/> +</classpath> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.project b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.project new file mode 100644 index 0000000000000000000000000000000000000000..f19131ebc36f89db96e80347f983d5365aad65eb --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.project @@ -0,0 +1,43 @@ +<?xml version="1.0" encoding="UTF-8"?> +<projectDescription> + <name>rest-gae-server</name> + <comment></comment> + <projects> + </projects> + <buildSpec> + <buildCommand> + <name>org.eclipse.wst.common.project.facet.core.builder</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>org.eclipse.jdt.core.javabuilder</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>com.google.gdt.eclipse.core.webAppProjectValidator</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>com.google.appengine.eclipse.core.gaeProjectChangeNotifier</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>com.google.appengine.eclipse.core.projectValidator</name> + <arguments> + </arguments> + </buildCommand> + <buildCommand> + <name>com.google.appengine.eclipse.core.enhancerbuilder</name> + <arguments></arguments> + </buildCommand> + </buildSpec> + <natures> + <nature>org.eclipse.jdt.core.javanature</nature> + <nature>com.google.appengine.eclipse.core.gaeNature</nature> + <nature>org.eclipse.wst.common.project.facet.core.nature</nature> + </natures> +</projectDescription> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.settings/com.google.appengine.eclipse.core.prefs b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.settings/com.google.appengine.eclipse.core.prefs new file mode 100644 index 0000000000000000000000000000000000000000..47aa0ca55e3cdb7369b5bf26f96d054e8744f0fe --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.settings/com.google.appengine.eclipse.core.prefs @@ -0,0 +1,9 @@ +#Thu Sep 19 14:41:16 CEST 2013 +eclipse.preferences.version=1 +filesCopiedToWebInfLib=appengine-api-1.0-sdk-1.7.4.jar|appengine-api-labs.jar|appengine-endpoints.jar|appengine-jsr107cache-1.7.4.jar|asm-4.0.jar|datanucleus-api-jdo-3.1.1.jar|datanucleus-api-jpa-3.1.1.jar|datanucleus-appengine-2.1.1.jar|datanucleus-core-3.1.1.jar|geronimo-jpa_2.0_spec-1.0.jar|jdo-api-3.0.1.jar|jsr107cache-1.1.jar|jta-1.1.jar +gaeDatanucleusVersion=v2 +gaeDeployDialogSettings=true +gaeIsEclipseDefaultInstPath=true +googleCloudSqlEnabled=false +localDevMySqlEnabled=true +ormEnhancementInclusions=src/ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.settings/com.google.gdt.eclipse.core.prefs b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.settings/com.google.gdt.eclipse.core.prefs new file mode 100644 index 0000000000000000000000000000000000000000..c6960b06d20cea0ba528ab07a27ea5f518e567cb --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/.settings/com.google.gdt.eclipse.core.prefs @@ -0,0 +1,5 @@ +#Wed Dec 12 09:47:01 CET 2012 +eclipse.preferences.version=1 +jarsExcludedFromWebInfLib= +warSrcDir=war +warSrcDirIsOutput=true diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/gcm-server.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/gcm-server.jar new file mode 100644 index 0000000000000000000000000000000000000000..41264c809503f122f5080cb4fbe8b2750b3dce9e Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/gcm-server.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/json_simple-1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/json_simple-1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..f395f41471a796876278a6c5667ded4632546893 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/json_simple-1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.json.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.json.jar new file mode 100644 index 0000000000000000000000000000000000000000..543438992fab3d820a1d900d65537347be419d21 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.json.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.json.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.json.jar new file mode 100644 index 0000000000000000000000000000000000000000..49b2873525c43f14d4852c8ed50ccb5fe455b32f Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.json.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.servlet.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.servlet.jar new file mode 100644 index 0000000000000000000000000000000000000000..5df964798b6407def1cdcca6a4d377767c556ce9 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.servlet.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.wadl.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.wadl.jar new file mode 100644 index 0000000000000000000000000000000000000000..5a96ccac00d0b35b175060368f5d5115a5093753 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.wadl.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.xml.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.xml.jar new file mode 100644 index 0000000000000000000000000000000000000000..213199baaa7a5e96c2cb62df5c4e324f4eea9aa1 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.ext.xml.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.jar new file mode 100644 index 0000000000000000000000000000000000000000..32a3c8483f275be2fae7d2880af54d473564edc7 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.restlet.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.beans-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.beans-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..a69bcb1f3dab93803c9fb9cc9d3c4dba319640a6 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.beans-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.context-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.context-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..a35e486989fc422b5acf528f29d5dd6c5fade129 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.context-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.web-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.web-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..42ed92c7c2cae23ccf4124f14ed63a04b8a70e8d Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.web-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.web.servlet-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.web.servlet-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..8e27a5bc7e073af91224d384445e7189c5fd8ed7 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/lib/org.springframework.web.servlet-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/META-INF/jdoconfig.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/META-INF/jdoconfig.xml new file mode 100644 index 0000000000000000000000000000000000000000..6961b7ed9bc3356d82bcc8ef44e74416158c76d8 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/META-INF/jdoconfig.xml @@ -0,0 +1,13 @@ +<?xml version="1.0" encoding="utf-8" standalone="no"?> +<jdoconfig xmlns="http://java.sun.com/xml/ns/jdo/jdoconfig" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://java.sun.com/xml/ns/jdo/jdoconfig"> + + <persistence-manager-factory name="transactions-optional"> + <property name="javax.jdo.PersistenceManagerFactoryClass" value="org.datanucleus.api.jdo.JDOPersistenceManagerFactory"/> + <property name="javax.jdo.option.ConnectionURL" value="appengine"/> + <property name="javax.jdo.option.NontransactionalRead" value="true"/> + <property name="javax.jdo.option.NontransactionalWrite" value="true"/> + <property name="javax.jdo.option.RetainValues" value="true"/> + <property name="datanucleus.appengine.autoCreateDatastoreTxns" value="true"/> + <property name="datanucleus.appengine.singletonPMFForName" value="true"/> + </persistence-manager-factory> +</jdoconfig> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/META-INF/persistence.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/META-INF/persistence.xml new file mode 100644 index 0000000000000000000000000000000000000000..d15d3fbbfa0be02a9a5f7bfc0644b4cabd223493 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/META-INF/persistence.xml @@ -0,0 +1,12 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<persistence xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="1.0" xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_1_0.xsd"> + + <persistence-unit name="transactions-optional"> + <provider>org.datanucleus.api.jpa.PersistenceProviderImpl</provider> + <properties> + <property name="datanucleus.NontransactionalRead" value="true"/> + <property name="datanucleus.NontransactionalWrite" value="true"/> + <property name="datanucleus.ConnectionURL" value="appengine"/> + </properties> + </persistence-unit> +</persistence> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/DeviceManager.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/DeviceManager.java new file mode 100644 index 0000000000000000000000000000000000000000..8f0f5e49226541ccb530247b91694e2c7583192c --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/DeviceManager.java @@ -0,0 +1,181 @@ +package fr.eurecom.pushing; + +import java.io.IOException; +import java.io.PrintWriter; +import java.util.Date; +import java.util.Iterator; +import java.util.List; +import java.util.UUID; +import java.util.logging.Level; +import java.util.logging.Logger; + +import javax.servlet.ServletException; +import javax.servlet.http.HttpServlet; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; +import org.mortbay.log.Log; +import org.restlet.resource.Get; +import fr.eurecom.senml.entity.RegisteredDeviceType; +import fr.eurecom.senml.persistence.JDOStorage; + +/** + * Exposes the following URLs: + * - /pushing/register, POST, to register a device + * - /pushing/unregister, POST, to unregister a device + * - /pushing/list, GET, to retrieve the list of registered devices + * - /pushing/test, GET, test a notification to the registered devices + * + */ +public class DeviceManager extends HttpServlet { + /** + * + */ + private static final long serialVersionUID = 1L; + private static final Logger log = Logger.getLogger("DeviceManager"); + + public void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { + System.out.println("OK doPost"); + System.out.println(request.getParameter("reqType") + " " + request.getParameter("deviceType")); + try { + if (request.getParameter("reqType").equals("reg")) { + // Registration of a device. Android or iOS? + if (request.getParameter("deviceType").equals("android")) { + registerAndroidDevice(request, response); + } else if (request.getParameter("deviceType").equals("ios")) { + registerAppleDevice(request, response); + } + } else if (request.getParameter("reqType").equals("unreg")) { + System.out.println("OK UNREGISTERING!"); + if (request.getParameter("deviceType").equals("android")) { + System.out.println("UNREGISTERING ANDROID!"); + unregisterAndroidDevice(request, response); + } else if (request.getParameter("deviceType").equals("ios")) { + unregisterAppleDevice(request, response); + } + } + } catch (JSONException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + response.setStatus(500); + } + } + + + private void unregisterAppleDevice(HttpServletRequest request, + HttpServletResponse response) { + // TODO Auto-generated method stub + response.setStatus(403); + } + + + private void unregisterAndroidDevice(HttpServletRequest request, + HttpServletResponse response) { + String key = request.getParameter("registrationID"); + // TODO Auto-generated method stub + List<RegisteredDeviceType> devices = JDOStorage.getInstance().getAll(RegisteredDeviceType.class); + Iterator<RegisteredDeviceType> iter = devices.iterator(); + while (iter.hasNext()) { + RegisteredDeviceType device = iter.next(); + System.out.print("Analyzing " + device.getDeviceRegistrationKey()); + if (device.getDeviceRegistrationKey().equals(key)) { + JDOStorage.getInstance().delete(device); + System.out.println("OK"); + return; + } else { + System.out.println("NO!"); + } + } + } + + private void registerAppleDevice(HttpServletRequest request, HttpServletResponse response) { + // TODO Auto-generated method stub + response.setStatus(403); + } + + private void registerAndroidDevice(HttpServletRequest request, HttpServletResponse response) throws JSONException, IOException { + Date n = new Date(); + + boolean isadmin = (request.getParameter("isadmin") != null); + + + + RegisteredDeviceType device = new RegisteredDeviceType( + UUID.randomUUID().toString(), + request.getParameter("deviceType") + ": " + request.getParameter("deviceDet"), + request.getParameter("devicekey"), n.getTime()); + if (isadmin) { + System.out.println("ADMIN"); + device.setIsadmin(isadmin); + } + JDOStorage.getInstance().write(device); + response.setContentType("text/html"); + PrintWriter writer = response.getWriter(); + response.setStatus(200); + writer.write("Registered"); + } + + /** + * Handle HTTP GET method / Json + * + * @return a JSON Zone Representation + */ + @Get + public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { + PrintWriter writer = response.getWriter(); + if (request.getPathInfo().equals("/test")) { + GCMPushingServer GCM = GCMPushingServer.getInstance(); + try { + GCM.sendNotification("notification test "); + writer.write("Sent a notification test to the following devices:"); + } catch (JSONException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } else if (request.getPathInfo().equals("/alarm")) { + GCMPushingServer GCM = GCMPushingServer.getInstance(); + try { + GCM.sendNotification("ALARM for sensor: " + request.getParameter("sensor")); + writer.write("OK"); + } catch (JSONException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + writer.write("NO"); + } + return; + } + List<RegisteredDeviceType> devices = JDOStorage.getInstance().getAll(RegisteredDeviceType.class); + if (devices.isEmpty()) { + // No devices registered + response.setStatus(200); + writer.write("The list of devices is empty."); + return; + } + response.setContentType("text/html"); + JSONArray jsonDevices = new JSONArray(); + Iterator<RegisteredDeviceType> iter = devices.iterator(); + writer.write("<table border=1><tr style=\"font-weight: bold;\"><th>Device Logo</th><th>Device Type</th><th>registered on</th><th>Registration Key</th></tr>"); + // Get the list of devices and build the json + try { + while (iter.hasNext()) { + RegisteredDeviceType current = iter.next(); + JSONObject obj = new JSONObject(current); + jsonDevices.put(obj); + String deviceLogo = obj.getString("deviceType").split(":")[0].toLowerCase(); + + writer.write("<tr><td><img src=\"/images/" + deviceLogo + ".jpg\"/></td><td>" + + obj.getString("deviceType") + "</td><td>" + + obj.getString("time") + "</td><td>" + + obj.getString("deviceRegistrationKey") + "</td></tr>"); + } + writer.write("</table>"); + } catch (Exception e) { + response.setStatus(500); + log.log(Level.WARNING, "", e); + + } + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/GCMPushingServer.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/GCMPushingServer.java new file mode 100644 index 0000000000000000000000000000000000000000..fae42ba44badbf51878919aa30bf9cc5e23f4c87 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/GCMPushingServer.java @@ -0,0 +1,206 @@ +package fr.eurecom.pushing; + +import java.io.IOException; +import java.io.OutputStream; +import java.net.HttpURLConnection; +import java.net.URL; +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import com.google.android.gcm.server.Message; +import com.google.android.gcm.server.MulticastResult; +import com.google.android.gcm.server.Sender; + +import fr.eurecom.senml.entity.RegisteredDeviceType; +import fr.eurecom.senml.entity.SensorAdmin; +import fr.eurecom.senml.persistence.JDOStorage; + +public class GCMPushingServer implements IPushingServer { + + private List<String> REG_IDS, ADMIN_REG_IDS; + // HARD-CODED, move on a different configuration file! + private String API_KEY = "AIzaSyDO3RS9FZPTp0d2BNn-0zKUuiiOI45-ED8"; + private boolean initialized = false; + + private static final GCMPushingServer instance = new GCMPushingServer(); + + protected GCMPushingServer() { + + } + + public static GCMPushingServer getInstance() { + if (!instance.isInitialized()) { + instance.initialize(); + } + return instance; + } + + public boolean isInitialized() { + return initialized; + } + + public void initialize() { + REG_IDS = ADMIN_REG_IDS = new ArrayList<String>(); + initialized = true; + } + + public void refreshDevicesList() { + REG_IDS = new ArrayList<String>(); + List<RegisteredDeviceType> list = JDOStorage.getInstance().getAll( + RegisteredDeviceType.class); + Iterator<RegisteredDeviceType> iter = list.iterator(); + while (iter.hasNext()) { + RegisteredDeviceType current = iter.next(); + if (isAndroidDevice(current)) { + if (current.isIsadmin()) { + if (ADMIN_REG_IDS.indexOf(current) == -1) { + System.out.println("Refreshing the list of admins...adding " + + current.getDeviceRegistrationKey()); + ADMIN_REG_IDS.add(current.getDeviceRegistrationKey()); + } + } else { + if (REG_IDS.indexOf(current) == -1) { + System.out.println("Refreshing the list...adding " + + current.getDeviceRegistrationKey()); + REG_IDS.add(current.getDeviceRegistrationKey()); + } + } + } + } + } + + /** + * Return true if the device is of type Android + * + * @param device + * @return + */ + private boolean isAndroidDevice(RegisteredDeviceType device) { + System.out.print("isAndroidDevice? "); + if (device.getDeviceType().toLowerCase().indexOf("android") > -1) { + System.out.println("Yes"); + } else { + System.out.println("No"); + } + return (device.getDeviceType().toLowerCase().indexOf("android") > -1); + } + + @Override + public void sendNotification(String data) throws JSONException { + // Refresh the devices list and put it on the JSON object + refreshDevicesList(); + System.out.println("Calling gcmnotify with data = " + data); + GCMnotify(data); + // TODO: Check that is part of monitored sensors + GCMnotify(data, true); + } + + private void GCMnotify(String data) { + + System.out.println("Sending data: " + data); + + // Instance of com.android.gcm.server.Sender, that does the + // transmission of a Message to the Google Cloud Messaging service. + Sender sender = new Sender(API_KEY); + + // This Message object will h old the data that is being transmitted + // to the Android client devices. For this demo, it is a simple text + // string, but could certainly be a JSON object. + Message message = new Message.Builder() + + // If multiple messages are sent using the same .collapseKey() + // the android target device, if it was offline during earlier + // message + // transmissions, will only receive the latest message for that + // key when + // it goes back on-line. + // .collapseKey(collapseKey) + .timeToLive(30).delayWhileIdle(true) + .addData("type", "zoneupdated") + .addData("notificationmessage", "WL-BOX, zone updated: " + data.replace("ZoneUpdated", "")) + .addData("alertstatus", "info") + .addData("message", data).build(); + + if (REG_IDS.size() == 0) { + // Noone to notify + return; + } + try { + // use this for multicast messages. The second parameter + // of sender.send() will need to be an array of register ids. + + MulticastResult result = sender.send(message, REG_IDS, 1); + + if (result.getResults() != null) { + int canonicalRegId = result.getCanonicalIds(); + if (canonicalRegId != 0) { + + } + } else { + int error = result.getFailure(); + System.out.println("Broadcast failure: " + error); + } + + } catch (Exception e) { + e.printStackTrace(); + } + } + + // TODO: Only one constructor, most of the code is the same... + private void GCMnotify(String data, boolean isadmin) { + System.out.println("ADMIN! Sending data: " + data); + + // Instance of com.android.gcm.server.Sender, that does the + // transmission of a Message to the Google Cloud Messaging service. + Sender sender = new Sender(API_KEY); + + // This Message object will h old the data that is being transmitted + // to the Android client devices. For this demo, it is a simple text + // string, but could certainly be a JSON object. + Message message = new Message.Builder() + + // If multiple messages are sent using the same .collapseKey() + // the android target device, if it was offline during earlier + // message + // transmissions, will only receive the latest message for that + // key when + // it goes back on-line. + // .collapseKey(collapseKey) + .timeToLive(30).delayWhileIdle(true) + .addData("type", "alert") + .addData("notificationmessage", "WL-BOX admin, SENSOR ALERT!") + .addData("alertstatus", "alert") + .addData("message", data).build(); + + if (ADMIN_REG_IDS.size() == 0) { + // Noone to notify + System.out.println("Admin IDS empty"); + return; + } + try { + // use this for multicast messages. The second parameter + // of sender.send() will need to be an array of register ids. + + MulticastResult result = sender.send(message, ADMIN_REG_IDS, 1); + + if (result.getResults() != null) { + System.out.println("!) null"); + int canonicalRegId = result.getCanonicalIds(); + if (canonicalRegId != 0) { + + } + } else { + int error = result.getFailure(); + System.out.println("Broadcast failure: " + error); + } + + } catch (Exception e) { + e.printStackTrace(); + } + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/IPushingServer.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/IPushingServer.java new file mode 100644 index 0000000000000000000000000000000000000000..29d2ef070278beaceacf8add87330004dd3159ac --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/pushing/IPushingServer.java @@ -0,0 +1,7 @@ +package fr.eurecom.pushing; + +import org.json.JSONException; + +public interface IPushingServer { + public void sendNotification(String data) throws JSONException; +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/Contact.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/Contact.java new file mode 100644 index 0000000000000000000000000000000000000000..95b66ab6781a62d09de9e7fe9794c6cbadb6a844 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/Contact.java @@ -0,0 +1,146 @@ +package fr.eurecom.restlet.resources.common; + +/** + * This class implements the Contact entity. + * + */ +public class Contact { + + // private static final long serialVersionUID = 7390103290165670089L; + /** + * Contact's identifier. + */ + private final int id; + /** + * Contact's first name + */ + private String firstName = "N/A"; + /** + * Contact's last name + */ + private String lastName = "N/A"; + /** + * Contact's email address + */ + private String mail = "N/A"; + /** + * Contact's phone number + */ + private String phone = "N/A"; + + /** + * Constructs a contact with the specified id. + * + * @param id + */ + public Contact(int id) { + this.id = id; + } + + /** + * Constructs a contact with the specified values. + * + * @param id + * Contact's identifier. + * @param firstname + * Contact's first name. + * @param lastname + * Contact's last name. + * @param mail + * Contact's email address. + * @param phone + * Contact's Phone number. + */ + public Contact(int id, String firstname, String lastname, String mail, + String phone) { + this.id = id; + this.firstName = firstname; + this.lastName = lastname; + this.mail = mail; + this.phone = phone; + } + + /** + * + * @return the contact id. + */ + public int getId() { + return id; + } + + /** + * Set an email address + * + * @param mail + * email address. + */ + public void setMail(String mail) { + this.mail = mail; + } + + /** + * Retrieve the email address. + * + * @return the mail address if set, "N/A" otherwise. + */ + public String getMail() { + return mail; + } + + /** + * Retrieve the first name. + * + * @return the first name if set, "N/A" otherwise. + */ + public String getFirstName() { + return firstName; + } + + /** + * Set a first name. + * + * @param firstname + * to set. + */ + public void setFirstName(String firstname) { + this.firstName = firstname; + } + + /** + * Retrieve the last name. + * + * @return the last name set, "N/A" otherwise. + */ + public String getLastName() { + return lastName; + } + + /** + * Set a last name. + * + * @param lastname + * to set, "N/A" otherwise. + */ + public void setLastName(String lastname) { + this.lastName = lastname; + } + + /** + * Retrieve the phone number. + * + * @return the phone set, "N/A" otherwise. + */ + public String getPhone() { + return phone; + } + + /** + * Set the phone number. + * + * @param phone + * number to set. + */ + public void setPhone(String phone) { + this.phone = phone; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactBaseResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactBaseResource.java new file mode 100644 index 0000000000000000000000000000000000000000..4f3b3c6c464326a62fedb87315f5b11271d0dcd1 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactBaseResource.java @@ -0,0 +1,173 @@ +package fr.eurecom.restlet.resources.common; + +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.logging.Level; +import java.util.logging.Logger; + +import org.restlet.data.Status; +import org.restlet.ext.wadl.WadlServerResource; + +import com.google.appengine.api.channel.ChannelMessage; +import com.google.appengine.api.channel.ChannelService; +import com.google.appengine.api.channel.ChannelServiceFactory; + +import fr.eurecom.restlet.resources.persistence.Storage; + +public abstract class ContactBaseResource extends WadlServerResource { + private final static String CONTACTS_UPDATED = "contactsupdated"; + private static final Logger log = Logger.getLogger("ContactBaseResource"); + + public enum MEMBERS { + CONTACT, ID, FIRSTNAME, LASTNAME, PHONE, MAIL + } + + protected static final String CONTACTS = "contacts"; + protected static final String CONTACT = "contact"; + protected static final String ID = "id"; + protected static final String FIRST_NAME = "firstname"; + protected static final String LAST_NAME = "lastname"; + protected static final String PHONE = "phone"; + protected static final String MAIL = "mail"; + + protected static final String URL = "url"; + protected static final String REQUEST_QUERY_SORT = "sort"; + +// HACK, this class is also used to support subscribers for senml objects. Move the subscribers channel API Stuffs to a common place. + protected static final Map<String, Token> subscribers = new HashMap<String, Token>(); + + final static class Token { + public String token; + public long timestamp; + + public Token(String token) { + this.token = token; + this.timestamp = System.currentTimeMillis(); + } + } + /** + * Retrieve the contacts from the persistence layer. + * + * @return the contact's list. + */ + protected List<Contact> getContacts() { + // Dispatcher d = (Dispatcher)getApplication(); + return Storage.getInstance().getContacts(MEMBERS.ID, 0, 999); + } + + /** + * Retrieve the contacts sorted by Last name from the persistence layer. + * + * @return a sorted contacts list. + */ + @SuppressWarnings("unchecked") + protected List<Contact> getSortedContacts(String sort) { + try { + MEMBERS mSorted = sort == null || sort.trim().isEmpty() ? MEMBERS.LASTNAME + : MEMBERS.valueOf(sort.trim().toUpperCase()); + + return Storage.getInstance().getContacts(mSorted, 0, 999); + } catch (Exception e) { + e.printStackTrace(); + return Collections.EMPTY_LIST; + } + } + + /** + * Retrieve the contact which matches this id from the persistence layer. + * + * @param id + * @return contact or null. + */ + protected Contact getContact(int id) { + return Storage.getInstance().getContact(id); + } + + /** + * Delete a contact from the persistence layer. + * + * @param id + * @return Status.SUCCESS_NO_CONTENT if deleted, + * Status.CLIENT_ERROR_NOT_FOUND otherwise. + */ + protected Status removeContact(int id) { + if (Storage.getInstance().removeContact(id)) { + sendUpdateSubscribers(CONTACTS_UPDATED); + return Status.SUCCESS_NO_CONTENT; + } + else { + return Status.CLIENT_ERROR_NOT_FOUND; + } + } + + /** + * Save a new contact to the persistence layer. (restricted to 50 contacts + * currently). + * + * @param contact + * @return Status.SUCCESS_CREATED if saved, + * Status.SERVER_ERROR_INSUFFICIENT_STORAGE otherwise. + */ + protected Status addContact(Contact contact) { + if (Storage.getInstance().saveContact(contact)) { + sendUpdateSubscribers(CONTACTS_UPDATED); + return Status.SUCCESS_CREATED; + } + return Status.SERVER_ERROR_INSUFFICIENT_STORAGE; + } + + /** + * Update a contact to the persistence layer. + * + * @param contact + * @return Status.SUCCESS_OK if updated, Status.CLIENT_ERROR_NOT_FOUND + * otherwise. + */ + protected Status updateContact(Contact contact) { + if (Storage.getInstance().saveContact(contact)) { + sendUpdateSubscribers(CONTACTS_UPDATED); + return Status.SUCCESS_NO_CONTENT; + } else { + return Status.CLIENT_ERROR_NOT_FOUND; + } + } + + + public static void addSubscriber(String clientID, String token) { + subscribers.put(clientID, new Token(token)); + } + + public static void deleteSubscriber(String clientID) { + subscribers.remove(clientID); + } + + public static String getToken(String clientID) { + Token t = subscribers.get(clientID); + if (t != null && (System.currentTimeMillis() - t.timestamp) < 7200000) { + return t.token; + } + else { + return null; + } + } + + public static void sendUpdateSubscribers(String message) { + ChannelService service = ChannelServiceFactory.getChannelService(); + try { + service.sendMessage(new ChannelMessage("I235", message)); + log.info("Notified I235" + message); + } + catch (Exception e) { + log.log(Level.WARNING, "Error Notifying client I235", e); + System.out.println("Error notifying client I235"); + } + +/* for (String clientID : subscribers.keySet()) { + System.out.println("Will notify " + clientID + " " + message); + service.sendMessage(new ChannelMessage(clientID, message)); + } + */ + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactResource.java new file mode 100644 index 0000000000000000000000000000000000000000..ae313671bb63bc6dfb25cb5696db0475f350bb75 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactResource.java @@ -0,0 +1,242 @@ +package fr.eurecom.restlet.resources.common; + +import java.util.Map; + +import org.json.JSONObject; +import org.restlet.data.Form; +import org.restlet.data.MediaType; +import org.restlet.data.Status; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.ext.wadl.ParameterInfo; +import org.restlet.ext.wadl.ParameterStyle; +import org.restlet.ext.wadl.RepresentationInfo; +import org.restlet.ext.wadl.RequestInfo; +import org.restlet.ext.wadl.ResponseInfo; +import org.restlet.ext.xml.DomRepresentation; +import org.restlet.representation.Representation; +import org.restlet.resource.Delete; +import org.restlet.resource.Get; +import org.restlet.resource.Put; +import org.restlet.resource.ResourceException; +import org.w3c.dom.Document; +import org.w3c.dom.Element; + +/** + * + * This class represents the Contact resource + */ + +public class ContactResource extends ContactBaseResource { + + /** + * underlying contact. + */ + private Contact contact = null; + + /** + * Resource contact identifier. + */ + private int identifier = -1; + + @Override + public Representation describe() { + System.out.println("describe"); + setName("Contact Resource"); + setDescription("Manage the current Contact"); + return super.describe(); + } + + @Override + public void describeDelete(MethodInfo mInfo) { + mInfo.setIdentifier("ContactDelete"); + mInfo.setDocumentation("To Delete the current contact."); + + ResponseInfo response = new ResponseInfo( + "Contact deleted. No contents returned."); + response.getStatuses().add(Status.SUCCESS_NO_CONTENT); + mInfo.getResponses().add(response); + + response = new ResponseInfo("Contact not found"); + response.getStatuses().add(Status.CLIENT_ERROR_NOT_FOUND); + mInfo.getResponses().add(response); + } + + @Override + public void describeGet(MethodInfo mInfo) { + mInfo.setIdentifier("ContactDetail"); + mInfo.setDocumentation("To retrieve details of the current contact."); + + ResponseInfo response = new ResponseInfo("Current contact details"); + response.getStatuses().add(Status.SUCCESS_OK); + + RepresentationInfo repInfo = new RepresentationInfo( + MediaType.APPLICATION_XML); + repInfo.setXmlElement(CONTACT); + repInfo.setDocumentation("XML representation of the current contact."); + response.getRepresentations().add(repInfo); + + repInfo = new RepresentationInfo(MediaType.APPLICATION_JSON); + repInfo.setDocumentation("JSON representation of the current contact"); + repInfo.setXmlElement(CONTACT); + response.getRepresentations().add(repInfo); + mInfo.getResponses().add(response); + + response = new ResponseInfo("Contact not found"); + repInfo = new RepresentationInfo(MediaType.TEXT_HTML); + repInfo.setIdentifier("contactError"); + response.getStatuses().add(Status.CLIENT_ERROR_NOT_FOUND); + response.getRepresentations().add(repInfo); + mInfo.getResponses().add(response); + } + + @Override + public void describePut(MethodInfo mInfo) { + mInfo.setIdentifier("ContactUpdate"); + mInfo.setDocumentation("To update details of the current contact"); + + RequestInfo requestInfo = new RequestInfo(); + // requestInfo.getParameters().add( + // new ParameterInfo(ID, true, "Number", ParameterStyle.QUERY, + // "Contact identifier")); + requestInfo.getParameters().add( + new ParameterInfo(FIRST_NAME, false, "String", + ParameterStyle.QUERY, "Contact First Name")); + requestInfo.getParameters().add( + new ParameterInfo(LAST_NAME, false, "String", + ParameterStyle.QUERY, "Contact Last Name")); + requestInfo.getParameters().add( + new ParameterInfo(PHONE, false, "String", ParameterStyle.QUERY, + "Contact Phone number")); + requestInfo.getParameters().add( + new ParameterInfo(MAIL, false, "String", ParameterStyle.QUERY, + "Contact email address")); + + mInfo.setRequest(requestInfo); + + ResponseInfo responseInfo = new ResponseInfo("Contact not Found"); + responseInfo.getStatuses().add(Status.CLIENT_ERROR_NOT_FOUND); + mInfo.getResponses().add(responseInfo); + + responseInfo = new ResponseInfo( + "Contact updated, no contents returned."); + responseInfo.getStatuses().add(Status.SUCCESS_NO_CONTENT); + mInfo.getResponses().add(responseInfo); + } + + @Override + protected void doInit() throws ResourceException { + String sIdentifier = (String) getRequest().getAttributes().get("id"); + if (sIdentifier != null) { + identifier = Integer.parseInt(sIdentifier); + contact = getContact(identifier); + } + setExisting(contact != null); + } + + /** + * Handle HTTP GET method / xml + * + * @return a xml resource representation + */ + @Get("xml") + public Representation toXML() { + try { + DomRepresentation representation = new DomRepresentation( + MediaType.TEXT_XML); + + Document d = representation.getDocument(); + Element eltItem = d.createElement(CONTACT); + d.appendChild(eltItem); + + Element el = d.createElement(ID); + el.appendChild(d.createTextNode("" + contact.getId())); + eltItem.appendChild(el); + + el = d.createElement(FIRST_NAME); + el.appendChild(d.createTextNode(contact.getFirstName())); + eltItem.appendChild(el); + + el = d.createElement(LAST_NAME); + el.appendChild(d.createTextNode(contact.getLastName())); + eltItem.appendChild(el); + + el = d.createElement(PHONE); + el.appendChild(d.createTextNode(contact.getPhone())); + eltItem.appendChild(el); + + el = d.createElement(MAIL); + el.appendChild(d.createTextNode(contact.getMail())); + eltItem.appendChild(el); + + d.normalizeDocument(); + + return representation; + } catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + /** + * Handle HTTP GET method / json + * + * @return a json resource representation. + */ + @Get("json") + public Representation toJSON() { + JSONObject jObject = new JSONObject(); + try { + jObject.put(ID, "" + contact.getId()); + jObject.put(FIRST_NAME, contact.getFirstName()); + jObject.put(LAST_NAME, contact.getLastName()); + jObject.put(PHONE, contact.getPhone()); + jObject.put(MAIL, contact.getMail()); + + return new JsonRepresentation(jObject); + } catch (Exception e) { + return null; + } + } + + /** + * Handle HTTP PUT method. Update an existing contact. + * + * @param entity + */ + @Put + public void updateContact(Representation entity) { + Map<String, String> map = new Form(entity).getValuesMap(); + + for (String k : map.keySet()) { + MEMBERS m = MEMBERS.valueOf(k.toUpperCase()); + + switch (m) { + case FIRSTNAME: + contact.setFirstName(map.get(k)); + break; + case LASTNAME: + contact.setLastName(map.get(k)); + break; + case PHONE: + contact.setPhone(map.get(k)); + break; + case MAIL: + contact.setMail(map.get(k)); + break; + } + } + + Status updateStatus = super.updateContact(contact); + setStatus(updateStatus); + } + + /** + * Handle HTTP DELETE method. Delete an existing contact. + */ + @Delete + public void deleteContact() { + Status deleteStatus = super.removeContact(identifier); + setStatus(deleteStatus); + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactsResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactsResource.java new file mode 100644 index 0000000000000000000000000000000000000000..ce9426c977a4d40ac225a3a062acfbdae47b8e3f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactsResource.java @@ -0,0 +1,335 @@ +package fr.eurecom.restlet.resources.common; + +import java.util.Iterator; + +import org.json.JSONArray; +import org.json.JSONObject; +import org.restlet.Context; +import org.restlet.Request; +import org.restlet.Response; +import org.restlet.data.Form; +import org.restlet.data.MediaType; +import org.restlet.data.Method; +import org.restlet.data.Reference; +import org.restlet.data.Status; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.ext.wadl.OptionInfo; +import org.restlet.ext.wadl.ParameterInfo; +import org.restlet.ext.wadl.ParameterStyle; +import org.restlet.ext.wadl.RepresentationInfo; +import org.restlet.ext.wadl.RequestInfo; +import org.restlet.ext.wadl.ResponseInfo; +import org.restlet.ext.xml.DomRepresentation; +import org.restlet.representation.Representation; +import org.restlet.representation.StringRepresentation; +import org.restlet.resource.Get; +import org.restlet.resource.Post; +import org.w3c.dom.Document; +import org.w3c.dom.Element; + +public class ContactsResource extends ContactBaseResource { + + /** + * {@inheritDoc} + */ + @Override + public Representation describe() { + setName("Contacts Resource"); + setDescription("Retrieve contacts list"); + return super.describe(); + } + + /** + * {@inheritDoc} + */ + @Override + public void describeGet(MethodInfo mInfo) { + mInfo.setIdentifier("ContactsList"); + mInfo.setDocumentation("To Retrieve a sorted list of contacts"); + + RequestInfo requestInfo = new RequestInfo(); + ParameterInfo pInfo = createCommonParameter(REQUEST_QUERY_SORT, + "Sort query", LAST_NAME); + pInfo.getOptions().add( + createOptionInfo(FIRST_NAME, "Sort by first name")); + pInfo.getOptions() + .add(createOptionInfo(LAST_NAME, "Sort by last name")); + pInfo.getOptions().add(createOptionInfo(MAIL, "Sort by Email")); + pInfo.getOptions().add(createOptionInfo(PHONE, "Sort by Phone")); + + requestInfo.getParameters().add(pInfo); + mInfo.setRequest(requestInfo); + + ResponseInfo response = new ResponseInfo("Current list of contacts"); + response.getStatuses().add(Status.SUCCESS_OK); + + RepresentationInfo repInfo = new RepresentationInfo( + MediaType.APPLICATION_XML); + repInfo.setXmlElement(CONTACTS); + repInfo.setDocumentation("XML List of contacts"); + response.getRepresentations().add(repInfo); + + repInfo = new RepresentationInfo(MediaType.APPLICATION_JSON); + repInfo.setXmlElement(CONTACTS); + repInfo.setDocumentation("JSON List of contacts"); + response.getRepresentations().add(repInfo); + + mInfo.getResponses().add(response); + } + + /** + * {@inheritDoc} + */ + @Override + public void describePost(MethodInfo mInfo) { + mInfo.setIdentifier("ContactsAdd"); + mInfo.setIdentifier("To add a new contact to the list"); + + RequestInfo requestInfo = new RequestInfo(); + requestInfo.getParameters().add( + new ParameterInfo(ID, true, "Number", ParameterStyle.QUERY, + "Contact identifier")); + requestInfo.getParameters().add( + createCommonParameter(FIRST_NAME, "Contact First Name")); + requestInfo.getParameters().add( + createCommonParameter(LAST_NAME, "Contact Last Name")); + requestInfo.getParameters().add( + createCommonParameter(PHONE, "Contact Phone Number")); + requestInfo.getParameters().add( + createCommonParameter(MAIL, "Contact Email address")); + + mInfo.setRequest(requestInfo); + + ResponseInfo response = new ResponseInfo("Contact saved"); + response.getStatuses().add(Status.SUCCESS_CREATED); + response.getParameters().add( + new ParameterInfo("Content-Location", ParameterStyle.HEADER, + "URI contact saved")); + mInfo.getResponses().add(response); + + response = new ResponseInfo("Contact not saved. (restricted to 50)"); + response.getStatuses().add(Status.SERVER_ERROR_INSUFFICIENT_STORAGE); + mInfo.getResponses().add(response); + } + + // HACK: Version Restlet-gae 2.1-rc5. + // There is an error routing the GET request when + // the Request Header Content-Type = application/x-www-form-urlencoded and + // Header Accept = "application/json" and + // URI have a query : http://localhost:8888/contacts?sort=firstname. + // In this case the router ignores the Accept header and route the request + // to @Get("xml") systematically because the query is badly interpreted. + // To workaround this problem, at the init we reformat the Entity with an + // empty Representation. + + @Override + public void init(Context context, Request request, Response response) { + // System.out.println("init" + request + " " + response); + // System.out.println("Content-Type: " + + // request.getEntity().getMediaType()); + if (request.getMethod().compareTo(Method.GET) == 0) { + Representation entity = request.getEntity(); + System.out.println("Entity " + entity + " Content-Type: " + + request.getEntity().getMediaType()); + request.setEntity(new StringRepresentation("")); + } + super.init(context, request, response); + } + + /** + * Handle HTTP GET Metod / xml + * + * @return an XML list of contacts. + */ + @Get("xml") + public Representation toXML() { + // System.out.println("Request Original Ref " + getOriginalRef()); + // System.out.println("Request Entity " + getRequest().getEntityAsText() + // + + // " entity mediaType " + getRequest().getEntity().getMediaType()); + + Reference ref = getRequest().getResourceRef(); + final String baseURL = ref.getHierarchicalPart(); + Form formQuery = ref.getQueryAsForm(); + + try { + DomRepresentation representation = new DomRepresentation( + MediaType.TEXT_XML); + + Document d = representation.getDocument(); + Element elContacts = d.createElement(CONTACTS); + d.appendChild(elContacts); + + Iterator<Contact> it = getSortedContacts( + formQuery.getFirstValue(REQUEST_QUERY_SORT, LAST_NAME)) + .iterator(); + while (it.hasNext()) { + Contact contact = it.next(); + + Element el = d.createElement(CONTACT); + + Element id = d.createElement(ID); + id.appendChild(d.createTextNode(String.format("%s", + contact.getId()))); + el.appendChild(id); + + Element firstname = d.createElement(FIRST_NAME); + firstname.appendChild(d.createTextNode(contact.getFirstName())); + el.appendChild(firstname); + + Element lastname = d.createElement(LAST_NAME); + lastname.appendChild(d.createTextNode(contact.getLastName())); + el.appendChild(lastname); + + Element url = d.createElement(URL); + url.appendChild(d.createTextNode(String.format("%s/%s", + baseURL, contact.getId()))); + el.appendChild(url); + + elContacts.appendChild(el); + } + + d.normalizeDocument(); + return representation; + } catch (Exception e) { + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } + + /** + * Handle HTTP GET method / Json + * + * @return a JSON list of contacts. + */ + @Get("json") + public Representation toJSON() { + // System.out.println("Request Original Ref " + getOriginalRef()); + // System.out.println("Request Entity " + getRequest().getEntityAsText() + // + + // " entity mediaType " + getRequest().getEntity().getMediaType()); + + try { + JSONArray jcontacts = new JSONArray(); + Reference ref = getRequest().getResourceRef(); + final String baseURL = ref.getHierarchicalPart(); + Form formQuery = ref.getQueryAsForm(); + + Iterator<Contact> it = getSortedContacts( + formQuery.getFirstValue(REQUEST_QUERY_SORT, LAST_NAME)) + .iterator(); + + while (it.hasNext()) { + Contact contact = it.next(); + + JSONObject jcontact = new JSONObject(); + jcontact.put(ID, String.format("%s", contact.getId())); + jcontact.put(URL, String.format("%s/%s", baseURL, contact.getId())); + jcontact.put(FIRST_NAME, contact.getFirstName()); + jcontact.put(LAST_NAME, contact.getLastName()); + jcontacts.put(jcontact); + } + + JSONObject contacts = new JSONObject(); + contacts.put(CONTACTS, jcontacts); + return new JsonRepresentation(contacts); + } catch (Exception e) { + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } + + /** + * Handle HTTP POST Method. + * + * @param entity + * contact to create. + * @return The new contact URI representation (Header Content-Location). + */ + @Post + public Representation createContact(Representation entity) { + Form form = new Form(entity); + + try { + int id = Integer.parseInt(form.getFirstValue(ID, "-1")); + if (id == -1) { + System.out.println("id = -1"); + setStatus(Status.CLIENT_ERROR_EXPECTATION_FAILED, + "Must provide 'id' field."); + return null; + } + + Contact contact = getContact(id) == null ? new Contact(id) + : getContact(id); + contact.setFirstName(form.getFirstValue(FIRST_NAME, "N/A")); + contact.setLastName(form.getFirstValue(LAST_NAME, "N/A")); + contact.setPhone(form.getFirstValue(PHONE, "N/A")); + contact.setMail(form.getFirstValue(MAIL, "N/A")); + + Status createStatus = super.addContact(contact); + setStatus(createStatus); + + Representation rep = new StringRepresentation("Contact created", + MediaType.TEXT_PLAIN); + rep.setLocationRef(String.format("%s/%s", getRequest() + .getResourceRef().getIdentifier(), id)); + return rep; + } catch (Exception e) { + e.printStackTrace(); + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } + + /** + * Helper Method creating common form of ParameterInfo + * + * @param name + * ParameterInfo name + * @param description + * ParameterInfo description + * @return ParameterInfo formatted : required=false, parameterStyle=Query, + * defaultValue="N/A", type="String". + */ + private static ParameterInfo createCommonParameter(String name, + String description) { + return createCommonParameter(name, description, "N/A"); + } + + /** + * Helper Method creating common form of ParameterInfo + * + * @param name + * - ParameterInfo name + * @param description + * - ParameterInfo description + * @param defaultValue + * - ParameterInfo defaultValue. + * @return ParameterInfo formatted : required=false, parameterStyle=Query, + * type="String". + */ + private static ParameterInfo createCommonParameter(String name, + String description, String defaultValue) { + ParameterInfo pi = new ParameterInfo(name, false, "String", + ParameterStyle.QUERY, description); + pi.setDefaultValue(defaultValue); + return pi; + + } + + /** + * Helper method creating an OptionInfo + * + * @param value + * Option's value + * @param description + * - Option's description + * @return an OptionInfo with the specified values. + */ + private static OptionInfo createOptionInfo(String value, String description) { + OptionInfo oi = new OptionInfo(description); + oi.setValue(value); + return oi; + } +} \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactsSubResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactsSubResource.java new file mode 100644 index 0000000000000000000000000000000000000000..e59b890131162b77c2370fe78e230284f84c7706 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ContactsSubResource.java @@ -0,0 +1,69 @@ +package fr.eurecom.restlet.resources.common; + +import java.util.logging.Level; +import java.util.logging.Logger; + +import org.json.JSONObject; +import org.restlet.data.Form; +import org.restlet.data.Reference; +import org.restlet.data.Status; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.representation.Representation; +import org.restlet.resource.Get; + +import com.google.appengine.api.channel.ChannelServiceFactory; + +public class ContactsSubResource extends ContactBaseResource { + private static final Logger log = Logger.getLogger("ContactSubResource"); + + /** + * {@inheritDoc} + */ + @Override + public void describePost(MethodInfo mInfo) { + } + + + + /** + * Handle HTTP GET Subscribe method + * + * @param entity - must contains the client id channel. + * @return a JSON Representation of the token. + */ + @Get + public Representation subscribe(Representation entity) { + Reference ref = getRequest().getResourceRef(); + Form formQuery = ref.getQueryAsForm(); + JSONObject json = new JSONObject(); + + try { + String clientID = formQuery.getFirstValue("clientID", "").trim(); + if (clientID.isEmpty()) { + log.warning("client ID not found"); + setStatus(Status.CLIENT_ERROR_EXPECTATION_FAILED, + "Must provide a clientID field."); + return null; + } + String token = getToken(clientID); + if (token == null) { + token = ChannelServiceFactory.getChannelService().createChannel(clientID); + log.info("Create a new token [" + token + "]"); + addSubscriber(clientID, token); + } + else { + log.info("Token already existing for client " + clientID); + } + + json.append("token", token); + return new JsonRepresentation(json); + + + } catch (Exception e) { + e.printStackTrace(); + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } +} \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/SensorInResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/SensorInResource.java new file mode 100644 index 0000000000000000000000000000000000000000..629132ee246dd03841d68129f19dd3763456c936 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/SensorInResource.java @@ -0,0 +1,149 @@ +package fr.eurecom.restlet.resources.common; + +import java.util.logging.Level; +import java.util.logging.Logger; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; +import org.restlet.data.MediaType; +import org.restlet.data.Status; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.ext.wadl.RepresentationInfo; +import org.restlet.ext.wadl.ResponseInfo; +import org.restlet.ext.wadl.WadlServerResource; +import org.restlet.representation.Representation; +import org.restlet.resource.Get; +import org.restlet.resource.Delete; +import org.restlet.resource.Put; +import org.restlet.resource.Post; + +import fr.eurecom.senml.entity.*; +import fr.eurecom.senml.persistence.JDOStorage; + + +public class SensorInResource extends WadlServerResource { + private static final Logger log = Logger.getLogger("SensorInResource"); + private static final MediaType SENML_JSON = MediaType.valueOf("application/senml+json"); + /** + * {@inheritDoc} + */ + @Override + public Representation describe() { + setName("Sensor-In Resource"); + setDescription("Acquisition sensors receiver"); + return super.describe(); + } + + @Post + @Put + public void updateSensor(Representation entity) { + JSONObject jObject = null; + String text = null; + try { + MediaType mt = entity.getMediaType(); + text = entity.getText(); + if (mt.isCompatible(MediaType.APPLICATION_JSON) || mt.isCompatible(SENML_JSON) ) { + jObject = new JSONObject(text); + } + else { + setStatus(Status.CLIENT_ERROR_NOT_ACCEPTABLE, "Content is not JSON compatible"); + log.info("This entry cause a 406 response / reason not json"); + return; + } + + String bn = jObject.has(ISensor.BASE_NAME) ? jObject.getString(ISensor.BASE_NAME) : null; + long bt = jObject.has(ISensor.BASE_TIME) ? jObject.getLong(ISensor.BASE_TIME) : Long.MIN_VALUE; + String bu = jObject.has(ISensor.BASE_UNIT) ? jObject.getString(ISensor.BASE_UNIT) : null; + + if (! jObject.has(ISensor.MEASUREMENT) || bn == null) { + setStatus(Status.CLIENT_ERROR_BAD_REQUEST, "Parameter Base Name[bn] or measure[e] missing."); + log.info("This entry cause a 400 response. reason bn - e missing ||" + text + "||"); + return; + } + JSONArray measures = jObject.getJSONArray(ISensor.MEASUREMENT); + +// TODO: Remove this hack: zone key is static. For first tests with IQSIM and demo. + IZone zone = JDOStorage.getInstance().getById("Aix", ZoneAdmin.class); + SensorAdmin sa = new SensorAdmin(bn, "Generic Sensor", zone); + boolean parsingOK = true; + String reasonPhrase = null; + + for (int i = 0; i < measures.length(); i++) { + + JSONObject m = measures.getJSONObject(i); + + Measure.ParamTypeValue tv = getUnitType(m); + if (tv == null) { + parsingOK = false; + reasonPhrase = "Measure miss type value [v|sv|bv]"; + break; + } + + Units unit = getUnit(m, bu); + if (unit == null) { + parsingOK = false; + reasonPhrase = "parameter unit [bu|u] invalid (missing or unknown)"; + break; + } + + String name = m.has(Measure.PARAM_NAME) ? m.getString(Measure.PARAM_NAME) : null; + if ( name == null || name.trim().isEmpty()) { + parsingOK = false; + reasonPhrase = "parameter name [n] missing or empty. Not supported"; + break; + } + + String type = m.has(Measure.PARAM_TYPE) ? m.getString(Measure.PARAM_TYPE) : null; + if ( type == null || type.trim().isEmpty()) { + parsingOK = false; + reasonPhrase = "parameter type [n] missing or empty. Not supported"; + break; + } + + + sa.addMeasure(name.trim(), unit, tv, + m.getString(tv.getSenML()), + m.has(Measure.PARAM_TIME) ? m.getLong(Measure.PARAM_TIME) : + bt == Long.MIN_VALUE ? 0 : bt, type.trim()); + } + + if (!parsingOK) { + setStatus(Status.CLIENT_ERROR_BAD_REQUEST, reasonPhrase); + log.info("This entry cause a 400 response / reason " + reasonPhrase + " ||" + text + "||"); + } + else { + setStatus(Status.SUCCESS_CREATED); + } + } + catch (Exception e) { + log.warning("Unexpected error receiving this entry : ||" + text + "||"); + log.log(Level.WARNING, "Unexpected Error", e); + + setStatus(Status.SERVER_ERROR_INTERNAL, "Unexpected Error"); + } + return; + } + + private static Measure.ParamTypeValue getUnitType(JSONObject m) { + if (m.has(Measure.PARAM_VALUE)) { + return Measure.ParamTypeValue.FLOAT; + } + else if (m.has(Measure.PARAM_BOOLEAN_VALUE)) { + return Measure.ParamTypeValue.BOOL; + } + else if (m.has(Measure.PARAM_STRING_VALUE)) { + return Measure.ParamTypeValue.STRING; + } + else if (m.has(Measure.PARAM_SUM_VALUE)) { + return Measure.ParamTypeValue.SUM; + } + return null; + } + + private static Units getUnit(JSONObject m, String bu) throws JSONException { + String u = m.has(Measure.PARAM_UNIT) ? m.getString(Measure.PARAM_UNIT) : bu != null ? bu : null; + return u == null ? null : Units.getUnit(u); + } +} \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/SensorPost.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/SensorPost.java new file mode 100644 index 0000000000000000000000000000000000000000..583652f797df64bf5a37769423f09ff419f4fe79 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/SensorPost.java @@ -0,0 +1,278 @@ +package fr.eurecom.restlet.resources.common; + +import java.math.BigInteger; +import java.security.MessageDigest; +import java.util.Date; +import java.util.List; +import java.util.logging.Level; +import java.util.logging.Logger; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; +import org.restlet.data.MediaType; +import org.restlet.data.Reference; +import org.restlet.data.Status; +import org.restlet.data.Tag; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.ext.wadl.RepresentationInfo; +import org.restlet.ext.wadl.ResponseInfo; +import org.restlet.ext.wadl.WadlServerResource; +import org.restlet.ext.xml.DomRepresentation; +import org.restlet.representation.Representation; +import org.restlet.resource.Get; +import org.restlet.resource.Post; +import org.w3c.dom.Document; +import org.w3c.dom.Element; +import org.w3c.dom.Node; + +import fr.eurecom.restlet.resources.common.ZonesResource.ZoneFormat; +import fr.eurecom.senml.entity.ISensor; +import fr.eurecom.senml.entity.IZone; +import fr.eurecom.senml.entity.PostValAdmin; +import fr.eurecom.senml.entity.ZoneAdmin; +import fr.eurecom.senml.persistence.JDOStorage; + +public class SensorPost extends WadlServerResource { + private static final Logger log = Logger.getLogger("SensorPost"); + + /** + * {@inheritDoc} + */ + @Override + public Representation describe() { + setName("Sensor Post"); + setDescription("Post Something, is a test!"); + return super.describe(); + } + + @Post("json") + public Representation doPost(String entity) { + String senmlValue = (String) getRequest().getAttributes().get("senml"); + //IZone zone = JDOStorage.getInstance().getById(id, ZoneAdmin.class); + JSONObject jzone = new JSONObject(); + try { + jzone.put("senml", senmlValue); + } catch (JSONException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + Date n = new Date(); + PostValAdmin sa = new PostValAdmin(); + sa.addPost("testing post " + String.valueOf(n), + entity, + "NOW".equals(n) ? 0 : System.currentTimeMillis()); + + Representation out = new JsonRepresentation(jzone); + return out; + + } + + /** + * {@inheritDoc} + */ + @Override + public void describeGet(MethodInfo mInfo) { + mInfo.setIdentifier("Sensor Posting"); + mInfo.setDocumentation("To Retrieve a zone description"); + ResponseInfo response = new ResponseInfo("Current list of zones"); + response.getStatuses().add(Status.SUCCESS_OK); + + RepresentationInfo repInfo = new RepresentationInfo( + MediaType.APPLICATION_JSON); + repInfo.setDocumentation("JSON array of of the current list zone"); + response.getRepresentations().add(repInfo); + + mInfo.getResponses().add(response); + } + + /** + * Handle HTTP GET method / Json + * + * @return a JSON Zone Representation + */ + @Get("json") + public Representation toJSON() { + + final String baseURL = buildBaseURL(getRequest().getResourceRef()); + + String id = (String) getRequest().getAttributes().get("id"); + IZone zone = JDOStorage.getInstance().getById(id, ZoneAdmin.class); + JSONObject jzone = new JSONObject(); + + try { + + jzone.put(ZoneFormat.NAME.getRestAtr(), zone.getName()); + + JSONArray subZones = new JSONArray(); + + for (IZone sub : zone.getSubZones()) { + JSONObject z = new JSONObject(); + z.put(ZoneFormat.URI.getRestAtr(), + String.format("%s/%s", baseURL, sub.getKey())); + z.put(ZoneFormat.NAME.getRestAtr(), sub.getName()); + z.put(ZoneFormat.HAS_SENSORS.getRestAtr(), sub.getSensors() + .size() > 0); + + z.put(ZoneFormat.HAS_SUBZONES.getRestAtr(), sub.getSubZones() + .size() > 0); + + subZones.put(z); + } + if (subZones.length() > 0) { + jzone.put(ZoneFormat.SUBZONE.getRestAtr(), subZones); + } + + JSONArray sensors = new JSONArray(); + for (ISensor sensor : zone.getSensors()) { + sensors.put(sensor.toJSONSenML()); + } + if (sensors.length() > 0) { + jzone.put(ZoneFormat.SENSORS_SENML.getRestAtr(), sensors); + } + +/* String md5 = md5(jzone.toString()); + Tag tag = md5 == null ? null : new Tag(md5, false); + if (!isModified(tag)) { + setStatus(Status.REDIRECTION_NOT_MODIFIED); + return null; + } +*/ + Representation json = new JsonRepresentation(jzone); +// json.setTag(tag); + return json; + } catch (Exception e) { + setStatus(Status.SERVER_ERROR_INTERNAL); + log.log(Level.WARNING, "", e); + return null; + } + } + + /** + * HTTP GET method / xml accept + * + * @return a XML Zone Representation + */ + + @Get("xml") + public Representation toXML() { + long before = System.currentTimeMillis(); + String id = (String) getRequest().getAttributes().get("id"); + IZone zone = JDOStorage.getInstance().getById(id, ZoneAdmin.class); + final String baseURL = buildBaseURL(getRequest().getResourceRef()); + + try { + DomRepresentation xml = new DomRepresentation(MediaType.TEXT_XML); + Document d = xml.getDocument(); + + Element currentZone = d.createElement(ZoneFormat.ZONE.getRestAtr()); + currentZone.setAttribute(ZoneFormat.NAME.getRestAtr(), + zone.getName()); + d.appendChild(currentZone); + + for (IZone subzone : zone.getSubZones()) { + Element sub = d.createElement(ZoneFormat.SUBZONE.getRestAtr()); + sub.setAttribute(ZoneFormat.NAME.getRestAtr(), + subzone.getName()); + sub.setAttribute(ZoneFormat.URI.getRestAtr(), + String.format("%s/%s", baseURL, subzone.getKey())); + sub.setAttribute(ZoneFormat.HAS_SENSORS.getRestAtr(), "" + + (subzone.getSensors().size() > 0)); + + sub.setAttribute(ZoneFormat.HAS_SUBZONES.getRestAtr(), "" + + (subzone.getSubZones().size() > 0)); + + currentZone.appendChild(sub); + + } + + for (ISensor sensor : zone.getSensors()) { + Node senml = sensor.toXMLSenML(d); + currentZone.appendChild(senml); + } + +/* String md5 = md5(xml.getText()); + Tag tag = md5 == null ? null : new Tag(md5, false); + if (!isModified(tag)) { + setStatus(Status.REDIRECTION_NOT_MODIFIED); + return null; + } +*/ +// xml.setTag(tag); + return xml; + } catch (Exception e) { + log.log(Level.WARNING, "", e); + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } + + /* + * Alternative method to add an header to the response. Series<Header> + * responseHeaders = (Series<Header>) + * getResponse().getAttributes().get(HeaderConstants.ATTRIBUTE_HEADERS); if + * (responseHeaders == null) { responseHeaders = new + * Series<Header>(Header.class); getResponse().getAttributes().put( + * HeaderConstants.ATTRIBUTE_HEADERS, responseHeaders); + * + * } responseHeaders.add(new Header("ETag", md5)); + */ + + /** + * Build a String baseURL = to current URI/Reference less the last path + * segment. + * + * @param reference + * the current reference object + * @return String base URI. + */ + private String buildBaseURL(Reference reference) { + Reference ref = new Reference(reference); + List<String> segments = ref.getSegments(); + if (segments.size() > 0) { + segments.remove(segments.size() - 1); + ref.setSegments(segments); + } + return ref.getHierarchicalPart(); + } + + private String md5(String value) { + try { + MessageDigest m = MessageDigest.getInstance("MD5"); + m.reset(); + m.update(value.getBytes()); + byte[] digest = m.digest(); + BigInteger bigInt = new BigInteger(1, digest); + String hashtext = bigInt.toString(16); + // Now we need to zero pad it if you actually want the full 32 + // chars. + while (hashtext.length() < 32) { + hashtext = "0" + hashtext; + } + return hashtext; + } catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + private boolean isModified(Tag currentTag) { + boolean modified = true; + if (currentTag == null) { + return modified; + } + + if (getRequest().getConditions() != null + && getRequest().getConditions().getNoneMatch() != null) { + + for (Tag noMatch : getRequest().getConditions().getNoneMatch()) { + if (currentTag.equals(noMatch)) { + modified = false; + break; + } + } + } + return modified; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ZoneResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ZoneResource.java new file mode 100644 index 0000000000000000000000000000000000000000..c4ae2f4aafe590db346059b5011a74e8064430a3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ZoneResource.java @@ -0,0 +1,255 @@ +package fr.eurecom.restlet.resources.common; + +import java.math.BigInteger; +import java.security.MessageDigest; +import java.util.List; +import java.util.logging.Level; +import java.util.logging.Logger; + +import org.json.JSONArray; +import org.json.JSONObject; +import org.restlet.data.MediaType; +import org.restlet.data.Reference; +import org.restlet.data.Status; +import org.restlet.data.Tag; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.ext.wadl.RepresentationInfo; +import org.restlet.ext.wadl.ResponseInfo; +import org.restlet.ext.wadl.WadlServerResource; +import org.restlet.ext.xml.DomRepresentation; +import org.restlet.representation.Representation; +import org.restlet.resource.Get; +import org.w3c.dom.Document; +import org.w3c.dom.Element; +import org.w3c.dom.Node; + +import fr.eurecom.restlet.resources.common.ZonesResource.ZoneFormat; +import fr.eurecom.senml.entity.ISensor; +import fr.eurecom.senml.entity.IZone; +import fr.eurecom.senml.entity.ZoneAdmin; +import fr.eurecom.senml.persistence.JDOStorage; + +public class ZoneResource extends WadlServerResource { +// Map<String, Tag> etags = new HashMap<String, Tag>(); + private static final Logger log = Logger.getLogger("ZoneResource"); + + /** + * {@inheritDoc} + */ + @Override + public Representation describe() { + setName("Zone Resource"); + setDescription("Retrieve a zone description"); + return super.describe(); + } + + /** + * {@inheritDoc} + */ + @Override + public void describeGet(MethodInfo mInfo) { + mInfo.setIdentifier("Zone description"); + mInfo.setDocumentation("To Retrieve a zone description"); + ResponseInfo response = new ResponseInfo("Current list of zones"); + response.getStatuses().add(Status.SUCCESS_OK); + + RepresentationInfo repInfo = new RepresentationInfo( + MediaType.APPLICATION_JSON); + repInfo.setDocumentation("JSON array of of the current list zone"); + response.getRepresentations().add(repInfo); + + mInfo.getResponses().add(response); + } + + /** + * Handle HTTP GET method / Json + * + * @return a JSON Zone Representation + */ + @Get("json") + public Representation toJSON() { + + final String baseURL = buildBaseURL(getRequest().getResourceRef()); + + String id = (String) getRequest().getAttributes().get("id"); + IZone zone = JDOStorage.getInstance().getById(id, ZoneAdmin.class); + JSONObject jzone = new JSONObject(); + + try { + + jzone.put(ZoneFormat.NAME.getRestAtr(), zone.getName()); + + JSONArray subZones = new JSONArray(); + + for (IZone sub : zone.getSubZones()) { + JSONObject z = new JSONObject(); + z.put(ZoneFormat.URI.getRestAtr(), + String.format("%s/%s", baseURL, sub.getKey())); + z.put(ZoneFormat.NAME.getRestAtr(), sub.getName()); + z.put(ZoneFormat.ZONE_TYPE.getRestAtr(), sub.getType()); + z.put(ZoneFormat.HAS_SENSORS.getRestAtr(), sub.getSensors() + .size() > 0); + + z.put(ZoneFormat.HAS_SUBZONES.getRestAtr(), sub.getSubZones() + .size() > 0); + + subZones.put(z); + } + if (subZones.length() > 0) { + jzone.put(ZoneFormat.SUBZONE.getRestAtr(), subZones); + } + + JSONArray sensors = new JSONArray(); + for (ISensor sensor : zone.getSensors()) { + sensors.put(sensor.toJSONSenML()); + } + if (sensors.length() > 0) { + jzone.put(ZoneFormat.SENSORS_SENML.getRestAtr(), sensors); + } + +/* String md5 = md5(jzone.toString()); + Tag tag = md5 == null ? null : new Tag(md5, false); + if (!isModified(tag)) { + setStatus(Status.REDIRECTION_NOT_MODIFIED); + return null; + } +*/ + Representation json = new JsonRepresentation(jzone); +// json.setTag(tag); + return json; + } catch (Exception e) { + setStatus(Status.SERVER_ERROR_INTERNAL); + log.log(Level.WARNING, "", e); + return null; + } + } + + /** + * HTTP GET method / xml accept + * + * @return a XML Zone Representation + */ + + @Get("xml") + public Representation toXML() { + long before = System.currentTimeMillis(); + String id = (String) getRequest().getAttributes().get("id"); + IZone zone = JDOStorage.getInstance().getById(id, ZoneAdmin.class); + final String baseURL = buildBaseURL(getRequest().getResourceRef()); + + try { + DomRepresentation xml = new DomRepresentation(MediaType.TEXT_XML); + Document d = xml.getDocument(); + + Element currentZone = d.createElement(ZoneFormat.ZONE.getRestAtr()); + currentZone.setAttribute(ZoneFormat.NAME.getRestAtr(), + zone.getName()); + currentZone.setAttribute(ZoneFormat.ZONE_TYPE.getRestAtr(), zone.getType()); + d.appendChild(currentZone); + + for (IZone subzone : zone.getSubZones()) { + Element sub = d.createElement(ZoneFormat.SUBZONE.getRestAtr()); + sub.setAttribute(ZoneFormat.NAME.getRestAtr(), + subzone.getName()); + sub.setAttribute(ZoneFormat.URI.getRestAtr(), + String.format("%s/%s", baseURL, subzone.getKey())); + sub.setAttribute(ZoneFormat.HAS_SENSORS.getRestAtr(), "" + + (subzone.getSensors().size() > 0)); + + sub.setAttribute(ZoneFormat.HAS_SUBZONES.getRestAtr(), "" + + (subzone.getSubZones().size() > 0)); + + currentZone.appendChild(sub); + + } + + for (ISensor sensor : zone.getSensors()) { + Node senml = sensor.toXMLSenML(d); + currentZone.appendChild(senml); + } + +/* String md5 = md5(xml.getText()); + Tag tag = md5 == null ? null : new Tag(md5, false); + if (!isModified(tag)) { + setStatus(Status.REDIRECTION_NOT_MODIFIED); + return null; + } +*/ +// xml.setTag(tag); + return xml; + } catch (Exception e) { + log.log(Level.WARNING, "", e); + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } + + /* + * Alternative method to add an header to the response. Series<Header> + * responseHeaders = (Series<Header>) + * getResponse().getAttributes().get(HeaderConstants.ATTRIBUTE_HEADERS); if + * (responseHeaders == null) { responseHeaders = new + * Series<Header>(Header.class); getResponse().getAttributes().put( + * HeaderConstants.ATTRIBUTE_HEADERS, responseHeaders); + * + * } responseHeaders.add(new Header("ETag", md5)); + */ + + /** + * Build a String baseURL = to current URI/Reference less the last path + * segment. + * + * @param reference + * the current reference object + * @return String base URI. + */ + private String buildBaseURL(Reference reference) { + Reference ref = new Reference(reference); + List<String> segments = ref.getSegments(); + if (segments.size() > 0) { + segments.remove(segments.size() - 1); + ref.setSegments(segments); + } + return ref.getHierarchicalPart(); + } + + private String md5(String value) { + try { + MessageDigest m = MessageDigest.getInstance("MD5"); + m.reset(); + m.update(value.getBytes()); + byte[] digest = m.digest(); + BigInteger bigInt = new BigInteger(1, digest); + String hashtext = bigInt.toString(16); + // Now we need to zero pad it if you actually want the full 32 + // chars. + while (hashtext.length() < 32) { + hashtext = "0" + hashtext; + } + return hashtext; + } catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + private boolean isModified(Tag currentTag) { + boolean modified = true; + if (currentTag == null) { + return modified; + } + + if (getRequest().getConditions() != null + && getRequest().getConditions().getNoneMatch() != null) { + + for (Tag noMatch : getRequest().getConditions().getNoneMatch()) { + if (currentTag.equals(noMatch)) { + modified = false; + break; + } + } + } + return modified; + } +} \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ZonesResource.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ZonesResource.java new file mode 100644 index 0000000000000000000000000000000000000000..252e362a12597b3803b48b6f8c37ad652bf41277 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/common/ZonesResource.java @@ -0,0 +1,200 @@ +package fr.eurecom.restlet.resources.common; + +import java.util.List; + +import org.json.JSONArray; +import org.json.JSONObject; +import org.mortbay.log.Log; +import org.restlet.Context; +import org.restlet.Request; +import org.restlet.Response; +import org.restlet.data.MediaType; +import org.restlet.data.Method; +import org.restlet.data.Reference; +import org.restlet.data.Status; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.ext.wadl.MethodInfo; +import org.restlet.ext.wadl.RepresentationInfo; +import org.restlet.ext.wadl.ResponseInfo; +import org.restlet.ext.wadl.WadlServerResource; +import org.restlet.ext.xml.DomRepresentation; +import org.restlet.representation.Representation; +import org.restlet.representation.StringRepresentation; +import org.restlet.resource.Get; +import org.w3c.dom.Document; +import org.w3c.dom.Element; + +import fr.eurecom.senml.entity.IZone; +import fr.eurecom.senml.entity.ZoneAdmin; +import fr.eurecom.senml.persistence.JDOStorage; + +public class ZonesResource extends WadlServerResource { + + public static enum ZoneFormat { + ID("id", "zone identifier"), NAME("name", "zone name"), SUBZONE( + "subzone", "subzone description"), + // SENSORS("sensors","sensors lsit uri"), + PARENT_ZONE("parent", "name zone parent"), ZONE("zone", "Zone"), URI( + "uri", "URI"), HAS_SUBZONES("has_subzones", "Has subzones"), HAS_SENSORS( + "has_sensors", "Has_sensors"), SENSORS_SENML("senml", + "sensors over senml"),ZONE_TYPE("type", "type of the zone"); + + private final String restAttr; + private final String desc; + + ZoneFormat(String restAttr, String desc) { + this.restAttr = restAttr; + this.desc = desc; + } + + public String getRestAtr() { + return restAttr; + } + + public String getDescription() { + return desc; + } + }; + + /** + * {@inheritDoc} + */ + @Override + public Representation describe() { + setName("Zones Resource"); + setDescription("Retrieve zones list"); + return super.describe(); + } + + /** + * {@inheritDoc} + */ + @Override + public void describeGet(MethodInfo mInfo) { + mInfo.setIdentifier("Zones List"); + Log.info("oooooo"); + mInfo.setDocumentation("To Retrieve a list of zones"); + ResponseInfo response = new ResponseInfo("Current list of zones"); + response.getStatuses().add(Status.SUCCESS_OK); + + RepresentationInfo repInfo = new RepresentationInfo( + MediaType.APPLICATION_JSON); + repInfo.setDocumentation("JSON array of of the current list zone"); + response.getRepresentations().add(repInfo); + + mInfo.getResponses().add(response); + } + + // HACK: Version Restlet-gae 2.1-rc5. + // There is an error routing the GET request when + // the Request Header Content-Type = application/x-www-form-urlencoded and + // Header Accept = "application/json" and + // URI have a query : http://localhost:8888/contacts?sort=firstname. + // In this case the router ignores the Accept header and route the request + // to @Get("xml") systematically because the query is badly interpreted. + // To workaround this problem, at the init we reformat the Entity with an + // empty Representation. + + @Override + public void init(Context context, Request request, Response response) { + // System.out.println("init" + request + " " + response); + // System.out.println("Content-Type: " + + // request.getEntity().getMediaType()); + if (request.getMethod().compareTo(Method.GET) == 0) { + Representation entity = request.getEntity(); + System.out.println("Entity " + entity + " Content-Type: " + + request.getEntity().getMediaType()); + request.setEntity(new StringRepresentation("")); + } + super.init(context, request, response); + } + + /** + * Handle HTTP GET method / Json + * + * @return a JSON list of root zones by default. + */ + @Get("json") + public Representation toJSON() { + // System.out.println("Request Original Ref " + getOriginalRef()); + // System.out.println("Request Entity " + getRequest().getEntityAsText() + // + + // " entity mediaType " + getRequest().getEntity().getMediaType()); + + try { + List<IZone> zones = JDOStorage.getInstance() + .getAll(ZoneAdmin.class); + JSONArray jzones = new JSONArray(); + Reference ref = getRequest().getResourceRef(); + final String baseURL = ref.getHierarchicalPart(); + + for (IZone zone : zones) { + if (zone.getParentZone() == null) { + + JSONObject jzone = new JSONObject(); + jzone.put(ZoneFormat.URI.getRestAtr(), + String.format("%s/%s", baseURL, zone.getKey())); + + jzone.put(ZoneFormat.NAME.getRestAtr(), zone.getName()); + jzone.put(ZoneFormat.ZONE_TYPE.getRestAtr(), zone.getType()); + + jzone.put(ZoneFormat.HAS_SENSORS.getRestAtr(), zone + .getSensors().size() > 0); + + jzone.put(ZoneFormat.HAS_SUBZONES.getRestAtr(), zone + .getSubZones().size() > 0); + + jzones.put(jzone); + } + } + + JSONObject zoneHead = new JSONObject(); + zoneHead.put(ZoneFormat.SUBZONE.getRestAtr(), jzones); + return new JsonRepresentation(zoneHead); + } catch (Exception e) { + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } + + @Get("xml") + public Representation toXML() { + try { + List<IZone> zones = JDOStorage.getInstance() + .getAll(ZoneAdmin.class); + Reference ref = getRequest().getResourceRef(); + final String baseURL = ref.getHierarchicalPart(); + + DomRepresentation xml = new DomRepresentation(MediaType.TEXT_XML); + Document d = xml.getDocument(); + Element root = d.createElement(ZoneFormat.ZONE.getRestAtr()); + d.appendChild(root); + + for (IZone zone : zones) { + if (zone.getParentZone() == null) { + Element subzone = d.createElement(ZoneFormat.SUBZONE + .getRestAtr()); + + subzone.setAttribute(ZoneFormat.NAME.getRestAtr(), + zone.getName()); + + subzone.setAttribute(ZoneFormat.URI.getRestAtr(), + String.format("%s/%s", baseURL, zone.getKey())); + + subzone.setAttribute(ZoneFormat.HAS_SENSORS.getRestAtr(), + "" + (zone.getSensors().size() > 0)); + + subzone.setAttribute(ZoneFormat.HAS_SUBZONES.getRestAtr(), + "" + (zone.getSubZones().size() > 0)); + root.appendChild(subzone); + } + } + + return xml; + } catch (Exception e) { + e.printStackTrace(); + setStatus(Status.SERVER_ERROR_INTERNAL); + return null; + } + } +} \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/Dispatcher.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/Dispatcher.java new file mode 100644 index 0000000000000000000000000000000000000000..3e2dd2675e67bbca3eeef353bf3759d644cd6a17 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/Dispatcher.java @@ -0,0 +1,43 @@ +package fr.eurecom.restlet.resources.gae; + +import org.restlet.Request; +import org.restlet.Response; +import org.restlet.Restlet; +import org.restlet.ext.wadl.ApplicationInfo; +import org.restlet.ext.wadl.DocumentationInfo; +import org.restlet.ext.wadl.WadlApplication; +import org.restlet.resource.Directory; +import org.restlet.routing.Router; + +import fr.eurecom.restlet.resources.common.ContactResource; +import fr.eurecom.restlet.resources.common.ContactsResource; +import fr.eurecom.restlet.resources.common.ContactsSubResource; + +public class Dispatcher extends WadlApplication { + + @Override + public Restlet createInboundRoot() { + Router router = new Router(getContext()); + router.attachDefault(new Directory(getContext(), "war:///")); +// web xml mapping = /* +// router.attach("/contacts", ContactsResource.class); +// router.attach("/contacts/{id}", ContactResource.class); +// web xml mapping = /contacts/* + router.attach("", ContactsResource.class); + router.attach("/subscribe", ContactsSubResource.class); + router.attach("/{id}", ContactResource.class); + return router; + } + + @Override + public ApplicationInfo getApplicationInfo(Request request, Response response) { + ApplicationInfo info = super.getApplicationInfo(request, response); + + DocumentationInfo docInfo = new DocumentationInfo( + "emulator-box-services application documentation"); + docInfo.setTitle("Emulator Box Services application infos."); + info.setDocumentation(docInfo); + + return info; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/DispatcherSenml.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/DispatcherSenml.java new file mode 100644 index 0000000000000000000000000000000000000000..d103d44d5af876ac3c6017999dff670a4ac7ae28 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/DispatcherSenml.java @@ -0,0 +1,48 @@ +package fr.eurecom.restlet.resources.gae; + +import java.util.logging.Logger; + +import org.restlet.Request; +import org.restlet.Response; +import org.restlet.Restlet; +import org.restlet.ext.wadl.ApplicationInfo; +import org.restlet.ext.wadl.DocumentationInfo; +import org.restlet.ext.wadl.WadlApplication; +import org.restlet.resource.Directory; +import org.restlet.routing.Router; + +import fr.eurecom.restlet.resources.common.SensorPost; +import fr.eurecom.restlet.resources.common.ZonesResource; +import fr.eurecom.restlet.resources.common.ZoneResource; +import fr.eurecom.restlet.resources.common.SensorInResource; + + +public class DispatcherSenml extends WadlApplication { + private static final Logger log = Logger.getLogger("DispatcherSenml"); + + @Override + public Restlet createInboundRoot() { + log.info("SenML dispatcher attaches resources"); + Router router = new Router(getContext()); + router.attachDefault(new Directory(getContext(), "war:///")); +// web xml mapping = /* + + router.attach("/zones", ZonesResource.class); + router.attach("/zones/{id}", ZoneResource.class); + router.attach("/sensorsin", SensorInResource.class); + router.attach("/posttest", SensorPost.class); + return router; + } + + @Override + public ApplicationInfo getApplicationInfo(Request request, Response response) { + ApplicationInfo info = super.getApplicationInfo(request, response); + + DocumentationInfo docInfo = new DocumentationInfo( + "emulator-box-services application documentation"); + docInfo.setTitle("Emulator Box Services application infos."); + info.setDocumentation(docInfo); + + return info; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/ResourcesServletGAE.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/ResourcesServletGAE.java new file mode 100644 index 0000000000000000000000000000000000000000..5d46ceb2dc89f7c4b4eefdf6d1bbf0fef25a618f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/gae/ResourcesServletGAE.java @@ -0,0 +1,24 @@ +package fr.eurecom.restlet.resources.gae; + +import java.io.IOException; +import javax.servlet.http.HttpServlet; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + +/** + * This class serves for test only. Can be deleted. + * @author ouest + * + */ + +@SuppressWarnings("serial") +public class ResourcesServletGAE extends HttpServlet { + + @Override + public void doGet(HttpServletRequest req, HttpServletResponse resp) + throws IOException { + resp.setContentType("text/plain"); + resp.getWriter().println("Hello, world"); + } + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/persistence/PMF.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/persistence/PMF.java new file mode 100644 index 0000000000000000000000000000000000000000..77d3e5f7416c47e35be850ade14b36740e5ea504 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/persistence/PMF.java @@ -0,0 +1,18 @@ +package fr.eurecom.restlet.resources.persistence; + +import javax.jdo.JDOHelper; +import javax.jdo.PersistenceManagerFactory; + +public final class PMF { + private static final PersistenceManagerFactory pmfInstance = JDOHelper + .getPersistenceManagerFactory("transactions-optional"); + + private PMF() { + } + + public static PersistenceManagerFactory get() { + return pmfInstance; + } +} + +// PersistenceManager pm = PMF.get().getPersistenceManager(); diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/persistence/Storage.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/persistence/Storage.java new file mode 100644 index 0000000000000000000000000000000000000000..c0f0983d9ccb905e34054bdf9ce676164e78f251 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/restlet/resources/persistence/Storage.java @@ -0,0 +1,252 @@ +package fr.eurecom.restlet.resources.persistence; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.TreeMap; + +import fr.eurecom.restlet.resources.common.Contact; +import fr.eurecom.restlet.resources.common.ContactBaseResource.MEMBERS; + +/** + * This class is a naive implementation of contact's datastore, it should be + * replaced by a NOSQL JDO storage. + */ +// TODO: Replace this class by NOSQL JDO implementation. +public class Storage { + + /** + * Array of contacts. Restricted to 50 contacts. + */ + private final Contact[] backend = new Contact[50]; + + /** + * Storage instance. + */ + private static Storage instance = null; + + private static final String TEMPLATE_SORT = "%s-%s"; + + /** + * Constructs a storage. Initialize a bunch of contacts. + */ + private Storage() { + backend[0] = new Contact(0, "Bonnie", "Parker", "bonnie@gangsters.org", + "+1-555-329-4589"); + backend[1] = new Contact(1, "Clyde", "Barrow", "clyde@gangsters.org", + "+1-555-329-4589"); + backend[2] = new Contact(2, "John", "Dillinger", "john@gangsters.org", + "+1-555-355-3232"); + backend[3] = new Contact(3, "Herman", "Lamm", "herman@gangsters.org", + "+1-555-258-2365"); + backend[4] = new Contact(4, "Willie", "Sutton", "will@gangsters.org", + "+1-555-258-2365"); + + for (int i = 5; i < 50; i++) { + backend[i] = null; + } + } + + /** + * Retrieve the singleton storage. + * + * @return a Storage instance. + */ + public static Storage getInstance() { + if (instance == null) { + instance = new Storage(); + } + return instance; + } + + /** + * Retrieve a list of sorted contacts, starting with the given offset. + * + * @param sort + * - Sort parameter. + * @param minLimit + * - reserved for a future usage + * @param maxLimit + * - reserved for a future usage + * @return a sorted list of the contacts. + */ + @SuppressWarnings("unchecked") + public List<Contact> getContacts(MEMBERS sort, int minLimit, int maxLimit) { + switch (sort) { + case ID: + return getSortedById(); + case FIRSTNAME: + return getSortedByFirstName(); + case LASTNAME: + return getSortedByLastName(); + case PHONE: + return getSortedByPhone(); + case MAIL: + return getSortedByEmail(); + } + + return Collections.EMPTY_LIST; + } + + /** + * Save a contact, the Contact's id must be comprise between 0 and 49. + * + * @param contact + * to save. + * @return true if correctly saved, false otherwise. + */ + public boolean saveContact(Contact contact) { + if (checkIndex(contact.getId())) { + backend[contact.getId()] = contact; + return true; + } + return false; + } + + /** + * Delete a contact. Same as {@link #removeContact(int)} + * {@code + * removeContact(conctact.getId(); + * } + * + * @param contact + * to delete. + * @return true if deleted, false otherwise. + */ + public boolean removeContact(Contact contact) { + return removeContact(contact.getId()); + } + + /** + * Delete a contact + * + * @param identifier + * - Contact's identifier + * @return true if deleted, false otherwise + * @see Storage.removeContact(Contact) + */ + public boolean removeContact(int identifier) { + if (checkIndex(identifier)) { + backend[identifier] = null; + return true; + } + return false; + } + + /** + * Retrieve a contact from an identifier. + * + * @param identifier + * - Contact's identifier + * @return Contact. + */ + public Contact getContact(int identifier) { + if (checkIndex(identifier)) { + return backend[identifier]; + } + return null; + } + + private boolean checkIndex(int index) { + return index > -1 && index < 50; + } + + /** + * Sort contacts by email field. + * + * @return a sorted list of contacts. + */ + private List<Contact> getSortedByEmail() { + Map<String, Contact> sorted = new TreeMap<String, Contact>( + String.CASE_INSENSITIVE_ORDER); + for (int i = 0; i < 50; i++) { + if (backend[i] != null) { + sorted.put(unicKey(backend[i].getMail(), i), backend[i]); + } + } + return new ArrayList<Contact>(sorted.values()); + } + + /** + * Sort contacts by phone number + * + * @return a sorted list of contacts. + */ + private List<Contact> getSortedByPhone() { + Map<String, Contact> sorted = new TreeMap<String, Contact>( + String.CASE_INSENSITIVE_ORDER); + for (int i = 0; i < 50; i++) { + if (backend[i] != null) { + sorted.put(unicKey(toNumeric(backend[i].getPhone()), i), + backend[i]); + } + } + return new ArrayList<Contact>(sorted.values()); + } + + /** + * Sort contacts by last name + * + * @return a sorted list of contacts. + */ + private List<Contact> getSortedByLastName() { + Map<String, Contact> sorted = new TreeMap<String, Contact>( + String.CASE_INSENSITIVE_ORDER); + for (int i = 0; i < 50; i++) { + if (backend[i] != null) { + sorted.put(unicKey(backend[i].getLastName(), i), backend[i]); + } + } + return new ArrayList<Contact>(sorted.values()); + } + + /** + * Sort contacts by first name + * + * @return a sorted list of contacts. + */ + private List<Contact> getSortedByFirstName() { + Map<String, Contact> sorted = new TreeMap<String, Contact>( + String.CASE_INSENSITIVE_ORDER); + for (int i = 0; i < 50; i++) { + if (backend[i] != null) { + sorted.put(unicKey(backend[i].getFirstName(), i), backend[i]); + } + } + return new ArrayList<Contact>(sorted.values()); + } + + /** + * Sort contacts by identifier. + * + * @return a sorted list of contacts. + */ + private List<Contact> getSortedById() { + return Arrays.asList(backend); + } + + private static String toNumeric(String phone) { + char[] chars = phone.toCharArray(); + StringBuilder builder = new StringBuilder(); + for (char c : chars) { + if (Character.isDigit(c)) { + builder.append(c); + } + } + return builder.toString(); + } + + /** + * Format a unic key (necessary when at least 2 contacts have a same + * property value like firstname/lastname..) + * + * @param key + * @param id + * @return + */ + private static String unicKey(String key, int id) { + return String.format(TEMPLATE_SORT, key, id); + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ContactTest.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ContactTest.java new file mode 100644 index 0000000000000000000000000000000000000000..a3d5a2c7caaf2ffd85c313a40dd607be01c13336 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ContactTest.java @@ -0,0 +1,36 @@ +package fr.eurecom.senml.entity; + +import javax.jdo.annotations.IdGeneratorStrategy; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; + +import com.google.appengine.api.datastore.Key; + +@PersistenceCapable +public class ContactTest { + + @PrimaryKey + @Persistent(valueStrategy = IdGeneratorStrategy.IDENTITY) + private Key key; + + @Persistent + private String firstName; + + @Persistent + private String lastName; + + @Persistent + private String email; + + public ContactTest(String firstName, String lastName, String email) { + this.firstName = firstName; + this.lastName = lastName; + this.email = email; + } + + public String getirstName() {return firstName;} + public String getLastName() {return lastName;} + public String getEmail() {return email;} + public Key getKey() {return key;} +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IPostVal.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IPostVal.java new file mode 100644 index 0000000000000000000000000000000000000000..1e759d7af5d5dcbece4849ebce2bd0faf1449228 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IPostVal.java @@ -0,0 +1,9 @@ +package fr.eurecom.senml.entity; + + +public interface IPostVal { + public final static String BASE_NAME = "bn"; + public final static String STATUS = "status"; + + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ISensor.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ISensor.java new file mode 100644 index 0000000000000000000000000000000000000000..6bb154b9391eb225a34079e68d589657a8637d01 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ISensor.java @@ -0,0 +1,37 @@ +package fr.eurecom.senml.entity; + +import java.util.List; + +import org.json.JSONObject; +import org.w3c.dom.Document; +import org.w3c.dom.Node; + +public interface ISensor { + public final static String BASE_NAME = "bn"; + public final static String BASE_TIME = "bt"; + public final static String BASE_UNIT = "bu"; + public final static String VERSION = "ver"; + public final static String MEASUREMENT = "e"; + + public String getUUID(); + + public IZone getZone(); + + public String getTitle(); + + public List <Measure> getMeasures(); + +// public List<String> getMeasuresKey(); + +// public List<String> getMeasuresPrint(); + + public JSONObject toJSONSenML(); + + public Document toXMLSenML(); + + public Node toXMLSenML(Document rootDocument); + + public String getBaseUnit(); + + public String zoneType(); +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ISensorAdmin.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ISensorAdmin.java new file mode 100644 index 0000000000000000000000000000000000000000..69b14b2f07e07ebdf632b9a62d0f159215765d0e --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ISensorAdmin.java @@ -0,0 +1,34 @@ +package fr.eurecom.senml.entity; + +import fr.eurecom.senml.entity.Measure.ParamTypeValue; + +public interface ISensorAdmin extends ISensor { + + public boolean linkZone(IZone zone); + + public boolean linkZone(String zoneKey); + + public void unlinkZone(); + + public Measure addMeasure(String measureName, + Units measureUnit, + ParamTypeValue measureTypeValue, + String measureValue, long timeMeasure, String type); + + public void deleteMeasure(String measureKey); + + public void deleteMeasure(Measure measure); + + public void setTitle(String title); + +// void tag(Tag tag); + +// void untag(Tag tag); + + public void destroy(); + + public Measure addMeasure(String measureName, Units measureUnit, + ParamTypeValue measureTypeValue, String measureValue, + long timeMeasure, String type, String sum); + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IZone.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IZone.java new file mode 100644 index 0000000000000000000000000000000000000000..08e3465ab0d99d1a540eb3eba12272be2bfde477 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IZone.java @@ -0,0 +1,23 @@ +package fr.eurecom.senml.entity; + +import java.util.List; + +public interface IZone extends Comparable<IZone> { + +// public boolean hasSensors(); + +// public boolean hasSubZones(); + + public List<IZone> getSubZones(); + + public List<ISensor> getSensors(); + + public String getName(); + + public String getKey(); + + public IZone getParentZone(); + + String getType(); + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IZoneAdmin.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IZoneAdmin.java new file mode 100644 index 0000000000000000000000000000000000000000..15e665ff0c60cdfdfa2b934f4e752a6d4991170c --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/IZoneAdmin.java @@ -0,0 +1,17 @@ +package fr.eurecom.senml.entity; + + +public interface IZoneAdmin extends IZone { + + public boolean linkParentZone(String parentZoneKey); + + public boolean linkParentZone(IZone parentZone); + + public void unlinkParentZone(); + +// public boolean addSensor(String sensorKey); + +// public boolean removeSensor(String sensorKey); + + public void destroy(); +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Measure.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Measure.java new file mode 100644 index 0000000000000000000000000000000000000000..5d707a1ff06074ed0deb378f7805a8ae4636c627 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Measure.java @@ -0,0 +1,301 @@ +package fr.eurecom.senml.entity; + +import java.sql.Date; +import java.text.DateFormat; +import java.text.SimpleDateFormat; +import java.util.HashMap; +import java.util.UUID; +import java.util.logging.Logger; +import java.util.Map; + +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; + +import org.json.JSONObject; +import org.w3c.dom.Document; +import org.w3c.dom.Element; +import org.w3c.dom.Node; + +import com.google.appengine.api.datastore.Key; +import com.google.appengine.api.datastore.KeyFactory; + +import fr.eurecom.senml.persistence.JDOStorage; + +/** + * Immutable object. Represents the SenML Measurement entry. + * + * @author dalmassi + * + */ + +@PersistenceCapable(detachable = "true") +public class Measure { + + public enum ParamTypeValue { + BOOL("bv"), FLOAT("v"), STRING("sv"), SUM("s"); + + private String encoded; + private final static Map<String, ParamTypeValue> mTypeValue = new HashMap<String, ParamTypeValue>(); + + ParamTypeValue(String encoded) { + this.encoded = encoded; + } + + public String getSenML() { + return encoded; + } + + public static ParamTypeValue getParamTypeValue(String senmlEncoded) { + if (mTypeValue == null) { + initMapValue(); + } + return mTypeValue.get(senmlEncoded); + } + + private static synchronized void initMapValue() { + if (mTypeValue.size() > 0) { + return; + } + for (ParamTypeValue v : values()) { + mTypeValue.put(v.encoded, v); + } + } + }; + + @NotPersistent + private static final DateFormat df = new SimpleDateFormat( + "yyyy.MM.dd G 'at' HH:mm:ss z"); + + @NotPersistent + private static final Logger log = Logger + .getLogger("sensors.entity.Measure"); + @NotPersistent + public static final String PARAM_UNIT = "u"; + @NotPersistent + public static final String PARAM_NAME = "n"; + @NotPersistent + public static final String PARAM_TIME = "t"; + @NotPersistent + public static final String PARAM_VALUE = "v"; + @NotPersistent + public static final String PARAM_STRING_VALUE = "sv"; + @NotPersistent + public static final String PARAM_BOOLEAN_VALUE = "bv"; + @NotPersistent + public static final String PARAM_SUM_VALUE = "s"; + @NotPersistent + public static final String PARAM_UPDATE_TIME = "ut"; + @NotPersistent + public static final String PARAM_TYPE = "type"; + + @PrimaryKey + @Persistent + Key key; + + @Persistent + private String name; + + @Persistent + private String unitCS; + + @Persistent + private String unitPrint; + + @Persistent + private String unitKindOfQty; + + @Persistent + private String value; + + @Persistent + private String paramTypeValueEncoded; + + private long time; + + @Persistent + private String sensorKey; + + private String baseunitCS; + + private String baseunitPrint; + + private String baseunitKindOfQty; + + private String type; + + private String sum; + + public Measure(String sensorKey, String name, Units unit, + ParamTypeValue typeValue, String value, long time, String type) { + this.sensorKey = sensorKey; + + this.name = name == null || name.isEmpty() ? generateName() : name; + this.unitCS = unit == null ? null : unit.getCS(); + this.unitPrint = unit == null ? null : unit.getPrint(); + this.unitKindOfQty = unit == null ? null : unit.getKindofQty(); + this.paramTypeValueEncoded = typeValue.getSenML(); + this.value = value; + this.time = time; + this.key = KeyFactory.createKey(Measure.class.getSimpleName(), + sensorKey + name); + this.type = type; + } + + public Measure(String sensorKey, String name, Units unit, + ParamTypeValue typeValue, String value, long time, String type, String sum) { + this.sensorKey = sensorKey; + + this.name = name == null || name.isEmpty() ? generateName() : name; + this.unitCS = unit == null ? null : unit.getCS(); + this.unitPrint = unit == null ? null : unit.getPrint(); + this.unitKindOfQty = unit == null ? null : unit.getKindofQty(); + this.paramTypeValueEncoded = typeValue.getSenML(); + this.value = value; + this.time = time; + this.sum = sum; + this.key = KeyFactory.createKey(Measure.class.getSimpleName(), + sensorKey + name); + this.type = type; + + } + + public void destroy() { + JDOStorage.getInstance().delete(this); + } + + public String getKey() { + return key == null ? null : KeyFactory.keyToString(key); + } + + public String getType() { + return type; + } + + public JSONObject toJSONSenML() { + try { + JSONObject entrySenML = new JSONObject(); + if (name != null) { + entrySenML.put(PARAM_NAME, name); + } + + if (sum != null) { + entrySenML.put(PARAM_SUM_VALUE, sum); + } + + entrySenML.put(PARAM_TYPE, type); + + entrySenML.put(PARAM_TIME, time); + if (this.unitPrint != null) { + entrySenML.put(PARAM_UNIT, unitCS); + } + + if (paramTypeValueEncoded != null) { + entrySenML.put(paramTypeValueEncoded, value); + } + return entrySenML; + } catch (Exception e) { + e.printStackTrace(); + log.warning("Error Encoding a measure"); + return null; + } + } + + public Node toXMLSenML(Document d) { + Element measure = d.createElement("e"); + measure.setAttribute(PARAM_NAME, name); + measure.setAttribute(PARAM_TYPE, type); + measure.setAttribute(PARAM_TIME, "" + time); + if (this.unitPrint != null) { + measure.setAttribute(PARAM_UNIT, unitCS); + } + + if (this.sum != null) { + measure.setAttribute(PARAM_SUM_VALUE, sum); + } + + if (paramTypeValueEncoded != null) { + measure.setAttribute(paramTypeValueEncoded, value); + } + return measure; + } + + public String toString() { + StringBuilder builder = new StringBuilder(); + if (this.unitCS != null) { + builder.append("key[" + getKey()).append("] name[" + name) + .append("] unitCS[" + unitCS) + .append("] unitPrint[" + unitPrint) + .append(this.sum == null ? "" : "] sum[" + sum) + .append("] unitKindOfQty[" + unitKindOfQty) + .append("] paramTypeEncoded[" + paramTypeValueEncoded) + .append("] value[" + value).append("] time[" + time) + .append("] sensorKey[" + sensorKey).append("]"); + } else { + // Check if we have to initialize the base unit + baseUnitInit(); + builder.append("key[" + getKey()) + .append("] name[" + name) + .append("] unitCS (base unit) [" + baseunitCS) + .append("] unitPrint (base unit) [" + baseunitPrint) + .append(this.sum == null ? "" : "] sum[" + sum) + .append("] unitKindOfQty (base unit) [" + baseunitKindOfQty) + .append("] paramTypeEncoded[" + paramTypeValueEncoded) + .append("] value[" + value).append("] time[" + time) + .append("] sensorKey[" + sensorKey).append("]"); + } + return builder.toString(); + } + + public String toPrettyString() { + StringBuilder builder = new StringBuilder(); + if (this.unitCS != null) { + builder.append(name).append(": ").append("(kind: ") + .append(unitKindOfQty).append(") ").append(value) + .append(this.sum == null ? "" : " sum: " + sum) + .append(" ").append(unitPrint).append(" @ ") + .append(formatDate(time)); + } else { + // Check if we have to initialize the base unit + baseUnitInit(); + builder.append(name).append(": ").append("(kind (baseunit): ") + .append(baseunitKindOfQty).append(") ").append(value) + .append(" ").append(baseunitPrint).append("(baseunit) @ ") + .append(formatDate(time)); + } + return builder.toString(); + } + + private void baseUnitInit() { + System.out.println("unit is null, parent sensor is " + sensorKey); + // Retrieve info about base unit from the parent sensor + Sensor parent = JDOStorage.getInstance().getById(sensorKey, + Sensor.class); + System.out.println("sensor is " + parent.getTitle()); + String bu = parent.getBaseUnit(); + System.out.println("sensor bu is " + bu); + Units baseUnits = Units.getUnit(bu); + this.baseunitCS = baseUnits.getCS(); + this.baseunitKindOfQty = baseUnits.getKindofQty(); + this.baseunitPrint = baseUnits.getPrint(); + System.out.println("base unit is: " + this.baseunitCS); + } + + private String formatDate(long ms) { + if (ms == 0) + return "now"; + if (ms < 0) + return ("now - " + ms); + + return df.format(new Date(ms)); + } + + public String getSensorKey() { + return sensorKey; + } + + private String generateName() { + return UUID.randomUUID().toString(); + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/PostValAdmin.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/PostValAdmin.java new file mode 100644 index 0000000000000000000000000000000000000000..f11c4e9f9c0a8173d95ec5b6ea3a78cab3e5815b --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/PostValAdmin.java @@ -0,0 +1,21 @@ +package fr.eurecom.senml.entity; + +import java.util.UUID; + +import javax.jdo.annotations.Persistent; + +import fr.eurecom.senml.persistence.JDOStorage; + +public class PostValAdmin implements IPostVal { + + @Persistent + boolean active = true; + + public PostValType addPost(String name, String val, long timeMeasure) { + PostValType m = new PostValType(UUID.randomUUID().toString(), name, val, timeMeasure); + JDOStorage.getInstance().write(m); + //JDOStorage.getInstance().deleteAll(PostValType.class); + return m; + //return null; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/PostValType.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/PostValType.java new file mode 100644 index 0000000000000000000000000000000000000000..65a7ff828fca364e3f40bad4bfc1e8ce37ef5a67 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/PostValType.java @@ -0,0 +1,173 @@ +package fr.eurecom.senml.entity; + +import java.sql.Date; +import java.text.DateFormat; +import java.text.SimpleDateFormat; +import java.util.HashMap; +import java.util.UUID; +import java.util.logging.Logger; +import java.util.Map; + +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; + +import org.json.JSONObject; +import org.w3c.dom.Document; +import org.w3c.dom.Element; +import org.w3c.dom.Node; + +import com.google.appengine.api.datastore.Key; +import com.google.appengine.api.datastore.KeyFactory; + +import fr.eurecom.senml.persistence.JDOStorage; + +/** + * Immutable object. Represents the SenML Measurement entry. + * + * + */ + +@PersistenceCapable(detachable="true") +public class PostValType { + + public enum ParamTypeValue { + STR("sv"); + + private String encoded; + private final static Map<String, ParamTypeValue> mTypeValue = new HashMap<String, ParamTypeValue>(); + + ParamTypeValue(String encoded) { + this.encoded = encoded; + } + + public String getSenML() { + return encoded; + } + + public static ParamTypeValue getParamTypeValue(String senmlEncoded) { + if (mTypeValue == null) { + initMapValue(); + } + return mTypeValue.get(senmlEncoded); + } + + private static synchronized void initMapValue() { + if (mTypeValue.size() > 0) { + return; + } + for (ParamTypeValue v : values()) { + mTypeValue.put(v.encoded, v); + } + } + }; + + @NotPersistent + private static final DateFormat df = new SimpleDateFormat("yyyy.MM.dd G 'at' HH:mm:ss z"); + + @NotPersistent + private static final Logger log = Logger + .getLogger("sensors.entity.PostValType"); + + @NotPersistent + public static final String PARAM_STRING_VALUE = "sv"; + + @NotPersistent + public static final String PARAM_NAME = "n"; + + @PrimaryKey + @Persistent + Key key; + + @Persistent + private String name; + + @Persistent + private String value; + + @Persistent + private long time; + + public String getName() { + return name; + } + + public String getValue() { + return String.valueOf(value); + } + + public String getTime() { + return formatDate(time); + } + + public PostValType(String sensorKey, String name, + String val, long time) { + this.name = name == null || name.isEmpty() ? generateName() : name; + this.value = val; + this.time = time; + this.key = KeyFactory.createKey(PostValType.class.getSimpleName(), sensorKey + name +"" + time); + } + + public void destroy() { + JDOStorage.getInstance().delete(this); + } + + public String getKey() { + return key == null ? null : KeyFactory.keyToString(key); + } + + public JSONObject toJSONSenML() { + try { + JSONObject entrySenML = new JSONObject(); + if (name != null) { + entrySenML.put(PARAM_NAME, name); + } + entrySenML.put(PARAM_STRING_VALUE, value); + return entrySenML; + } catch (Exception e) { + e.printStackTrace(); + log.warning("Error Encoding a measure"); + return null; + } + } + + public Node toXMLSenML(Document d) { + Element measure = d.createElement("e"); + measure.setAttribute(PARAM_NAME, name); + measure.setAttribute(PARAM_STRING_VALUE, String.valueOf(value)); + return measure; + } + + public String toString() { + StringBuilder builder = new StringBuilder(); + builder.append("key[" + getKey()) + .append("] name[" + name) + .append("] value[" + value) + .append("] time[" + formatDate(time)) + .append("]"); + return builder.toString(); + } + + public String toPrettyString() { + StringBuilder builder = new StringBuilder(); + builder.append(name) + .append(": ") + .append(") ") + .append(value) + .append(" @ ") + .append(formatDate(time)); + return builder.toString(); + } + + private String formatDate(long ms) { + if (ms == 0) return "now"; + if (ms < 0) return ("now - " + ms); + + return df.format(new Date(ms)); + } + + private String generateName() { + return UUID.randomUUID().toString(); + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/RegisteredDeviceType.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/RegisteredDeviceType.java new file mode 100644 index 0000000000000000000000000000000000000000..5b6db90132cddf98a702fc949db3d489ea13b43e --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/RegisteredDeviceType.java @@ -0,0 +1,188 @@ +package fr.eurecom.senml.entity; + +import java.sql.Date; +import java.text.DateFormat; +import java.text.SimpleDateFormat; +import java.util.HashMap; +import java.util.UUID; +import java.util.logging.Logger; +import java.util.Map; + +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; + +import org.json.JSONObject; +import org.w3c.dom.Document; +import org.w3c.dom.Element; +import org.w3c.dom.Node; + +import com.google.appengine.api.datastore.Key; +import com.google.appengine.api.datastore.KeyFactory; + +import fr.eurecom.senml.persistence.JDOStorage; + +/** + * Immutable object. Represents the registered device. + */ + +@PersistenceCapable(detachable="true") +public class RegisteredDeviceType { + + public enum ParamTypeValue { + STR("sv"); + + private String encoded; + private final static Map<String, ParamTypeValue> mTypeValue = new HashMap<String, ParamTypeValue>(); + + ParamTypeValue(String encoded) { + this.encoded = encoded; + } + + public String getSenML() { + return encoded; + } + + public static ParamTypeValue getParamTypeValue(String senmlEncoded) { + if (mTypeValue == null) { + initMapValue(); + } + return mTypeValue.get(senmlEncoded); + } + + private static synchronized void initMapValue() { + if (mTypeValue.size() > 0) { + return; + } + for (ParamTypeValue v : values()) { + mTypeValue.put(v.encoded, v); + } + } + }; + + @NotPersistent + private static final DateFormat df = new SimpleDateFormat("yyyy.MM.dd G 'at' HH:mm:ss z"); + + @NotPersistent + private static final Logger log = Logger + .getLogger("sensors.entity.RegisteredDeviceType"); + + @NotPersistent + public static final String PARAM_STRING_VALUE = "sv"; + + @NotPersistent + public static final String PARAM_NAME = "n"; + + @PrimaryKey + @Persistent + Key key; + + @Persistent + private String deviceType; + + @Persistent + private String registrationKey; + + @Persistent + private long time; + + private boolean isadmin = false; + + public String getDeviceType() { + return deviceType; + } + + public String getValue() { + return String.valueOf(registrationKey); + } + + public String getTime() { + return formatDate(time); + } + + public RegisteredDeviceType(String deviceKey, String deviceType, + String val, long time) { + this.deviceType = + deviceType == null || deviceType.isEmpty() ? generateName() : deviceType; + this.registrationKey = val; + this.time = time; + this.key = KeyFactory.createKey( + RegisteredDeviceType.class.getSimpleName(), + deviceKey + deviceType +"" + time); + } + + public void destroy() { + JDOStorage.getInstance().delete(this); + } + + public String getKey() { + return key == null ? null : KeyFactory.keyToString(key); + } + + public String getDeviceRegistrationKey() { + return this.registrationKey; + } + + public JSONObject toJSONSenML() { + try { + JSONObject entrySenML = new JSONObject(); + if (deviceType != null) { + entrySenML.put(PARAM_NAME, deviceType); + } + entrySenML.put(PARAM_STRING_VALUE, registrationKey); + return entrySenML; + } catch (Exception e) { + e.printStackTrace(); + log.warning("Error Encoding a measure"); + return null; + } + } + + public Node toXMLSenML(Document d) { + Element measure = d.createElement("e"); + measure.setAttribute(PARAM_NAME, deviceType); + measure.setAttribute(PARAM_STRING_VALUE, String.valueOf(registrationKey)); + return measure; + } + + public String toString() { + StringBuilder builder = new StringBuilder(); + builder.append("key[" + getKey()) + .append("] name[" + deviceType) + .append("] value[" + registrationKey) + .append("] time[" + formatDate(time)) + .append("]"); + return builder.toString(); + } + + public String toPrettyString() { + StringBuilder builder = new StringBuilder(); + builder.append(deviceType) + .append(": ") + .append(") ") + .append(registrationKey) + .append(" @ ") + .append(formatDate(time)); + return builder.toString(); + } + + private String formatDate(long ms) { + if (ms == 0) return "now"; + if (ms < 0) return ("now - " + ms); + + return df.format(new Date(ms)); + } + + private String generateName() { + return UUID.randomUUID().toString(); + } + + public boolean isIsadmin() { + return isadmin; + } + + public void setIsadmin(boolean isadmin) { + this.isadmin = isadmin; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Sensor.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Sensor.java new file mode 100644 index 0000000000000000000000000000000000000000..e00eefefb642c81dac1a9c4d047db93f9aa2f987 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Sensor.java @@ -0,0 +1,202 @@ +package fr.eurecom.senml.entity; + +import java.util.Iterator; +import java.util.List; + +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; +import javax.xml.parsers.DocumentBuilder; +import javax.xml.parsers.DocumentBuilderFactory; + +import org.json.JSONObject; +import org.json.JSONArray; +import org.w3c.dom.Document; +import org.w3c.dom.Element; + +import fr.eurecom.senml.persistence.JDOStorage; +import fr.eurecom.senml.persistence.JDOStorageZone; + +@PersistenceCapable(detachable="true") +public class Sensor implements ISensor { + + @Persistent + @PrimaryKey + protected String uuid; + + @Persistent + protected String title = null; + + @Persistent + protected String zoneKey = null; + + @Persistent + protected String baseUnit = null; + + @Persistent + protected String type = null; + +// Not supported. GAE load only lazy objects (empty objects) and does not support joins. +// @Persistent(defaultFetchGroup="true") +// protected Set<Measure> measures; + + protected Sensor(String uuid, String title, String zoneKey) { + this.uuid = uuid; + this.title = title.trim(); + this.zoneKey = zoneKey; + } + + public Sensor(String uuid, String title, String zoneKey, String baseUnit, String type) { + this.uuid = uuid; + this.title = title.trim(); + this.zoneKey = zoneKey; + this.baseUnit = baseUnit; + this.type = type; + } + + @Override + public String getUUID() { + return uuid; + } + + @Override + public String zoneType() { + return type; + } + + @Override + public String getTitle() { + return title; + } + + @Override + public IZone getZone() { + if (zoneKey != null) { + return JDOStorage.getInstance().getById(zoneKey, ZoneAdmin.class); + } + return null; + } + + @Override + public String getBaseUnit() { + return baseUnit; + } + + @Override + public JSONObject toJSONSenML() { + JSONObject sensorSenML = new JSONObject(); + try { + sensorSenML.put("bn", getBaseName()); + sensorSenML.put("type", zoneType()); + if (getBaseUnit() != null) { + sensorSenML.put("bu", getBaseUnit()); + } + + JSONArray mArray = new JSONArray(); + for (Measure m : getMeasures()) { + mArray.put(m.toJSONSenML()); + } + if (mArray.length() > 0) { + sensorSenML.put("e", mArray); + } + } + catch (Exception e) { + e.printStackTrace(); + } +// System.out.println(sensorSenML.toString()); + return sensorSenML; + } + + @Override + public Document toXMLSenML() { + try { + DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); + DocumentBuilder builder = factory.newDocumentBuilder(); + Document d = builder.newDocument(); + d.appendChild(toXMLSenML(d)); + return d; + } + catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + @Override + public Element toXMLSenML(Document documentRoot) { + try { + Element senml = documentRoot.createElement("senml"); + senml.setAttribute("bn", getBaseName()); + senml.setAttribute("type", zoneType()); + if (getBaseUnit() != null) { + senml.setAttribute("bu", getBaseUnit()); + } + + for (Measure m : getMeasures()) { + senml.appendChild(m.toXMLSenML(documentRoot)); + } + return senml; + } + catch (Exception e) { + e.printStackTrace(); + return null; + } + } + + @Override + public String toString() { + StringBuilder builder = new StringBuilder(); + builder.append("Title : " + title); + builder.append("\"bn\" :").append(" (" + type).append(") \"uuid:" + uuid + "\" Measures "); + builder.append("\r\n"); + for (Measure m : JDOStorageZone.getInstance().getMeasures(this)) { + builder.append("\t").append(m.toString()); + builder.append("\r\n"); + } + return builder.toString(); + } + +/* @Override + public List<String> getMeasuresPrint() { + List<String> mString = new ArrayList<String>(); + + for (String mKey : measuresKey) { + Measure m = JDOStorage.getInstance().getById(mKey, Measure.class); + mString.add(m.toPrettyString()); + } + return mString; + } +*/ + +/* @Override + public List<String> getMeasuresKey() { + return new ArrayList<String>(measuresKey); + } +*/ + + @Override + public List<Measure> getMeasures() { + return JDOStorageZone.getInstance().getMeasures(this); + } + +/* HACK: "Generic Sensor" when the sensor has been created receiving a senml file (sensorin resource). + else sensor has been entered by the jsp file the uuid is generated by this program. + this is a poor workaround to support both. in a future the jsp should disappears. + */ + private final String getBaseName() { + return uuid; + //return "Generic Sensor".equals(title) ? uuid : String.format("urn:%s:uuid:%s", title.toLowerCase(), uuid); + } + + public static Sensor getByUUID(String uuid) { + List<Sensor> sensors = JDOStorage.getInstance().getAll(Sensor.class); + Iterator<Sensor> iter = sensors.iterator(); + while (iter.hasNext()) { + Sensor current = iter.next(); + if (current.getUUID().equals(uuid)) { + return current; + } + } + return null; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/SensorAdmin.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/SensorAdmin.java new file mode 100644 index 0000000000000000000000000000000000000000..bd2ff2f8a723715a22fc9baffa69bb26289afc2c --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/SensorAdmin.java @@ -0,0 +1,141 @@ +package fr.eurecom.senml.entity; + +import java.util.UUID; + +import javax.jdo.annotations.Inheritance; +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; + +import org.json.JSONException; + +import fr.eurecom.pushing.GCMPushingServer; +import fr.eurecom.senml.entity.Measure.ParamTypeValue; +import fr.eurecom.senml.persistence.JDOStorage; + +@PersistenceCapable +@Inheritance(customStrategy = "complete-table") +public class SensorAdmin extends Sensor implements ISensorAdmin { + + @NotPersistent + private final static String ZONE_UPDATED = "ZoneUpdated"; + + @Persistent + boolean active = true; + + public SensorAdmin(String title, IZone zone, String baseUnit, String typeSensor) { + super(UUID.randomUUID().toString(), title, zone != null ? zone.getKey() + : null, baseUnit, typeSensor); + JDOStorage.getInstance().write(this); + } + + public SensorAdmin(String bn, String title, IZone zone) { + super(bn, title, zone != null ? zone.getKey() : null); + JDOStorage.getInstance().write(this); + } + + @Override + public void setTitle(String title) { + this.title = title; + JDOStorage.getInstance().write(this); + } + + @Override + public void destroy() { + try { + JDOStorage.getInstance().delete(getMeasures()); + } catch (Exception e) { + e.printStackTrace(); + } + + JDOStorage.getInstance().delete(this); + } + + @Override + public boolean linkZone(IZone zone) { + return zone == null ? false : linkZone(zone.getKey()); + } + + @Override + public boolean linkZone(String zoneKey) { + if (active && checkKey(zoneKey)) { + this.zoneKey = zoneKey; + JDOStorage.getInstance().write(this); + return active; + } + return false; + } + + @Override + public void unlinkZone() { + if (zoneKey != null) { + this.zoneKey = null; + JDOStorage.getInstance().write(this); + } + } + + @Override + public void deleteMeasure(String measureKey) { + if (measureKey != null) { + JDOStorage.getInstance() + .delete(JDOStorage.getInstance().getById(measureKey, + Measure.class)); + } + fr.eurecom.restlet.resources.common.ContactBaseResource + .sendUpdateSubscribers(ZONE_UPDATED + getZone().getName()); + } + + @Override + public void deleteMeasure(Measure measure) { + JDOStorage.getInstance().delete(measure); + fr.eurecom.restlet.resources.common.ContactBaseResource + .sendUpdateSubscribers(ZONE_UPDATED + getZone().getName()); + } + + private boolean checkKey(String key) { + return key != null && !key.trim().isEmpty(); + } + + @Override + public Measure addMeasure(String measureName, Units measureUnit, + ParamTypeValue measureTypeValue, String measureValue, + long timeMeasure, String type) { + + Measure m = new Measure(getUUID(), measureName, measureUnit, + measureTypeValue, measureValue, timeMeasure, type); + JDOStorage.getInstance().write(m); + + // fr.eurecom.restlet.resources.common.ContactBaseResource.sendUpdateSubscribers(ZONE_UPDATED + // + getZone().getName()); + GCMPushingServer GCM = GCMPushingServer.getInstance(); + try { + GCM.sendNotification(ZONE_UPDATED + getZone().getName()); + } catch (JSONException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + return m; + } + + @Override + public Measure addMeasure(String measureName, Units measureUnit, + ParamTypeValue measureTypeValue, String measureValue, + long timeMeasure, String type, String sum) { + + Measure m = new Measure(getUUID(), measureName, measureUnit, + measureTypeValue, measureValue, timeMeasure, type, sum); + JDOStorage.getInstance().write(m); + + // fr.eurecom.restlet.resources.common.ContactBaseResource.sendUpdateSubscribers(ZONE_UPDATED + // + getZone().getName()); + GCMPushingServer GCM = GCMPushingServer.getInstance(); + try { + GCM.sendNotification(ZONE_UPDATED + getZone().getName()); + } catch (JSONException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + return m; + } + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/SensorCheckAlarm.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/SensorCheckAlarm.java new file mode 100644 index 0000000000000000000000000000000000000000..e6b4ebd2f2ab2e686dce00fd6595973df1139e9d --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/SensorCheckAlarm.java @@ -0,0 +1,169 @@ +package fr.eurecom.senml.entity; + +import java.util.HashMap; +import java.util.List; +import java.util.logging.Logger; +import java.util.Map; + +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; + +import org.json.JSONObject; +import org.w3c.dom.Document; +import org.w3c.dom.Node; + +import com.google.appengine.api.datastore.Key; +import com.google.appengine.api.datastore.KeyFactory; + +import fr.eurecom.senml.persistence.JDOStorage; + +/** + * Represents the list of the monitored sensors, and the relative + * minimum and maximum values (allowed range) + */ +@PersistenceCapable(detachable = "true") +public class SensorCheckAlarm { + + public enum ParamTypeValue { + BOOL("bv"), FLOAT("v"), STRING("sv"); + + private String encoded; + private final static Map<String, ParamTypeValue> mTypeValue = new HashMap<String, ParamTypeValue>(); + + ParamTypeValue(String encoded) { + this.encoded = encoded; + } + + public String getSenML() { + return encoded; + } + + public static ParamTypeValue getParamTypeValue(String senmlEncoded) { + if (mTypeValue == null) { + initMapValue(); + } + return mTypeValue.get(senmlEncoded); + } + + private static synchronized void initMapValue() { + if (mTypeValue.size() > 0) { + return; + } + for (ParamTypeValue v : values()) { + mTypeValue.put(v.encoded, v); + } + } + }; + + @NotPersistent + private static final Logger log = Logger + .getLogger("sensors.entity.SensorCheckAlarm"); + @NotPersistent + public static final String PARAM_NAME = "n"; + @NotPersistent + public static final String PARAM_VALUE = "v"; + @NotPersistent + public static final String PARAM_STRING_VALUE = "sv"; + @NotPersistent + public static final String PARAM_RT_ALLOWED = "rt-allowed"; + + @PrimaryKey + @Persistent + Key key; + + @Persistent + private String sensorKey; + + @Persistent + private String sensorName; + + @Persistent + private String zoneKey; + + // This should be an Object, and then through reflection setted to the correct type + @Persistent + private float minVal; + + @Persistent + private float maxVal; + + public SensorCheckAlarm(String sensorKey, float minVal, float maxVal) { + this.sensorKey = sensorKey; + + this.key = KeyFactory.createKey(SensorCheckAlarm.class.getSimpleName(), + sensorKey + "sensoralarm"); + this.setMinVal(minVal); + this.setMaxVal(maxVal); + } + + public void destroy() { + JDOStorage.getInstance().delete(this); + } + + public String getKey() { + return key == null ? null : KeyFactory.keyToString(key); + } + + public JSONObject toJSONSenML() { + try { + JSONObject entrySenML = new JSONObject(); + + HashMap<String, Float> rtAllowed = new HashMap<String, Float>(); + rtAllowed.put("min", getMinVal()); + rtAllowed.put("max", getMaxVal()); + entrySenML.put("key", getKey()); + entrySenML.put("sensorkey", getSensorKey()); + entrySenML.put("sensorName", getParentSensorName()); + JSONObject rtAllowedJSON = new JSONObject(rtAllowed); + entrySenML.put(PARAM_RT_ALLOWED, rtAllowedJSON); + return entrySenML; + } catch (Exception e) { + e.printStackTrace(); + log.warning("Error Encoding a measure"); + return null; + } + } + + public Node toXMLSenML(Document d) { + //TODO + return null; + } + + public String getParentSensorName() { + SensorAdmin sensor = JDOStorage.getInstance().getById(sensorKey, SensorAdmin.class); + return sensor.getTitle(); + } + + public String toString() { + StringBuilder builder = new StringBuilder(); + builder.append("key: " + this.sensorKey).append(", [" + getMinVal() + ", ") + .append(getMaxVal() + "]"); + return builder.toString(); + } + + public String getSensorKey() { + return sensorKey; + } + + public static List<SensorCheckAlarm> getAllMonitoredSensors() { + return JDOStorage.getInstance().getAll(SensorCheckAlarm.class); + } + + public float getMinVal() { + return minVal; + } + + public void setMinVal(float minVal) { + this.minVal = minVal; + } + + public float getMaxVal() { + return maxVal; + } + + public void setMaxVal(float maxVal) { + this.maxVal = maxVal; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Units.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Units.java new file mode 100644 index 0000000000000000000000000000000000000000..cdc44e4db66549df0b79ecedacbd8d3c7c6e7474 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Units.java @@ -0,0 +1,115 @@ +package fr.eurecom.senml.entity; + +import java.util.HashMap; +import java.util.Map; + +/** + * Units Definition from UCUM organization. + * + * @author dalmassi + * + */ +public enum Units { + ACIDITY("acidity", "pH", "pH"), // senml 10 Nota : ucum c/s = [pH] + ACCELERATION("acceleration", "m/s2", "m/s2"), // senml 10 + AMPERE("electric current", "A", "A"), + AREA("Area", "m2", "m2"), // senml 10 + BATTERY_LEVEL_PERCENT("Remaining percent battery", "%EL", "%EL"), // senml 10 + BATTERY_LEVEL_SECOND("Remaining seconds battery", "EL", "EL"), // senml 10 + BAR("pressure", "bar", "bar"), + BEATS_PER_MINUTE("beats/minute", "beet/m", "beet/m"), // senml 10 says beet/m, but ucum m = meter, min = minutes. This is confusing, also beet looks mispelled. Check next draft version. + BEATS("Cumulative beats", "beets", "beets"), // senml 10. + BECQUEREL("becquerel", "bq", "bq"), + BEL_SOUND_LEVEL("bel sound pressure level", "Bspl", "Bspl"), // senml 10 + BIT_SECOND("bits per second", "bit/s", "bit/s"), // senml 10 + CANDELA("luminous intensity", "cd","cd"), + COUNT("count", "count", "count"), // senml 10 + COULOMB("electric charge", "C", "C"), + DAY("time", "d", "d"), + DEGREE_ANGLE("Plan Angle", "°", "deg"), + DEGREE_CELSIUS("temperature", "°C", "Cel"), + DEGREE_FAHRENHEIT("temperature","°F", "[degF]"), + FARAD("electric capacitance", "F", "F"), + GRAM("mass", "g", "g"), + GRAM_PER_LITER("gram per liter", "g/L", "g/L"), + GRAY("gray", "Gy", "Gy"), // senml 10 + KATAL("katal", "kat", "kat"), // senml 10 + KELVIN("temperature", "K", "K"), + HENRY("henry", "H", "H"), // senml 10 + HERTZ("hertz", "Hz", "Hz"), + HUMIDITY_RELATIVE("humidity", "%RH", "%RH"), // senml 10 + HOUR ("time", "h", "h"), + IRRIDIANCE("irridiance", "W/m2", "W/m2"), // senml 10 + JOULE("joule", "J", "J"), + KILO_GRAM("mass", "kg", "kg"), // senml 10 + KILO_WATT_HOUR("kilo watt per hour", "kWh", "kW/h"), + LATITUDE("WGS84-Latitude", "lat", "lat"), // draft senml 10 + LITER("volume", "l", "l"), + LITER_SECOND("flow liters per seconds", "l/s", "l/s"), // senml 10 Nota ucum = L/s + LUMEN("luminous flux", "lm", "lm"), + LUMINANCE("luminance", "cd/m2", "cd/m2"), // senml 10 + LONGITUDE("WGS84-Longitude", "lon", "lon"), // draft senml 10 + LUX("illuminance", "lx", "lx"), + METER("length", "m", "m"), + MINUTE_ANGLE("Plan angle", "′", "′"), + MINUTE_TIME("time", "min", "min"), + MOLE("mole", "mol", "mol"), // senml 10 + NEWTON("newton", "N", "N"), // senml 10 + OHM("electric resistance", "Ω","Ohm"), + PASCAL("pressure", "Pa", "Pa"), // draft senml 10 + PERCENT("percent", "%", "UCUM:%"), // ucum + RADIAN("plane angle", "rad", "rad"), + SECOND_ANGLE("Plan Angle", "″", "″"), + SECOND_TIME("time", "s", "s"), + SIEMENS("Siemens", "S", "S"), // senml 10 + SIEVERT("Sievert", "Sv", "Sv"), // senml 10 + STERADIAN("steradian", "sr", "sr"), // senml 10 + SWITCH("switch", "switch", "%"), // senml 10 conflict with ucum percent + TONNE("mass", "t", "t"), + TESLA("magnetic flux density", "T", "T"), + VELOCITY("velocity", "m/s", "m/s"), + VOLT("electric potential", "V", "V"), + WATT("power", "W", "W"), + WEBER("wever", "Wb", "Wb"), // senml 10 + YEAR("time", "a", "a"); + + private final String print; + private final String cs; + private final String kindOfQty; + + private final static Map<String, Units> csMap = new HashMap<String, Units>(); + + Units(String kind, String print, String cs) { + this.print = print; + this.cs = cs; + this.kindOfQty = kind; + } + + public String getCS() { + return cs; + } + + public String getKindofQty() { + return kindOfQty; + } + + public String getPrint() { + return print; + } + + public static Units getUnit(String cs) { + if (csMap.size() == 0) { + initCsMap(); + } + return csMap.get(cs); + } + + private static synchronized void initCsMap() { + if (csMap.size() > 0 ) { + return; + } + for (Units u : values()) { + csMap.put(u.cs, u); + } + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Zone.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Zone.java new file mode 100644 index 0000000000000000000000000000000000000000..7daba277d62aa81d1d552ff4ee03e001816cacf8 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/Zone.java @@ -0,0 +1,104 @@ +package fr.eurecom.senml.entity; + +import java.util.List; +import java.util.logging.Level; +import java.util.logging.Logger; + +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; +import javax.jdo.annotations.PrimaryKey; + +import com.google.appengine.api.datastore.Key; +import com.google.appengine.api.datastore.KeyFactory; + +import fr.eurecom.senml.persistence.JDOStorageZone; + +@PersistenceCapable(detachable="true") +public class Zone implements IZone { + @NotPersistent + private static final Logger log = Logger.getLogger("Zone"); + + @Persistent + @PrimaryKey + protected Key key; + + @Persistent + protected String name = ""; + + @Persistent + protected String parentKey = null; + + @Persistent + protected String zoneType = ""; + +/** + * Instantiates a zone. + * @param key Zone key + * @param name Zone name + * @param parentKey Zone parent key. Can be null. + * @param type + */ + protected Zone(Key key, String name, String parentKey, String type) { + this.key = key; + this.name = name; + this.parentKey = parentKey; + this.zoneType = type; + } + + @Override + public List<IZone> getSubZones() { + long before = System.currentTimeMillis(); + + List<IZone> childs = JDOStorageZone.getInstance().getSubZones(this); + + long after = System.currentTimeMillis(); + log.info("getSubzone " + name + " taken ms:" + (after - before)); + return childs; + } + + @Override + public List<ISensor> getSensors() { + long before = System.currentTimeMillis(); + + List<ISensor> sensors = JDOStorageZone.getInstance().getSensors(this); + + long after = System.currentTimeMillis(); + log.info("getSensors " + name + " taken ms:" + (after - before)); + return sensors; + } + + @Override + public String getName() { + return name; + } + + @Override + public String getKey() { + return KeyFactory.keyToString(key); + } + + @Override + public String getType() { + return zoneType; + } + + @Override + public int compareTo(IZone o) { + return this.name.compareTo(o.getName()); + } + + @Override + public IZone getParentZone() { + Zone p = null; + if (parentKey != null) { + try { + p = JDOStorageZone.getInstance().getById(parentKey, this.getClass()); + } + catch (Exception e) { + log.log(Level.WARNING, "Error retrieving parent zone", e); + } + } + return p; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ZoneAdmin.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ZoneAdmin.java new file mode 100644 index 0000000000000000000000000000000000000000..2f08c22b9b249cb4e4fd2dd0c8356812a1f8b0b0 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/entity/ZoneAdmin.java @@ -0,0 +1,86 @@ +package fr.eurecom.senml.entity; + +import java.util.Collections; +import java.util.List; +import java.util.SortedSet; +import java.util.logging.Logger; + +import javax.jdo.annotations.Index; +import javax.jdo.annotations.Inheritance; +import javax.jdo.annotations.NotPersistent; +import javax.jdo.annotations.PersistenceCapable; +import javax.jdo.annotations.Persistent; + +import com.google.appengine.api.datastore.Key; +import com.google.appengine.api.datastore.KeyFactory; + +import fr.eurecom.senml.persistence.JDOStorage; +import fr.eurecom.senml.persistence.JDOStorageZone; + +/** + * Not thread-safe. + * + * @author dalmassi + * + */ +@PersistenceCapable +@Inheritance(customStrategy = "complete-table") +// @Index(name="index_parentKey", members={"parentKeyString"}) +public class ZoneAdmin extends Zone implements IZoneAdmin { + @NotPersistent + private static final Logger log = Logger.getLogger("Zone"); + + @Persistent + private boolean isActive = true; + + public ZoneAdmin(String name, ZoneAdmin parent, String type) { + super(KeyFactory.createKey(ZoneAdmin.class.getSimpleName(), name), + name, parent == null ? null : parent.getKey(), type); + JDOStorage.getInstance().write(this); + } + + @Override + public void destroy() { + isActive = false; + parentKey = null; + + List<ISensorAdmin> sensors = JDOStorageZone.getInstance().getSensors( + this); + + for (ISensorAdmin s : sensors) { + s.destroy(); + } + + JDOStorage.getInstance().delete(this); + // TODO: Possible Race condition here. The delete is not effective when + // method JDOStorage delete returns. + log.fine("Removed from database " + this.name); + } + + @Override + public boolean linkParentZone(String parentZoneKey) { + if (isActive && checkKey(parentZoneKey)) { + this.parentKey = parentZoneKey; + JDOStorage.getInstance().write(this); + return isActive; + } + return false; + } + + @Override + public boolean linkParentZone(IZone parentZone) { + return parentZone == null ? false : linkParentZone(parentZone.getKey()); + } + + @Override + public void unlinkParentZone() { + if (parentKey != null) { + parentKey = null; + JDOStorage.getInstance().write(this); + } + } + + private boolean checkKey(String key) { + return key != null && !key.trim().isEmpty(); + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/jspcontroller/ControllerJsp.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/jspcontroller/ControllerJsp.java new file mode 100644 index 0000000000000000000000000000000000000000..676933e99a7d9504cf6ba2ee6c9cb8acb08ed8ef --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/jspcontroller/ControllerJsp.java @@ -0,0 +1,285 @@ +package fr.eurecom.senml.jspcontroller; + +import java.util.Calendar; +import java.util.List; +import java.util.TimeZone; +import java.util.concurrent.TimeUnit; + +import javax.servlet.http.HttpServletRequest; + +import org.springframework.stereotype.Controller; +import org.springframework.ui.ModelMap; +import org.springframework.web.bind.annotation.ModelAttribute; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestMethod; +import org.springframework.web.servlet.ModelAndView; + +import fr.eurecom.pushing.GCMPushingServer; +import fr.eurecom.senml.entity.IPostVal; +import fr.eurecom.senml.entity.IZone; +import fr.eurecom.senml.entity.Measure; +import fr.eurecom.senml.entity.PostValAdmin; +import fr.eurecom.senml.entity.PostValType; +import fr.eurecom.senml.entity.Sensor; +import fr.eurecom.senml.entity.SensorAdmin; +import fr.eurecom.senml.entity.SensorCheckAlarm; +import fr.eurecom.senml.entity.Units; +import fr.eurecom.senml.entity.Zone; +import fr.eurecom.senml.entity.ZoneAdmin; +import fr.eurecom.senml.persistence.JDOStorage; + +@Controller +@RequestMapping("/") +public class ControllerJsp { + + @RequestMapping(value = "/zones", method = RequestMethod.GET) +// @ModelAttribute + public String listZone(ModelMap model) { + List<IZone> zones = JDOStorage.getInstance().getAll(ZoneAdmin.class); + model.addAttribute("zoneList", zones); + return "zonelist"; + } + + @RequestMapping(value="/addZone", method = RequestMethod.POST) + public ModelAndView addZone(HttpServletRequest request) { + String name = request.getParameter("nameZone"); + String nameParent = request.getParameter("parentNameZone"); + String type = request.getParameter("typeZone"); + + ZoneAdmin parent = nameParent.isEmpty() + ? null + :JDOStorage.getInstance().getById(nameParent, ZoneAdmin.class); + + new ZoneAdmin(name, parent, type); + return new ModelAndView("redirect:zones"); + } + + @RequestMapping(value = "deleteZone/{zoneKey}", method = RequestMethod.GET) + public ModelAndView deleteZone(@PathVariable String zoneKey, HttpServletRequest request) { + + ZoneAdmin z = JDOStorage.getInstance().getById(zoneKey, ZoneAdmin.class); + if (z != null) { + z.destroy(); + } + return new ModelAndView("redirect:../zones"); + } + + + @RequestMapping(value="{zoneKey}/edit", method = RequestMethod.GET) + public String editZone(@PathVariable String zoneKey, HttpServletRequest request, + ModelMap model) { + ZoneAdmin z = JDOStorage.getInstance().getById(zoneKey, ZoneAdmin.class); + model.addAttribute("zone", z); + return "detailzone"; + } + + + @RequestMapping(value="{zoneKey}/addsubZone", method = RequestMethod.POST) + public String addSubzone(@PathVariable String zoneKey, HttpServletRequest request, ModelMap model) { + String subZone = request.getParameter("nameSubZone"); + String typeZone = request.getParameter("typeZone"); + + ZoneAdmin zone = JDOStorage.getInstance().getById( + zoneKey, ZoneAdmin.class); + + new ZoneAdmin(subZone, zone, typeZone); + model.addAttribute("zone", zone); + return "redirect:edit"; + } + + @RequestMapping(value="{zoneKey}/addSensor", method = RequestMethod.POST) + public String addSensor(@PathVariable String zoneKey, HttpServletRequest request, ModelMap model) { + String title = request.getParameter("title"); + String baseUnit = request.getParameter("baseunit"); + String typeSensor = request.getParameter("typeSensor"); + ZoneAdmin za = JDOStorage.getInstance().getById(zoneKey, ZoneAdmin.class); + new SensorAdmin(title, za, baseUnit, typeSensor); + + model.addAttribute("zone", za); + return "redirect:edit"; + } + + @RequestMapping(value= "{zoneKey}/deleteSensor/{sensorKey}", method = RequestMethod.GET) + public ModelAndView deleteSensor(@PathVariable String zoneKey, + @PathVariable String sensorKey, HttpServletRequest request, ModelMap model) { + + SensorAdmin s = JDOStorage.getInstance().getById(sensorKey, SensorAdmin.class); + s.destroy(); + + return new ModelAndView("redirect:../edit"); + } + + + @RequestMapping(value="{zoneKey}/delete/{subZoneKey}", method = RequestMethod.GET) + @ModelAttribute + public ModelAndView deleteZone(@PathVariable String zoneKey, @PathVariable String subZoneKey, + HttpServletRequest request, ModelMap model) { + ZoneAdmin z = JDOStorage.getInstance().getById(subZoneKey, ZoneAdmin.class); + if (z != null) { + z.destroy(); + } + + return new ModelAndView("redirect:../edit"); + } + + @RequestMapping(value="{zoneKey}/addMeasure", method = RequestMethod.POST) + public String addMeasure(@PathVariable String zoneKey, + HttpServletRequest request, ModelMap model) { + + // Mandatory fields + String sensorKey = request.getParameter("sensorKey"); + String unit = request.getParameter("unit"); + String name = request.getParameter("name"); + String typeValue = request.getParameter("typeValue"); + String value = request.getParameter("value"); + long time = getSecondsSince1970UTC(request.getParameter("time")); + String strVersion = request.getParameter("version"); + // Optional values + String sumValue = request.getParameter("sum"); + String specMeasure = request.getParameter("specmeasure"); + if (specMeasure != null) System.out.println("spec measure: " + specMeasure.toString()); + String baseUnit = request.getParameter("bu"); + String allowed = request.getParameter("allowed"); + String rtallowed = request.getParameter("rtallowed"); + String type = request.getParameter("measureType"); + + // If version is not setted, automatically is 1 + if (strVersion == null) { + strVersion = "1.0"; + } + + float version = Float.parseFloat(strVersion); + + SensorAdmin sa = JDOStorage.getInstance().getById(sensorKey, SensorAdmin.class); + + if (sumValue == null) { + if ((specMeasure == null) || (specMeasure.equalsIgnoreCase("off"))) { + System.out.println("using baseUnit " + baseUnit); + sa.addMeasure( + name, + null, + Measure.ParamTypeValue.valueOf(typeValue.toUpperCase()), + value, time, type); + } else { + System.out.println("specMeasure is " + specMeasure); + sa.addMeasure( + name, + Units.getUnit(unit), + Measure.ParamTypeValue.valueOf(typeValue.toUpperCase()), + value, time, type); + } + } else { + if ((specMeasure == null) || (specMeasure.equalsIgnoreCase("off"))) { + System.out.println("using baseUnit " + baseUnit); + sa.addMeasure( + name, + null, + Measure.ParamTypeValue.valueOf(typeValue.toUpperCase()), + value, time, type, sumValue); + } else { + System.out.println("specMeasure is " + specMeasure); + sa.addMeasure( + name, + Units.getUnit(unit), + Measure.ParamTypeValue.valueOf(typeValue.toUpperCase()), + value, time, type, sumValue); + } + + } + + model.addAttribute("zone", sa.getZone()); + // Eventually notify the registered devices + + return "redirect:edit"; + } + + @RequestMapping(value="{zoneKey}/addAlertMonitor", method = RequestMethod.POST) + public String addAlertMonitor(@PathVariable String zoneKey, + HttpServletRequest request, ModelMap model) { + + // Mandatory fields + String sensorKey = request.getParameter("sensorKey"); + float rtallowedmin = Float.parseFloat(request.getParameter("rtallowedmin")); + float rtallowedmax = Float.parseFloat(request.getParameter("rtallowedmax")); + + ZoneAdmin zone = JDOStorage.getInstance().getById(zoneKey, ZoneAdmin.class); + Sensor sensor = JDOStorage.getInstance().getById(sensorKey, SensorAdmin.class); + System.out.println("Adding monitoring for " + zone.getName() + ", " + sensor.getTitle()); + + SensorCheckAlarm checkAlarm = new SensorCheckAlarm(sensorKey, rtallowedmin, rtallowedmax); + JDOStorage.getInstance().write(checkAlarm); + return "redirect:edit"; + } + + // Admin website version - TODO, filter for sensors/zones + @RequestMapping(value="{zoneKey}/listAlertsMonitor", method = RequestMethod.GET) + public String listAlertsMonitor(@PathVariable String zoneKey, + HttpServletRequest request, ModelMap model) { + //TODO IF NEEDED!!!! + return "alarms"; + } + + // Admin website version for all the monitors + @RequestMapping(value="/listAllAlertsMonitorJSON", method = RequestMethod.GET) + public String listAllAlertsMonitorJSON(HttpServletRequest request, ModelMap model) { + return "allalarmsJSON"; + } + + // Admin mobile app version + @RequestMapping(value="{zoneKey}/listAlertsMonitorJSON", method = RequestMethod.GET) + public String listAlertsMonitorJSON(@PathVariable String zoneKey, + HttpServletRequest request, ModelMap model) { + return "alarmsJSON"; + } + + @RequestMapping(value="{zoneKey}/removemonitor", method = RequestMethod.GET) + public String removeMonitor(@PathVariable String zoneKey, + HttpServletRequest request, ModelMap model) { + String sensorKey = request.getParameter("key"); + SensorCheckAlarm s = JDOStorage.getInstance().getById(sensorKey, SensorCheckAlarm.class); + JDOStorage.getInstance().delete(s); + return "alarms"; + } + + @RequestMapping(value="{zoneKey}/getmonitor", method = RequestMethod.GET) + public String getMonitor(@PathVariable String zoneKey, + HttpServletRequest request, ModelMap model) { + return "sensormonitorsjson"; + } + + /** + * To respect the senML directives, we need seconds since 1970 UTC + * @param parameter + * @return + */ + private long getSecondsSince1970UTC(String parameter) { + Calendar calendar = Calendar.getInstance(TimeZone.getTimeZone("UTC")); + System.out.println("Calendar is: " + calendar); + int secondsSinceEpoch = (int) TimeUnit.MILLISECONDS.toSeconds(calendar.getTimeInMillis()); + System.out.println("seconds are " + secondsSinceEpoch); + return secondsSinceEpoch; + } + + @RequestMapping(value= "{zoneKey}/deleteMeasure/{keyMeasure}", + method = RequestMethod.GET) + public ModelAndView deleteMeasure(@PathVariable String zoneKey, @PathVariable String keyMeasure, + HttpServletRequest request, ModelMap model) { + + Measure m = JDOStorage.getInstance().getById(keyMeasure, Measure.class); +// m.destroy(); + SensorAdmin sa = JDOStorage.getInstance().getById(m.getSensorKey(), SensorAdmin.class); + sa.deleteMeasure(m); + return new ModelAndView("redirect:../edit"); + } + + @RequestMapping(value = "/postvals", method = RequestMethod.GET) + public String listPostVals(ModelMap model) { + List<PostValType> zones = JDOStorage.getInstance().getAll(PostValType.class); + model.addAttribute("postvalsList", zones); + return "postvalslist"; + } +// +// @RequestMapping(value = "/alertsadmin", method = Request.GET) +// public String listMonitoredSensors() +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/jspcontroller/ZoneTypes.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/jspcontroller/ZoneTypes.java new file mode 100644 index 0000000000000000000000000000000000000000..30738fb6659039a631e45b57d871d83b27f8c269 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/jspcontroller/ZoneTypes.java @@ -0,0 +1,14 @@ +package fr.eurecom.senml.jspcontroller; + +public class ZoneTypes { + + // Refers to + // http://www.sensormeasurement.appspot.com/json/m3/featureOfInterest + private static String[] zoneTypes = {"City", "Environment", "Military", + "Tracking", "Health", "Place", "Building", "Transportation", + "BrainWave"}; + + public static String[] getZoneTypes() { + return zoneTypes; + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/persistence/JDOStorage.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/persistence/JDOStorage.java new file mode 100644 index 0000000000000000000000000000000000000000..94008919598ff1cf659f50943d98246d3b127a42 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/persistence/JDOStorage.java @@ -0,0 +1,167 @@ +package fr.eurecom.senml.persistence; + +import java.util.Collection; +import java.util.Collections; +import java.util.List; + +import javax.jdo.PersistenceManager; +import javax.jdo.Query; + +import fr.eurecom.restlet.resources.persistence.PMF; +import fr.eurecom.senml.entity.Zone; +import fr.eurecom.senml.entity.ZoneAdmin; + +/** + * This class is a jdo wrapper. Implementing the persistence. + * @author dalmassi + * + */ +public class JDOStorage { + private static final JDOStorage instance = new JDOStorage(); + + protected JDOStorage() { + + } + + public static JDOStorage getInstance() { + return instance; + } + + @SuppressWarnings("unchecked") +/** + * Wrapper JDO PM.getById, return a detached object. + * @param key The String key + * @param c Class of the object + * @return a typed object of the class. + */ + public <T> T getById(String key, Class<T> c) { + PersistenceManager pm = PMF.get().getPersistenceManager(); + Object attached, detached = null; + + try { + attached = pm.getObjectById(c, key); + detached = pm.detachCopy(attached); + } + finally { + close(pm); + } + return (T) detached; + } + + public <T> void refresh(String key, Class<T> c) { + PersistenceManager pm = getPM(); + + try { + + Object attached = pm.getObjectById(c, key); + pm.refresh(attached); + } + catch (Exception e) { + e.printStackTrace(); + } + } + + + Query q = null; + + @SuppressWarnings("unchecked") + public <T> List<T> getAll(Class<?> c) { + List<T> detached = Collections.EMPTY_LIST; + PersistenceManager pm = getPM(); + Query q = pm.newQuery(c); + + try { + List<T> results = (List<T>) q.execute(); + if (!results.isEmpty()) { + detached = (List<T>) pm.detachCopyAll(results); + } + close(q); + close(pm); + } + catch (Exception e) { + e.printStackTrace(); + } + return detached; + } + +/** + * Save an object to dataStore. + * @param z object to save + * @return true if ok, false otherwise. + */ + public <T> boolean write(T z) { + PersistenceManager pm = getPM(); + try { + pm.makePersistent(z); +// System.out.println("write jdo done! zone name|" + z.getName() + "| key |" + z.getKey() + "|"); + return true; + } catch (Exception e) { + e.printStackTrace(); + return false; + } + finally { + close(pm); + } + } + + public <T> void delete(T z) { + PersistenceManager pm = getPM(); + try { + pm.deletePersistent(z); + } + catch (Exception e) { + e.printStackTrace(); + } + finally { + close(pm); + } + } + + public <T> void delete(Collection<T> z) { + PersistenceManager pm = getPM(); + try { + pm.deletePersistentAll(z); + } + catch (Exception e) { + e.printStackTrace(); + } + finally { + close(pm); + } + } + + public long deleteAll(Class<?> c) { + PersistenceManager pm = getPM(); + long count = -1; + try { + Query q = pm.newQuery(c); + count = q.deletePersistentAll(); + } + catch (Exception e) { + e.printStackTrace(); + } + finally { + close(q); + close(pm); + } + return count; + } + + + protected PersistenceManager getPM() { + return PMF.get().getPersistenceManager(); + } + + protected void close(PersistenceManager pm) { + if (pm != null) { + pm.close(); + } + } + + protected void close(Query q) { + if (q != null) { + q.closeAll(); + } + } + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/persistence/JDOStorageZone.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/persistence/JDOStorageZone.java new file mode 100644 index 0000000000000000000000000000000000000000..627c95dfc9c8d0ad08aa0aa5e0d97c0e7a1cff40 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/fr/eurecom/senml/persistence/JDOStorageZone.java @@ -0,0 +1,115 @@ +package fr.eurecom.senml.persistence; + +import java.util.Collections; +import java.util.List; + +import javax.jdo.PersistenceManager; +import javax.jdo.Query; +import fr.eurecom.senml.entity.Measure; +import fr.eurecom.senml.entity.Sensor; +import fr.eurecom.senml.entity.SensorAdmin; +import fr.eurecom.senml.entity.Zone; +import fr.eurecom.senml.entity.ZoneAdmin; + + + +public class JDOStorageZone extends JDOStorage { + private static final JDOStorageZone instance = new JDOStorageZone(); + private Query Q_SUBZONE_ADMIN = null; + private Query Q_SENSOR_ADMIN = null; + private Query Q_MEASURE_ADMIN = null; + + + private JDOStorageZone() { + super(); + createQuerySubZone(); + createQueryZoneSensors(); + createQueryMeasures(); + } + + + public static JDOStorageZone getInstance() { + return instance; + } + + + @SuppressWarnings("unchecked") + public <T> List<T> getSubZones(Zone zone) { + List<T> detached = Collections.EMPTY_LIST; + try { + List<T> results = (List<T>) Q_SUBZONE_ADMIN.execute(zone.getKey()); + if (!results.isEmpty()) { + detached = (List<T>) Q_SUBZONE_ADMIN.getPersistenceManager().detachCopyAll(results); + } + } + catch (Exception e) { + e.printStackTrace(); + } + return detached; + } + + @SuppressWarnings("unchecked") + public <T> List<T> getSensors(Zone zone) { + List<T> detached = Collections.EMPTY_LIST; + try { + List<T> results = (List<T>) Q_SENSOR_ADMIN.execute(zone.getKey()); + if (!results.isEmpty()) { + detached = (List<T>) Q_SENSOR_ADMIN.getPersistenceManager().detachCopyAll(results); + } + } + catch (Exception e) { + e.printStackTrace(); + } + return detached; + } + + @SuppressWarnings("unchecked") + public List<Measure> getMeasures(Sensor sensor) { + List<Measure> detached = Collections.EMPTY_LIST; + try { + List<Measure> results = (List<Measure>) Q_MEASURE_ADMIN.execute(sensor.getUUID()); + if (!results.isEmpty()) { + detached = (List<Measure>) Q_MEASURE_ADMIN.getPersistenceManager().detachCopyAll(results); + } + } + catch (Exception e) { + e.printStackTrace(); + } + return detached; + } + + + private void createQuerySubZone() { + if (Q_SUBZONE_ADMIN == null) { + Q_SUBZONE_ADMIN = getPM().newQuery(ZoneAdmin.class); + Q_SUBZONE_ADMIN.setFilter("parentKey == keyParentParam"); +// Q_SUBZONE.declareParameters(Key.class.getName() + " keyParentParam"); + Q_SUBZONE_ADMIN.declareParameters("String keyParentParam"); + Q_SUBZONE_ADMIN.compile(); + } + } + + private void createQueryZoneSensors() { + if (Q_SENSOR_ADMIN == null) { + PersistenceManager pm = getPM(); +// pm.getFetchPlan().setDetachmentOptions( FetchPlan.DETACH_LOAD_FIELDS | FetchPlan.DETACH_UNLOAD_FIELDS); +// pm.getFetchPlan().setMaxFetchDepth(4); + + Q_SENSOR_ADMIN = pm.newQuery(SensorAdmin.class); + Q_SENSOR_ADMIN.setFilter("zoneKey == zoneKeyParam"); + Q_SENSOR_ADMIN.declareParameters("String zoneKeyParam"); + Q_SENSOR_ADMIN.compile(); + } + } + + private void createQueryMeasures() { + if (Q_MEASURE_ADMIN == null) { + PersistenceManager pm = getPM(); + Q_MEASURE_ADMIN = pm.newQuery(Measure.class); + Q_MEASURE_ADMIN.setFilter("sensorKey == sensorKeyParam"); + Q_MEASURE_ADMIN.declareParameters("String sensorKeyParam"); + Q_MEASURE_ADMIN.compile(); + Q_MEASURE_ADMIN.setIgnoreCache(true); + } + } +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/log4j.properties b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/log4j.properties new file mode 100644 index 0000000000000000000000000000000000000000..d9c3edc9feeb5644e707aa6fc7f87d0f5620a19f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/src/log4j.properties @@ -0,0 +1,24 @@ +# A default log4j configuration for log4j users. +# +# To use this configuration, deploy it into your application's WEB-INF/classes +# directory. You are also encouraged to edit it as you like. + +# Configure the console as our one appender +log4j.appender.A1=org.apache.log4j.ConsoleAppender +log4j.appender.A1.layout=org.apache.log4j.PatternLayout +log4j.appender.A1.layout.ConversionPattern=%d{HH:mm:ss,SSS} %-5p [%c] - %m%n + +# tighten logging on the DataNucleus Categories +log4j.category.DataNucleus.JDO=WARN, A1 +log4j.category.DataNucleus.Persistence=WARN, A1 +log4j.category.DataNucleus.Cache=WARN, A1 +log4j.category.DataNucleus.MetaData=WARN, A1 +log4j.category.DataNucleus.General=WARN, A1 +log4j.category.DataNucleus.Utility=WARN, A1 +log4j.category.DataNucleus.Transaction=WARN, A1 +log4j.category.DataNucleus.Datastore=WARN, A1 +log4j.category.DataNucleus.ClassLoading=WARN, A1 +log4j.category.DataNucleus.Plugin=WARN, A1 +log4j.category.DataNucleus.ValueGeneration=WARN, A1 +log4j.category.DataNucleus.Enhancer=WARN, A1 +log4j.category.DataNucleus.SchemaTool=WARN, A1 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/appengine-generated/datastore-indexes-auto.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/appengine-generated/datastore-indexes-auto.xml new file mode 100644 index 0000000000000000000000000000000000000000..fd44d7144bab1ddd2921511fd0c947ac6710ab5d --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/appengine-generated/datastore-indexes-auto.xml @@ -0,0 +1,4 @@ +<!-- Indices written at Mon, 16 Sep 2013 09:25:09 UTC --> + +<datastore-indexes/> + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/appengine-web.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/appengine-web.xml new file mode 100644 index 0000000000000000000000000000000000000000..479f85751df1b1a8ddd2c0424bad470402fffc0c --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/appengine-web.xml @@ -0,0 +1,33 @@ +<?xml version="1.0" encoding="utf-8"?> +<appengine-web-app xmlns="http://appengine.google.com/ns/1.0"> + <application>wlboxv2</application> + <version>2</version> + + <!-- + Allows App Engine to send multiple requests to one instance in parallel: + --> + <threadsafe>true</threadsafe> + + <!-- Configure java.util.logging --> + <system-properties> + <property name="java.util.logging.config.file" value="WEB-INF/logging.properties"/> + </system-properties> + + <warmup-requests-enabled>true</warmup-requests-enabled> + <!-- + HTTP Sessions are disabled by default. To enable HTTP sessions specify: + + <sessions-enabled>true</sessions-enabled> + + It's possible to reduce request latency by configuring your application to + asynchronously write HTTP session data to the datastore: + + <async-session-persistence enabled="true" /> + + With this feature enabled, there is a very small chance your app will see + stale session data. For details, see + http://code.google.com/appengine/docs/java/config/appconfig.html#Enabling_Sessions + --> + + +</appengine-web-app> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/backends.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/backends.xml new file mode 100644 index 0000000000000000000000000000000000000000..8dc317be17bdcc962f427bd8604ec49462a42c45 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/backends.xml @@ -0,0 +1,10 @@ +<backends> + <backend name="back"> + <class>B1</class> + <instances>1</instances> + <options> + <dynamic>false</dynamic> + <public>true</public> + </options> + </backend> +</backends> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-api-1.0-sdk-1.7.4.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-api-1.0-sdk-1.7.4.jar new file mode 100644 index 0000000000000000000000000000000000000000..244505a1cf38bdc6c2ab5627366a1fa941bf56d5 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-api-1.0-sdk-1.7.4.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-api-labs.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-api-labs.jar new file mode 100644 index 0000000000000000000000000000000000000000..1fe9f839e45a135abe45fcecb3db3071e6f8213a Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-api-labs.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-endpoints.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-endpoints.jar new file mode 100644 index 0000000000000000000000000000000000000000..5bbba4a3ef9d920c7ab62cbe6054760413feec15 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-endpoints.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-jsr107cache-1.7.4.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-jsr107cache-1.7.4.jar new file mode 100644 index 0000000000000000000000000000000000000000..f3d111c63fdf8838b185a4a7397ab2f40caa4f73 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-jsr107cache-1.7.4.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-webapis.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-webapis.jar new file mode 100644 index 0000000000000000000000000000000000000000..ce00b7ea4262389d32720ed2cd1aeec79b1e426e Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/appengine-webapis.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/asm-4.0.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/asm-4.0.jar new file mode 100644 index 0000000000000000000000000000000000000000..cca0d9cebf10f36c1f377a0f5534a8728f529a6b Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/asm-4.0.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/commons-logging-api-1.1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/commons-logging-api-1.1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..bd4511684badd0cc710df5169b98e7b8f351858e Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/commons-logging-api-1.1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-api-jdo-3.1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-api-jdo-3.1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..9f3aaf5e971606a291b8107d4aee67a8cfd2eb90 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-api-jdo-3.1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-api-jpa-3.1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-api-jpa-3.1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..15b9cf68b62f350c5e02f5fedbe2ad47de13ed5c Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-api-jpa-3.1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-appengine-2.1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-appengine-2.1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..e1a40c1bb8e85c15e931c08716e3080bdeae53b7 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-appengine-2.1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-core-3.1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-core-3.1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..f8aac6aa323c184255f0dcdc24c7ed4c3b46b414 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/datanucleus-core-3.1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/gcm-server.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/gcm-server.jar new file mode 100644 index 0000000000000000000000000000000000000000..41264c809503f122f5080cb4fbe8b2750b3dce9e Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/gcm-server.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/geronimo-jpa_2.0_spec-1.0.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/geronimo-jpa_2.0_spec-1.0.jar new file mode 100644 index 0000000000000000000000000000000000000000..2ea9ca9c102edee64b2ae0038342ac22db21bbcb Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/geronimo-jpa_2.0_spec-1.0.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jdo-api-3.0.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jdo-api-3.0.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..6318d329ca9fc71037af1c59d1c8b4723f9c2795 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jdo-api-3.0.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/json_simple-1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/json_simple-1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..f395f41471a796876278a6c5667ded4632546893 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/json_simple-1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jsr107cache-1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jsr107cache-1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..a94aa3184e039f6b7ee503dc24a0598e858e5c86 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jsr107cache-1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jta-1.1.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jta-1.1.jar new file mode 100644 index 0000000000000000000000000000000000000000..7736ec9f08c2177f0068923020138bcd57d14f41 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/jta-1.1.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.json.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.json.jar new file mode 100644 index 0000000000000000000000000000000000000000..543438992fab3d820a1d900d65537347be419d21 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.json.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.json.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.json.jar new file mode 100644 index 0000000000000000000000000000000000000000..49b2873525c43f14d4852c8ed50ccb5fe455b32f Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.json.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.servlet.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.servlet.jar new file mode 100644 index 0000000000000000000000000000000000000000..5df964798b6407def1cdcca6a4d377767c556ce9 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.servlet.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.wadl.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.wadl.jar new file mode 100644 index 0000000000000000000000000000000000000000..5a96ccac00d0b35b175060368f5d5115a5093753 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.wadl.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.xml.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.xml.jar new file mode 100644 index 0000000000000000000000000000000000000000..213199baaa7a5e96c2cb62df5c4e324f4eea9aa1 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.ext.xml.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.jar new file mode 100644 index 0000000000000000000000000000000000000000..32a3c8483f275be2fae7d2880af54d473564edc7 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.restlet.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.asm-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.asm-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..20d79385104cdb8b3fd88ab83e417c1ff9f0c289 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.asm-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.beans-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.beans-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..a69bcb1f3dab93803c9fb9cc9d3c4dba319640a6 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.beans-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.context-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.context-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..a35e486989fc422b5acf528f29d5dd6c5fade129 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.context-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.core-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.core-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..bdd8944a727adc1e4df454d0d54d005311734551 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.core-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.expression-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.expression-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..0e445b97fa30a085f78a6208c67095dcc1d9729d Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.expression-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.web-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.web-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..42ed92c7c2cae23ccf4124f14ed63a04b8a70e8d Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.web-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.web.servlet-3.1.1.RELEASE.jar b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.web.servlet-3.1.1.RELEASE.jar new file mode 100644 index 0000000000000000000000000000000000000000..8e27a5bc7e073af91224d384445e7189c5fd8ed7 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/lib/org.springframework.web.servlet-3.1.1.RELEASE.jar differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/logging.properties b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/logging.properties new file mode 100644 index 0000000000000000000000000000000000000000..66e9625256cef316eae64c361d8d11bdd59730e5 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/logging.properties @@ -0,0 +1,39 @@ +# A default java.util.logging configuration. +# (All App Engine logging is through java.util.logging by default). +# +# To use this configuration, copy it into your application's WEB-INF +# folder and add the following to your appengine-web.xml: +# +# <system-properties> +# <property name="java.util.logging.config.file" value="WEB-INF/logging.properties"/> +# </system-properties> +# + +# Set the default logging level for all loggers to WARNING +#.level = DEBUG + +# A default log4j configuration for log4j users. +# +# To use this configuration, deploy it into your application's WEB-INF/classes +# directory. You are also encouraged to edit it as you like. + +# Configure the console as our one appender +log4j.appender.A1=org.apache.log4j.ConsoleAppender +log4j.appender.A1.layout=org.apache.log4j.PatternLayout +log4j.appender.A1.layout.ConversionPattern=%d{HH:mm:ss,SSS} %-5p [%c] - %m%n + +# tighten logging on the DataNucleus Categories +log4j.category.DataNucleus.JDO=WARN, A1 +log4j.category.DataNucleus.Persistence=WARN, A1 +log4j.category.DataNucleus.Cache=WARN, A1 +log4j.category.DataNucleus.MetaData=WARN, A1 +log4j.category.DataNucleus.General=WARN, A1 +log4j.category.DataNucleus.Utility=WARN, A1 +log4j.category.DataNucleus.Transaction=WARN, A1 +log4j.category.DataNucleus.Datastore=WARN, A1 +log4j.category.DataNucleus.ClassLoading=WARN, A1 +log4j.category.DataNucleus.Plugin=WARN, A1 +log4j.category.DataNucleus.ValueGeneration=WARN, A1 +log4j.category.DataNucleus.Enhancer=WARN, A1 +log4j.category.DataNucleus.SchemaTool=WARN, A1 + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/spring-dispatcher-servlet.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/spring-dispatcher-servlet.xml new file mode 100644 index 0000000000000000000000000000000000000000..5062bb6286b64bc12d4b406bec6970dc49446051 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/spring-dispatcher-servlet.xml @@ -0,0 +1,25 @@ +<beans xmlns="http://www.springframework.org/schema/beans" + xmlns:context="http://www.springframework.org/schema/context" + xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xsi:schemaLocation=" + http://www.springframework.org/schema/beans + http://www.springframework.org/schema/beans/spring-beans-3.0.xsd + http://www.springframework.org/schema/context + http://www.springframework.org/schema/context/spring-context-3.0.xsd + http://www.springframework.org/schema/mvc + http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"> + + <context:component-scan base-package="fr.eurecom.senml.jspcontroller" /> + <mvc:annotation-driven /> + + <bean + class="org.springframework.web.servlet.view.InternalResourceViewResolver"> + <property name="prefix"> + <value>/pages/</value> + </property> + <property name="suffix"> + <value>.jsp</value> + </property> + </bean> + +</beans> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/web.xml b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/web.xml new file mode 100644 index 0000000000000000000000000000000000000000..a02428a32345e47e5744c0ec1d75b5281ca61e0f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/WEB-INF/web.xml @@ -0,0 +1,94 @@ +<?xml version="1.0" encoding="utf-8" standalone="no"?><web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="2.5" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"> + +<!-- Web Services xml/json datas over Restlet framework --> +<servlet> + <servlet-name>restlet-dispatcher</servlet-name> + <servlet-class>org.restlet.ext.servlet.ServerServlet</servlet-class> + <init-param> + <param-name>org.restlet.application</param-name> + <param-value>fr.eurecom.restlet.resources.gae.Dispatcher</param-value> + </init-param> +</servlet> + +<servlet-mapping> + <servlet-name>restlet-dispatcher</servlet-name> + <url-pattern>/contacts/*</url-pattern> +</servlet-mapping> + +<!-- Hack does not support several url-patterns neither several servlet-mapping for same servlet-name --> +<servlet> + <servlet-name>restlet-dispatcher-senml</servlet-name> + <servlet-class>org.restlet.ext.servlet.ServerServlet</servlet-class> + <init-param> + <param-name>org.restlet.application</param-name> + <param-value>fr.eurecom.restlet.resources.gae.DispatcherSenml</param-value> + </init-param> +</servlet> + +<servlet-mapping> + <servlet-name>restlet-dispatcher-senml</servlet-name> + <url-pattern>/senml/*</url-pattern> +</servlet-mapping> + +<servlet> + <servlet-name>SystemServiceServlet</servlet-name> + <servlet-class>com.google.api.server.spi.SystemServiceServlet</servlet-class> + <init-param> + <param-name>services</param-name> + <param-value/> + </init-param> + </servlet> + <servlet-mapping> + <servlet-name>SystemServiceServlet</servlet-name> + <url-pattern>/_ah/spi/*</url-pattern> + </servlet-mapping> + + +<!-- JSP SenML over Spring framework --> + +<servlet> + <servlet-name>spring-dispatcher</servlet-name> + <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> + <load-on-startup>1</load-on-startup> +</servlet> + +<servlet-mapping> + <servlet-name>spring-dispatcher</servlet-name> + <url-pattern>/senmladmin/*</url-pattern> +</servlet-mapping> + +<!-- Pushing mechanism --> +<servlet> + <servlet-name>DeviceManager</servlet-name> + <servlet-class>fr.eurecom.pushing.DeviceManager</servlet-class> +</servlet> +<servlet-mapping> + <servlet-name>DeviceManager</servlet-name> + <url-pattern>/pushing/*</url-pattern> +</servlet-mapping> + +<context-param> + <param-name>contextConfigLocation</param-name> + <param-value>/WEB-INF/spring-dispatcher-servlet.xml</param-value> +</context-param> + +<listener> + <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> +</listener> + + +<welcome-file-list> + <welcome-file>index.html</welcome-file> +</welcome-file-list> + +</web-app><!-- JUST FOR TEST TO REMOVE + <servlet> + <servlet-name>test</servlet-name> + <servlet-class>fr.eurecom.restlet.resources.gae.ResourcesServletGAE</servlet-class> +</servlet> +<servlet-mapping> + <servlet-name>test</servlet-name> + <url-pattern>/testjdo</url-pattern> +</servlet-mapping> + +--> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/css/template_style.css b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/css/template_style.css new file mode 100644 index 0000000000000000000000000000000000000000..1cb94dfa533f4a7f945bb3eeb0445e1243c54905 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/css/template_style.css @@ -0,0 +1,248 @@ +/* +Based on a css from http://www.templatemo.com +*/ + +body{ + margin: 0; + padding: 0; + color: #555; + font-size: 12px; + line-height: 1.6em; + font-family: "Lucida Sans Unicode", "Lucida Grande", sans-serif; + background-color: #c7c8c1; + background-image: url(../images/noisy_grid.png); + background-repeat: repeat; + width: 100%: +} + +a, a:link, a:visited { + color: #525111; + font-weight: normal; + text-decoration: none; +} + +a:hover { + text-decoration: underline; +} + +ul, li { + padding:0; + margin:0; + list-style:none; +} + +h1, h2, h3, h4, h5, h6 { color: #525111; font-weight: normal; } +h1 { font-size: 36px; margin: 0 0 30px; padding: 5px 0 } +h2 { font-size: 24px; margin: 0 0 25px; padding: 5px 0 } +h3 { font-size: 20px; margin: 0 0 20px; padding: 0; } +h4 { font-size: 16px; margin: 0 0 15px; padding: 0; } +h5 { font-size: 14px; margin: 0 0 10px; padding: 0; } +h6 { font-size: 12px; margin: 0 0 5px; padding: 0; font-weight: 700 } + +p { + padding: 0; + margin: 0 0 15px 0; +} + +#zone-table +{ + margin: 0; + width: 98%; + text-align: left; + border-collapse: collapse; + font-size: 16px; + border-top: 1px solid #525111; + border-bottom: 1px solid #525111; + border-right: 1px solid #525111; + border-left: 1px solid #525111; +} +#zone-table thead +{ + padding: 8px 2px; + font-weight: bold; + font-size: 16px; +/* border-top: 1px solid #525111; + border-bottom: 1px solid #525111; + border-right: 1px solid #525111; + border-left: 1px solid #525111; +*/ + border-bottom: 1px solid #525111; + +} + +#zone-table td +{ + color: #000; + padding: 12px 2px 0px 2px; + font-weigth: normal; + font-size: 16px; +/* border-right: 30px solid #fff; + border-left: 30px solid #fff; +*/ +} + +#sensor td +{ + border-top: 1px solid #525111; +} + +#measure td +{ + color: #000; + padding: 0px 5px 5px 5px; + font-weigth: normal; + font-size: 14px; +} + +#zone-table caption +{ + font-size: 18px; + font-weight: bold; + padding: 8px 2px; + color:#525111; +} + + +.clear{ + clear:left; +} + + +cite { + font-size: 14px; + color:#525111; +} + + +.list_bullet { + margin: 10px 0 10px 15px; + padding: 0; + list-style: none; +} + +.no_bullet { + margin: 0; + padding: 0; + list-style: none; +} + +.no_bullet li { + margin: 0 0 20px 0; + padding: 0; +} + +a.header { + display: block; + font-weight: 700; +} + +.half { + width: 360px; +} + +.col_175 { + width: 175px; +} + +.h10 { height: 10px } +.h20 { height: 20px } +.h40 { height: 40px } + +.img { + margin: 0; + padding: 0; +} + +.left { + float: left; +} + +.right { + float: right; +} + +#template_header { + width: 100%; + min-height: 110px; + padding: 40px 20px 20px; + margin: 0 auto; + text-align: center; + margin-bottom: 40px; +} + +#template_main { + width: 100%; + margin: 0 auto; + padding: 0 10px; + overflow: hidden; + text-align: center; +} + +#template_footer { + width: 100%; + margin: 0 auto; + padding: 20px 20px; + text-align: center; +} + +#zone_form { + padding: 0; + width: 100%; +} + +#zone_form form { + margin: 0px; + padding: 0px; +} + + +#contact_form { + padding: 0; + width: 365px; +} + +#contact_form form { + margin: 0px; + padding: 0px; +} + +#contact_form form .input_field { + width: 163px; + padding: 5px; + color: #000; + border: 1px solid #adada3; + background: #e9e9e9; + font-family: Tahoma, Geneva, sans-serif; + font-size: 12px; + margin-top: 5px; +} + +#contact_form form label { + display: block; + font-size: 11px; +} + +#contact_form form textarea { + width: 353px; + height: 110px; + padding: 5px; + color: #000; + border: 1px solid #adada3; + background: #e9e9e9; + font-family: Tahoma, Geneva, sans-serif; + font-size: 12px; + margin-top: 5px; +} + +#contact_form form .submit_btn { + font-size: 12px; + color: #000; + border: 1px solid #adada3; + background: #e9e9e9; + padding: 5px 14px; + margin: 10px 0px; +} + +.hidden { + display: none; +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/favicon.ico b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..0062ab413e73fa137f91cc1b082ce6a9bc992730 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/favicon.ico differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/android.jpg b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/android.jpg new file mode 100644 index 0000000000000000000000000000000000000000..54261265c769baa5f956a4203b9d9474493eb870 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/android.jpg differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/apple.jpg b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/apple.jpg new file mode 100644 index 0000000000000000000000000000000000000000..3b8da490ec54493c3b4856177a53460649466ff1 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/apple.jpg differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/noisy_grid.png b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/noisy_grid.png new file mode 100644 index 0000000000000000000000000000000000000000..ba6365e5511c83383c6cd5fd86fbb0c61bc342ff Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/images/noisy_grid.png differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/index.html b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/index.html new file mode 100644 index 0000000000000000000000000000000000000000..8bd82c35b3a62756f0ac906289278e68ed6e5372 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/index.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> +<!-- The HTML 4.01 Transitional DOCTYPE declaration--> +<!-- above set at the top of the file will set --> +<!-- the browser's rendering engine into --> +<!-- "Quirks Mode". Replacing this declaration --> +<!-- with a "Standards Mode" doctype is supported, --> +<!-- but may lead to some differences in layout. --> + +<html> + <head> + <meta http-equiv="content-type" content="text/html; charset=UTF-8"> + <title>Hello App Engine</title> + </head> + + <body> + <h1>Hello App Engine!</h1> + Only authorized persons allowed! + </body> +</html> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/js/main.js b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/js/main.js new file mode 100644 index 0000000000000000000000000000000000000000..137f2cd88d13a018bcb9d2e032aee667e12f8ea9 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/js/main.js @@ -0,0 +1,43 @@ +/** + * Show a given element. It basically consists of removing the "hidden" class. + @param id the id of the DOM element to show + */ +function showDOMEl(id) { + document.getElementById(id).className = + document.getElementById(id).className.replace + ( /(?:^|\s)hidden(?!\S)/g , '' ); +} + +/** + * Hide a given element. It basically consists of adding the "hidden" class. + @param id the id of the DOM element to show + */ +function hideDOMEl(id) { + document.getElementById(id).className += "hidden"; +} + +/** + * Toggle the status of the element. + * Ex.: if the element is shown, hide it. And Viceversa. + * @param id the id of the DOM element to toggle + */ +function toggleDOMEl(id) { + if (document.getElementById(id).className.match(/(?:^|\s)hidden(?!\S)/)) { + showDOMEl(id); + } else { + hideDOMEl(id); + } +} + +/** + * Change the form action and register the sensor to be monitored. + */ +function registerForAlert(id) { + // change form action + f = document.getElementById("measureAdd" + id); + f.action = "addAlertMonitor"; + // remove required attribute from value, it's not needed for this action + var v = document.getElementById("value" + id); + v.required = false; + f.submit(); +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/alarms.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/alarms.jsp new file mode 100644 index 0000000000000000000000000000000000000000..0224e69131bd8979f32f25e8be88d91e855e2644 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/alarms.jsp @@ -0,0 +1,44 @@ +<%@ page contentType="text/html;charset=UTF-8" language="java"%> +<%@ page import="fr.eurecom.senml.entity.IZone"%> +<%@ page import="fr.eurecom.senml.entity.IZoneAdmin"%> +<%@ page import="fr.eurecom.senml.entity.ISensor"%> +<%@ page import="fr.eurecom.senml.entity.SensorCheckAlarm"%> +<%@ page import="java.util.List"%> +<%@ page import="java.util.Iterator"%> +<html> +<head> +<link href="/css/template_style.css" type="text/css" rel="stylesheet" /> +<script src="/js/main.js" type="text/javascript"> + +</script> +<title>Alarms</title> +</head> + +<body> +<div id="template_main"> + <table border="1"> + <tr><th>Name</th><th>Min value</th><th>Max value</th><th>JSON</th><th>Change</th><th>Remove</th></tr> + + <% + List<SensorCheckAlarm> list = SensorCheckAlarm.getAllMonitoredSensors(); + Iterator<SensorCheckAlarm> iter = list.iterator(); + while (iter.hasNext()) { + SensorCheckAlarm s = iter.next(); + %> + <tr> + <td> + <strong><%= s.getParentSensorName() %></strong></td> + <td> <%= s.getMinVal() %></td> + <td> <%= s.getMaxVal() %></td> + <td><a href="getmonitor?key=<%= s.getKey() %>">JSON</a></td> + <td><a href="edit">Edit</a></td> + <td><a href="removemonitor?key=<%= s.getKey() %>">Remove</a></td> + </tr> + <% } %> + </table> + </div> + +</body> +<footer id="template_footer">Eurecom / background pattern draw + by vectorpile.com</footer> +</html> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/alarmsJSON.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/alarmsJSON.jsp new file mode 100644 index 0000000000000000000000000000000000000000..119c32a0b1636cf30fa75d39ce5089813dc4da05 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/alarmsJSON.jsp @@ -0,0 +1,22 @@ +<%@ page trimDirectiveWhitespaces="true" %> +<%-- Set the content type header with the JSP directive --%> +<%@ page contentType="application/json"%> +<%-- Set the content disposition header --%> +<%response.setContentType("application/json");%> +<%@ page language="java"%> +<%@ page import="java.util.List"%> +<%@ page import="java.util.Iterator"%> +<%@ page import="org.json.JSONArray"%> +<%@ page import="fr.eurecom.senml.entity.SensorCheckAlarm"%> +<%@ page import="fr.eurecom.senml.persistence.JDOStorage"%> + + <% + JSONArray array = new JSONArray(); + List<SensorCheckAlarm> list = SensorCheckAlarm.getAllMonitoredSensors(); + Iterator<SensorCheckAlarm> iter = list.iterator(); + while (iter.hasNext()) { + SensorCheckAlarm s = iter.next(); + array.put(s.toJSONSenML()); + } + %> +<%= array %> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/allalarmsJSON.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/allalarmsJSON.jsp new file mode 100644 index 0000000000000000000000000000000000000000..015c261e740681d9cf27f4b84dbe7f832a5e2162 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/allalarmsJSON.jsp @@ -0,0 +1,45 @@ +<%@ page trimDirectiveWhitespaces="true" %> +<%-- Set the content type header with the JSP directive --%> +<%@ page contentType="application/json"%> +<%-- Set the content disposition header --%> +<%response.setContentType("application/json");%> +<%@ page language="java"%> +<%@ page import="java.util.List"%> +<%@ page import="java.util.Iterator"%> +<%@ page import="org.json.JSONArray"%> +<%@ page import="org.json.JSONObject"%> +<%@ page import="fr.eurecom.senml.entity.SensorAdmin"%> +<%@ page import="fr.eurecom.senml.entity.Measure"%> +<%@ page import="fr.eurecom.senml.entity.SensorCheckAlarm"%> +<%@ page import="fr.eurecom.senml.persistence.JDOStorage"%> + + <% + JSONArray array = new JSONArray(); + List<SensorCheckAlarm> list = SensorCheckAlarm.getAllMonitoredSensors(); + Iterator<SensorCheckAlarm> iter = list.iterator(); + while (iter.hasNext()) { + SensorCheckAlarm s = iter.next(); + SensorAdmin sensor = JDOStorage.getInstance().getById(s.getSensorKey(), SensorAdmin.class); + JSONObject element = new JSONObject(); + element.put("name", s.getParentSensorName()); + element.put("baseunit", sensor.getBaseUnit()); + element.put("monitor", s.toJSONSenML()); + + /* TODO: REMARK + CORELINKS will introduce intelligent sensors. + At that time, we'll be able to have different measures of different types + from the same sensor. + FOR NOW, 1 sensor corresponds to 1 measure of the same type, everytime! + */ + + // Retrieve value + List<Measure> measures = sensor.getMeasures(); + if (measures.isEmpty()) { + System.out.println("Measures empty!"); + } else { + element.put("measure", measures.get(0).toJSONSenML()); + array.put(element); + } + } + %> +<%= array %> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/detailzone.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/detailzone.jsp new file mode 100644 index 0000000000000000000000000000000000000000..4add59fb000d6e21564309a6d3f604678eb387a7 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/detailzone.jsp @@ -0,0 +1,357 @@ +<%@ page contentType="text/html;charset=UTF-8" language="java"%> +<%@ page import="fr.eurecom.senml.entity.IZone"%> +<%@ page import="fr.eurecom.senml.entity.IZoneAdmin"%> +<%@ page import="fr.eurecom.senml.entity.ISensor"%> +<%@ page import="fr.eurecom.senml.entity.Measure"%> +<%@ page import="fr.eurecom.senml.entity.Units"%> +<%@ page import="fr.eurecom.senml.jspcontroller.ZoneTypes"%> +<%@ page import="java.util.List"%> +<html> +<head> +<link href="/css/template_style.css" type="text/css" rel="stylesheet" /> +<script src="/js/main.js" type="text/javascript"> + +</script> +<title>Zone Detail page</title> +</head> + +<body> + <% + IZoneAdmin z = (IZoneAdmin) request.getAttribute("zone"); + List<IZone> subZones = z.getSubZones(); + List<ISensor> sensors = z.getSensors(); + Units[] units = Units.values(); + %> + <div id="template_header"> + <h1> + Zone: + <%=z.getName()%> + in: + <%=z.getParentZone() == null ? "" : z.getParentZone() + .getName()%> + </h1> + <a href="../zones"> Zones Full List </a> + <hr> + </div> + + <div id="template_main"> + <!-- subzones --> + <form action="addsubZone" method="post" id="zone_form"> + <label>New sub zone</label> <input type="text" name="nameSubZone" + placeholder="nameZone" required="required"> <input + type="hidden" name="zoneKey" value="<%=z.getKey()%>"> + + <p> + Choose a type: <select name="typeZone"> + <% + String[] listZones = ZoneTypes.getZoneTypes(); + for (int i = 0; i < listZones.length; i++) { + %> + <option value="<%=listZones[i]%>"> + <%=listZones[i]%> + <% + } + %> + + </select> + </p> + <input type="submit" value="Add Sub Zone"> + </form> + + <table id="zone-table"> + <caption>Sub Zones</caption> + <thead> + <tr> + <th>Name</th> + <th>SubZones</th> + <th>Sensors</th> + <th></th> + <th></th> + </tr> + </thead> + + <tbody> + <% + for (IZone sz : subZones) { + %> + <tr> + <td><%=sz.getName()%></td> + <td><%=sz.getSubZones().size() > 0 ? "Yes" : "No"%></td> + <td><%=sz.getSensors().size() > 0 ? "Yes" : "No"%></td> + <td><a href="../<%=sz.getKey()%>/edit">Edit</a></td> + <td><a href="delete/<%=sz.getKey()%>">Delete</a></td> + </tr> + <% + } + %> + </tbody> + <tfoot> + <tr> + <td></td> + <td></td> + <td></td> + <td></td> + <td></td> + </tr> + </tfoot> + </table> + + <!-- sensors --> + <br> <br> + <form action="addSensor" method="post" id="zone_form"> + <label>New Sensor</label> <input type="text" name="title" + placeholder="title sensor" required="required"> <br /> <label + for="baseunit">Base unit</label> <select name="baseunit"> + + <% + for (Units u : units) { + %> + <option value="<%=u.getCS()%>"><%=u.name().toLowerCase()%></option> + <% + } + %> + </select> <input type="hidden" name="zoneKey" value="<%=z.getKey()%>"> + <br /> + <p> + <select name="typeSensor"> + <option value="Magnetometer Sensor">Magnetometer Sensor</option> + <option value="GPS Sensor">GPS Sensor</option> + <option value="Gyrometer Sensor">Gyrometer Sensor</option> + <option value="Optical Sensor">Optical Sensor</option> + <option value="Gyroscope Sensor">Gyroscope Sensor</option> + <option value="Hygrometer">Hygrometer</option> + <option value="Conductivity Sensor">Conductivity Sensor</option> + <option value="Wind Direction Sensor">Wind Direction + Sensor</option> + <option value="Pluviometer">Pluviometer</option> + <option value="Thermometer">Thermometer</option> + <option value="Light Sensor">Light Sensor</option> + <option value="Smoke Detector">Smoke Detector</option> + <option value="ThermalSensor">ThermalSensor</option> + <option value="Respiration">Respiration</option> + <option value="Electromagnetic Sensor">Electromagnetic + Sensor</option> + <option + value="Solar Radiation Sensor. PAR (Photosynthetically Active Radiation) Sensor">Solar + Radiation Sensor. PAR (Photosynthetically Active Radiation) Sensor</option> + <option value="Pressure Sensor">Pressure Sensor</option> + <option value="Pollutant Sensor">Pollutant Sensor</option> + <option value="Pulse Oxymeter">Pulse Oxymeter</option> + <option value="Wind Velocity Sensor">Wind Velocity Sensor</option> + <option value="Mechanical Sensor">Mechanical Sensor</option> + <option value="ElectricalSensor">ElectricalSensor</option> + <option value="Acoustic Sensor">Acoustic Sensor</option> + <option value="Motion Sensor/ Accelerometer">Motion + Sensor/ Accelerometer</option> + <option value="Clock">Clock</option> + <option value="Pyroelectric IR Occupancy Detector">Pyroelectric + IR Occupancy Detector</option> + <option value="Chemical Sensor">Chemical Sensor</option> + <option value="Nuclear Sensor">Nuclear Sensor</option> + </select> + </p> + <input type="submit" value="Add Sensor"> + </form> + + <table id="zone-table"> + <caption>Sensors</caption> + <thead> + <tr> + <th>Title</th> + <th>UUID</th> + <th>Type</th> + <th>Base Unit</th> + <th>Measures</th> + <th></th> + <th></th> + <th></th> + </tr> + </thead> + + <tbody> + <% + for (ISensor s : sensors) { + %> + <tr id="sensor"> + <td><%=s.getTitle()%></td> + <td><%=s.getUUID()%></td> + <td><%=s.zoneType()%></td> + <% + if (s.getBaseUnit() != null) { + %> + <td><%=s.getBaseUnit()%></td> + <% + } else { + %> + <td>No base unit</td> + <% + + } + String formID = "measureAdd" + s.getUUID(); + String valueID = "value" + s.getUUID(); + %> + + <td><form action="addMeasure" method="post" id="<%= formID %>" + name="<%= formID %>"> + <fieldset> + <legend>Add the sensor to the list of monitored + sensors</legend> + <input name="rtallowedmin" placeholder="min value allowed"/><br/> + <input name="rtallowedmax" placeholder="max value allowed"/> + <input name="bn" type="hidden" value="<%= s.getTitle() %>"/> + <button onClick="registerForAlert('<%=s.getUUID()%>');">Monitor!</button> + </fieldset> + + <p> + Type: <select name="measureType"> + <option value="Animal">Animal</option> + <option value="Paymend Card">Payment Card</option> + <option value="Luggage">Luggage</option> + <option value="Passport">Passport</option> + <option value="Clothing">Clothing</option> + <option value="ParkingSpace">ParkingSpace</option> + <option value="Book">Book</option> + <option value="Food">Food</option> + <option value="Toll">Toll</option> + <option value="CD">CD</option> + <option value="DVD">DVD</option> + <option value="TransitPass">TransitPass</option> + <option value="Sodium">Sodium</option> + <option value="Conductivity">Conductivity</option> + <option value="Visibility">Visibility</option> + <option value="LuminousFlux">LuminousFlux</option> + <option value="MagneticFluxDensity">MagneticFluxDensity</option> + <option value="HeartBeat">HeartBeat</option> + <option value="Glucose">Glucose</option> + <option value="Oxygen">Oxygen</option> + <option value="Calcium">Calcium</option> + <option value="Luminous Intensity">Luminous Intensity</option> + <option value="Blood pressure">Blood pressure</option> + <option value="Wind Direction">Wind Direction</option> + <option value="Angular">Angular</option> + <option value="Electrical Potential">Electrical + Potential</option> + <option value="Temperature">Temperature</option> + <option value="Motion">Motion</option> + <option value="Electric Charge">Electric Charge</option> + <option value="Illuminance">Illuminance</option> + <option value="Electric Current">Electric Current</option> + <option value="Cholesterol">Cholesterol</option> + <option value="Atmospheric Pressure">Atmospheric + Pressure</option> + <option value="Power">Power</option> + <option value="Capacitance">Capacitance</option> + <option value="Potassium">Potassium</option> + <option value="RFID Measurement Type">RFID Measurement + Type</option> + <option value="Electrical Resistance">Electrical + Resistance</option> + <option value="Humidity">Humidity</option> + <option value="Skin conductance/ GSR">Skin + conductance/ GSR</option> + <option value="Location">Location</option> + <option value="Magnetic Field">Magnetic Field</option> + <option value="Precipitation">Precipitation</option> + <option value="Pressure">Pressure</option> + <option + value="PAR Measurement (Photosynthetically Active Radiation)">PAR + Measurement (Photosynthetically Active Radiation)</option> + <option value="Water level">Water level</option> + <option value="Wind Velocity">Wind Velocity</option> + </select> + </p> + <input type="hidden" name="sensorKey" value="<%=s.getUUID()%>"> + <input type="hidden" name="bu" value="<%=s.getBaseUnit()%>"> + Name: <input type="text" name="name" placeholder="measure name"> + <% + if (s.getBaseUnit() != null) { + %> + <br /> + <p> + <fieldset> + <legend>If not specified, the value is in the base + unit:</legend> + <%=s.getBaseUnit()%></p> + <p> + Specify value type: <input type="checkbox" name="specmeasure" + onClick="toggleDOMEl('<%=s.getUUID()%>measurediv');"> + </p> + <div id="<%=s.getUUID()%>measurediv" class="hidden"> + <% + } else { + %> + <div id="<%=s.getUUID()%>measurediv"> + <% + } + %> + <select name="unit"> + <% + for (Units u : units) { + %> + <option value="<%=u.getCS()%>"><%=u.name().toLowerCase()%></option> + <% + } + %> + </select> + </fieldset> + + </div> + <br /> When: <select name="time"> + <option value="NOW">now</option> + <option value="AUTO">auto</option> + </select> <br /> <br /> + <fieldset> + <legend>Value</legend> + Value type: <select name="typeValue"> + <option value="FLOAT">float</option> + <option value="BOOL">boolean</option> + <option value="String">String</option> + </select> Value: <input type="text" name="value" id="<%= valueID %>" placeholder="value" + required="required"> + </fieldset> + <br /> <label for="version">Version</label> <select + name="version"> + <option value="2.0">2.0</option> + <option value="1.0">1.0</option> + </select> <br /> <br /> Sum: <input type="text" placeholder="sum" + name="sum" /> <br /> <br /> <input type="submit" + value="add a Measure"> + </form></td> + <td></td> + <td></td> + <td><a href="deleteSensor/<%=s.getUUID()%>">Delete</a></td> + </tr> + <% + List<Measure> measures = s.getMeasures(); + for (Measure m : measures) { + %> + <tr id="measure"> + <td></td> + <td><%=m.toPrettyString()%></td> + <td><%=m.toJSONSenML().toString()%> + <td></td> + <td></td> + <td><a href="deleteMeasure/<%=m.getKey()%>">Delete</a></td> + </tr> + <% + } + } + %> + </tbody> + <tfoot> + <tr> + <td></td> + <td></td> + <td></td> + <td></td> + <td></td> + <td></td> + </tr> + </tfoot> + </table> + </div> + +</body> +<footer id="template_footer">Eurecom / background pattern draw + by vectorpile.com</footer> +</html> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/postvalslist.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/postvalslist.jsp new file mode 100644 index 0000000000000000000000000000000000000000..66fbf527add566cd674ceffb08ba09def3af920c --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/postvalslist.jsp @@ -0,0 +1,40 @@ +<%@ page contentType="text/html;charset=UTF-8" language="java" %> +<%@ page import="fr.eurecom.senml.entity.PostValType" %> +<%@ page import="java.util.List" %> +<html> +<head> +<link href="../css/template_style.css" type="text/css" rel="stylesheet"/> +<title>PostVal Main page</title> +</head> + +<body> + <br/><br/> + <div id="template_main"> + + <table id="zone-table"> + <caption>Full list</caption> + <thead> + <tr> <th>Name</th> <th>Value</th> </tr> + </thead> + <tbody> + +<% List<PostValType> postvals = (List<PostValType>)request.getAttribute("postvalsList"); + if (postvals != null && !postvals.isEmpty()) { + for (PostValType z : postvals) { +%> + <tr> + <td> <%= z.getName() %></td> + <td> <%= z.getValue() %> @ <%= z.getTime() %></td> + </tr> +<% + } + } +%> + </tbody> + <tfoot><tr> <td></td><td></td> </tr></tfoot> + </table> + </div> + +</body> +<footer id="template_footer">Eurecom / background pattern draw by vectorpile.com</footer> +</html> diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/sensormonitorsjson.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/sensormonitorsjson.jsp new file mode 100644 index 0000000000000000000000000000000000000000..34c02613165448c89ec82497ecf850cf12cc4475 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/sensormonitorsjson.jsp @@ -0,0 +1,13 @@ +<%@ page trimDirectiveWhitespaces="true" %> +<%-- Set the content type header with the JSP directive --%> +<%@ page contentType="application/json"%> +<%-- Set the content disposition header --%> +<%response.setContentType("application/json");%> +<%@ page language="java"%> +<%@ page import="fr.eurecom.senml.entity.SensorCheckAlarm"%> +<%@ page import="fr.eurecom.senml.persistence.JDOStorage"%> +<% + String sensorKey = request.getParameter("key"); + SensorCheckAlarm s = JDOStorage.getInstance().getById(sensorKey, SensorCheckAlarm.class); +%> +<%= s.toJSONSenML() %> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/zonelist.jsp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/zonelist.jsp new file mode 100644 index 0000000000000000000000000000000000000000..16e5f6e6e75f44c564dc4d58e4427e3d6cf7b2d7 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/server-gae/restlet/war/pages/zonelist.jsp @@ -0,0 +1,71 @@ +<%@ page contentType="text/html;charset=UTF-8" language="java" %> +<%@ page import="fr.eurecom.senml.entity.IZone" %> +<%@ page import="fr.eurecom.senml.jspcontroller.ZoneTypes" %> +<%@ page import="java.util.List" %> +<html> +<head> +<link href="../css/template_style.css" type="text/css" rel="stylesheet"/> +<title>Zone Main page</title> +</head> + +<body> + <div id="template_header"> + <h1>Supervisor Page</h1> + <br/><br/> + <hr> + <h4> New Zone</h4> + <form action="addZone" method="post" id="form-add"> + <p> + <input type="text" name="nameZone" + placeholder="name zone" required="required"> + <input type="text" name="parentNameZone" + placeholder=" parent zone name"> + </p> + <p>Choose a type: <select name="typeZone"> + <% String[] listZones = ZoneTypes.getZoneTypes(); + for (int i = 0; i < listZones.length; i++) { + %> + <option value="<%= listZones[i] %>"> <%= listZones[i] %> + <% + } + %> + </select> + </p> + <p><input type="submit" value="Save"> <input type="reset" value="Clear"></p> + </form> + <hr> + </div> + + <br/><br/> + <div id="template_main"> + + <table id="zone-table"> + <caption>Full list</caption> + <thead> + <tr> <th>Name</th> <th>Parent</th><th>Type</th><th></th><th></th> </tr> + </thead> + <tbody> + +<% List<IZone> zones = (List<IZone>)request.getAttribute("zoneList"); + if (zones != null && !zones.isEmpty()) { + for (IZone z : zones) { +%> + <tr> + <td> <%= z.getName() %></td> + <td> <%= z.getParentZone() == null ? "" : z.getParentZone().getName() %></td> + <td> <%= z.getType() %></td> + <td><a href="<%= z.getKey() %>/edit">Edit</a></td> + <td><a href="deleteZone/<%= z.getKey() %>">Delete</a></td> + </tr> +<% + } + } +%> + </tbody> + <tfoot><tr> <td></td><td></td><td></td><td></td> </tr></tfoot> + </table> + </div> + +</body> +<footer id="template_footer">Eurecom / background pattern draw by vectorpile.com</footer> +</html> \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient new file mode 100644 index 0000000000000000000000000000000000000000..29465e592af164bb52a3340f0900e1acead09fa9 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient.cpp new file mode 100644 index 0000000000000000000000000000000000000000..1d31bcf9c95c001d94d673211be1fd5d0e25d03d --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient.cpp @@ -0,0 +1,251 @@ +#include <cpprest/http_client.h> +#include <cpprest/json.h> +#include <iostream> +#include <ostream> +#include <sstream> +#include <fstream> +#include "cpprest/basic_types.h" +#include "cpprest/asyncrt_utils.h" +#include "cpprest/uri.h" +#include <string> + +using namespace std; +using namespace web; +using namespace web::http; +using namespace web::http::client; +using namespace utility; + +// Retrieves a JSON value from an HTTP request. +pplx::task<void> RequestJSONValueAsync(char * url) +{ + http_client client (U(url)); + return client.request(methods::GET).then([](http_response response) -> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + ofstream myfile; + myfile.open ("outputGET.txt"); + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + myfile << jsonString.c_str(); + myfile.close(); + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +// Stores a JSON value (a Policy) from an HTTP request. +pplx::task<void> StoreJSONValuePolicies(char * url, char * param1, char * param2, char * param3, char * param4) +{ + + http_client client (U(url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("pid"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("description"), json::value(param3))); + putvalue.push_back(make_pair(json::value("value"), json::value(param4))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); +} + + +// Stores a JSON value (a Measurement) from an HTTP request. +pplx::task<void> StoreJSONValuemeasurements(char * url, char * param1, char * param2, char * param3, char * param4, char * param5, char * param6) +{ + + http_client client (U(url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("key"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("type"), json::value(param3))); + putvalue.push_back(make_pair(json::value("unit"), json::value(param4))); + putvalue.push_back(make_pair(json::value("value"), json::value(param5))); + putvalue.push_back(make_pair(json::value("time"), json::value(param6))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +// Stores a JSON value (a Decision) from an HTTP request. +pplx::task<void> StoreJSONValuedecisions(char * url, char * param1, char * param2, char * param3, char * param4, char * param5) +{ + + http_client client (U(url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("did"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("description"), json::value(param3))); + putvalue.push_back(make_pair(json::value("value"), json::value(param4))); + putvalue.push_back(make_pair(json::value("time"), json::value(param5))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +pplx::task<void> Delete(char* url) +{ + return pplx::create_task([url] + { + http_client client(U(url)); + + return client.request(methods::DEL); + + }).then([](http_response response) + { + if(response.status_code() == status_codes::OK) + { + auto body = response.extract_string(); + + std::wcout << L"Deleted: " << body.get().c_str() << std::endl; + } + }); +} + + +int main(int argc,char *argv[]) +{ + char * policiesurl = (char*) malloc( 100 * sizeof(char) ); + strcpy(policiesurl,"http://1.gae-spectra.appspot.com/policies"); + char * measurementsurl = (char*) malloc( 100* sizeof(char) ); + strcpy(measurementsurl,"http://1.gae-spectra.appspot.com/measurements"); + char * decisionsurl = (char*) malloc( 100 * sizeof(char)); + strcpy(decisionsurl,"http://1.gae-spectra.appspot.com/decisions"); + int c; + + if (strcmp(argv[1], "GET") == 0) + { + c = atoi(argv[2]); + switch (c) + { + case 1: RequestJSONValueAsync(policiesurl).wait(); + break; + case 2: RequestJSONValueAsync(measurementsurl).wait(); + break; + case 3: RequestJSONValueAsync(decisionsurl).wait(); + break; + } + } + else if (strcmp(argv[1], "PUT") == 0) + { + c = atoi(argv[2]); + switch (c) + { + case 1: StoreJSONValuePolicies(policiesurl, argv[3], argv[4], argv[5], argv[6]).wait(); + break; + case 2: StoreJSONValuemeasurements(measurementsurl, argv[3], argv[4], argv[5], argv[6], argv[7], argv[8]).wait(); + break; + case 3: StoreJSONValuedecisions(decisionsurl, argv[3], argv[4], argv[5], argv[6], argv[7]).wait(); + break; + } + } + else if (strcmp(argv[1], "DEL") == 0) + { + c = atoi(argv[2]); + switch (c) + { + case 1: Delete(policiesurl).wait(); + break; + case 2: Delete(measurementsurl).wait(); + break; + case 3: Delete(decisionsurl).wait(); + break; + } + } + + return 0; +} + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient.java b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient.java new file mode 100644 index 0000000000000000000000000000000000000000..599b571ea98d370cf25e0bb0cad3545437d9f072 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/CRMClient.java @@ -0,0 +1,153 @@ +package fr.eurecom.spectra.crrm.client; + +import java.io.IOException; + +import net.sf.json.JSONObject; +import net.sf.json.JSONSerializer; + +import org.restlet.Context; +import org.restlet.data.MediaType; +import org.restlet.ext.json.JsonRepresentation; +import org.restlet.representation.Representation; +import org.restlet.resource.ClientResource; +import org.restlet.resource.ResourceException; + +import com.google.gson.Gson; + +import fr.eurecom.spectra.crrm.entities.Decision; +import fr.eurecom.spectra.crrm.entities.Measurement; +import fr.eurecom.spectra.crrm.entities.Policy; + +public class CRMClient { + + private static final String BASE_URL = "http://1.gae-spectra.appspot.com/"; + + // http://localhost:8888/ + public static void main(String[] args) throws IOException, + ResourceException { + + final Context context = new Context(); + + // Define our Restlet client resources policies. + ClientResource policiesResource = new ClientResource(context, + BASE_URL + "policies"); + ClientResource policyResource = null; + + // Define our Restlet client resources measurements. + ClientResource measurementsResource = new ClientResource( + context, BASE_URL + "measurements"); + ClientResource measurementResource = null; + + // Define our Restlet client resources decisions. + ClientResource decisionsResource = new ClientResource( + context, BASE_URL + "decisions"); + ClientResource decisionResource = null; + + context.getParameters().set("maxConnectionsPerHost", "20"); + + // Create a new Policy + Policy p = new Policy("p1", "priority.", + "Priority of the user on the secondary user", 1); + + Gson gson = new Gson(); + String myp = gson.toJson(p); + + JSONObject jObject = (JSONObject) JSONSerializer.toJSON(myp); + Representation entity = new JsonRepresentation(jObject); + entity.setMediaType(MediaType.APPLICATION_JSON); + policiesResource.setRequestEntityBuffering(true); + Representation reply = policiesResource.post(entity, + MediaType.APPLICATION_JSON); + policiesResource.release(); + // Create a new measurement + Measurement m1 = new Measurement("m1", "SNIR", + "the signal noise interference ratio", 1, + "Physical measurement"); + + Gson gson2 = new Gson(); + String mym = gson2.toJson(m1); + JSONObject jObject2 = (JSONObject) JSONSerializer.toJSON(mym); + Representation entity2 = new JsonRepresentation(jObject2); + entity2.setMediaType(MediaType.APPLICATION_JSON); + measurementsResource.setRequestEntityBuffering(true); + Representation reply2 = measurementsResource.post(entity2, + MediaType.APPLICATION_JSON); + + // Create a new Decision + Decision d1 = new Decision("md1", "Band", + "the allocated band for the UE", 2, 10); + Gson gson3 = new Gson(); + String myd = gson3.toJson(d1); + JSONObject jObject3 = (JSONObject) JSONSerializer.toJSON(myd); + Representation entity3 = new JsonRepresentation(jObject3); + entity2.setMediaType(MediaType.APPLICATION_JSON); + decisionsResource.setRequestEntityBuffering(true); + Representation reply3 = decisionsResource.post(entity3, + MediaType.APPLICATION_JSON); + + if (policiesResource.getStatus().isSuccess()) { + policyResource = new ClientResource(context, + reply.getLocationRef() + "/" + p.getpID()); + + policiesResource.get(MediaType.APPLICATION_JSON); + if (policiesResource.getStatus().isSuccess() + && policiesResource.getResponseEntity() + .isAvailable()) { + + Representation rep = policiesResource + .getResponseEntity(); + Gson gtest = new Gson(); + String text = rep.getText(); + Policy[] ptest = gtest.fromJson(text, Policy[].class); + System.out.println("Extracted Policy " + + ptest[0].getpID()); + } + + } + // policiesResource.release(); + policiesResource.delete(); + measurementsResource.delete(); + decisionsResource.delete(); + } + + /** + * Prints the resource's representation. + * + * @param clientResource + * The Restlet client resource. + * @throws IOException + * @throws ResourceException + */ + public static void get(ClientResource clientResource) + throws IOException, ResourceException { + clientResource.get(MediaType.APPLICATION_JSON); + if (clientResource.getStatus().isSuccess() + && clientResource.getResponseEntity().isAvailable()) { + clientResource.get(MediaType.APPLICATION_JSON).write( + System.out); + // clientResource.getResponseEntity().write(System.out); + } + } + + /** + * Returns the Representation of an item. + * + * @param item + * the item. + * + * @return The Representation of the item. + * @throws IOException + */ + public static Representation getRepresentation(Policy p) + throws IOException { + + Gson gson = new Gson(); + String myp = gson.toJson(p); + + JSONObject js = (JSONObject) JSONSerializer.toJSON(myp); + Representation entity = new JsonRepresentation(js); + entity.setMediaType(MediaType.APPLICATION_JSON); + return entity; + } + +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/README b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/README new file mode 100644 index 0000000000000000000000000000000000000000..6fba212daf3d330213bb75e879c9fb313b576ce0 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/README @@ -0,0 +1,26 @@ +Two CRM clients are provided: + +1)The C++ client: + +Files +source: CRMClient.cpp +binary: CRMClient + +to execute the C++ CRM client: + +./CRMClient arg1 arg2 arg3 arg4 arg5.. + +arg1= "GET" if the operation is a GET from the server + "PUT" if the operation is a PUT to the sever + "DEL" if the opertion is a DEL in the server + +arg2= 1 if the operation is on the POLICIES + = 2 if the operation is on the MEASUREMENTS + = 3 if the operation is on the DECISIONS + +arg3 arg4.. are the parameters of each entity (Policy has 4 arguments, Measurement has 6 arguments, Decision has 5 arguments) + +2)An example of a JAVA Client + +Files +source: CRMClient.java diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/del_rules_on_server.sh b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/del_rules_on_server.sh new file mode 100644 index 0000000000000000000000000000000000000000..200b12245eb1f1a63ec8a485853ff9636e07cd99 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/del_rules_on_server.sh @@ -0,0 +1,2 @@ +#!/bin/bash +./CRMClient DEL 1 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/put_rules_on_server.sh b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/put_rules_on_server.sh new file mode 100644 index 0000000000000000000000000000000000000000..b8899f5933cd0ec7cd770f5de4a13ce8c41112df --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/CRM/tosend/put_rules_on_server.sh @@ -0,0 +1,6 @@ +#!/bin/bash +./CRMClient PUT 1 p10 Fairness "Fairness among secondary users" 1 +./CRMClient PUT 1 p11 Priority "Priority of primary users against secondary users" 1 +./CRMClient PUT 1 p12 Pdetection "Probability of detection threshold" 3 +./CRMClient PUT 1 p14 Pfalse "Probability of false alarms trheshold" 5 + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/client b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/client new file mode 100644 index 0000000000000000000000000000000000000000..ad427a11b4ae714c74e20f9b5f037d56009b004e Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/client differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/server b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/server new file mode 100644 index 0000000000000000000000000000000000000000..4b6559bfa6a4e92c0dd826805cb19df6c6566bdd Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/server differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra.h b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra.h new file mode 100644 index 0000000000000000000000000000000000000000..dbdbbef7343e7f3f047f2fa6515fdd2a40ac9e59 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra.h @@ -0,0 +1,183 @@ +/* + * spectra.h + * + * Helper functions for the test server program. The functions are as + * close as possible to the ones used in the real server implementation + * that is supposed to be running in the board. + * + * Created on: Jun 26, 2014 + * Author: camara + * Copyright (c) 2008-2014 Institut Mines-Telecom / Telecom ParisTech + */ + +#ifndef SPECTRA_H_ +#define SPECTRA_H_ +#include <errno.h> + +#define IP_ADDRESS "10.42.0.104" +#define IP_MASK "255.255.255.0" +#define PORT 4546 + +/** + * Maximum packet size, dependent on the size of the + * number of elements field on the header, at the present 8 bits + * i.e. (256 * 4 bytes + the header that is 4 bytes) + */ +#define MAX_PACKET_SIZE 1028 + +/** + * Aligns packet, internally, into 32 bits, or 4 bytes + */ +#define PACKET_ALIGNEMENT_BYTES 4 + +//#define DEBUG +#ifdef DEBUG +#define PRINTD printf("-> %s:%d: ", __FUNCTION__, __LINE__); printf +#else +#define PRINTD(format, args...) ((void)0) +#endif + +/** + * Which is the type of the message, if it is, for example, an start sensing, + * or an answer to a previously requested score + */ +typedef enum { + Sense, //!< Sense + Classifying, //!< Classifying + ScoreResponse, //!< ScoreResponse + EndProcessing +} messageType; + +/** + * Which is the functions that should be performed by the embb + */ + +//! An enum. +/*! More detailed enum description. */ +typedef enum { + EnergyDetection, //!< EnergyDetection + WelchPeriodograms //!< WelchPeriodograms +} functionToBePerformed; + +/** + * Definition of the spectra control packets + */ +typedef struct { + uint16_t messageID; //!< Operation ID, used to identify an request and its answer + messageType type :4; //!< 16 different types of message possible + functionToBePerformed function :4; //!< Embb function to be performed 16 possible different functions + unsigned char numberOfparameters; //!< up to 255 parameters per message + uint32_t * parameters; //!< variable number of parameters, depending +}__attribute__((packed)) SpectraMessage; + +/** + * \typedef SpectraMsgHelper + * Helper structure to handle control packets + */ +typedef struct { + uint16_t messageID; //!< Operation ID + unsigned char type_function; //!< Assembled type and function for easier treatment + unsigned char numberOfparameters; //!< Number of parameters + uint32_t parameters[256]; //!< Parameters in sequence +}__attribute__((packed)) SpectraMsgHelper; + +/** + * Helper union structure to make easier to pack and unpack messages + */ +union uSpectraPackets { + SpectraMessage spectraMsg; //!< spectraMsg + SpectraMsgHelper spectraMsgHelper; //!< spectraMsgHelper + unsigned char spectraMsgStream[MAX_PACKET_SIZE]; //!< spectraMsgStream +}; + +union FuncPtr { + void (*fp)(); // function pointer + unsigned char c[1]; // address of the pointer as a vector of chars +}; + + +/** + * Serialize and send packet over the indicated socket + * + * @param sock - in use socket descriptor + * @param msg - message to be transmitted + * @return error status 0 if OK, 1 if something went wrong + */ +int returnSpectraPacket(SpectraMsgHelper msg, + int desc, + struct sockaddr remote) { + + int i; + // returns the response value to the OpenAirInterface + if ((i = sendto(desc,&msg, PACKET_ALIGNEMENT_BYTES + (msg.numberOfparameters * sizeof(uint32_t)), + 0, &remote, sizeof(remote)))<0) { + printf("Send failed (%i - %s)\n", i, strerror(errno)); + return -1; + } + +// // to avoid memory leaking +// free(msg.parameters); + return 0; +} + +/** + * Verifies if the machine architecture is big endian + * + * @return true if the machine architecture is big endian + */ +int isBigEndian(void) +{ + union { + uint32_t i; + char c[4]; + } e = { 0x01000000 }; + + return e.c[0]; +} + +/** + * Network order, byte swap function for 64bits size numbers + * + * @param value - the 64 bits number to swap, depending on the architecture + * @return the same value in the network order + */ +uint64_t htonll(uint64_t value) +{ + // Check the endianness + if (!isBigEndian()) + { + uint32_t high_part = htonl((uint32_t)(value >> 32)); + uint32_t low_part = htonl((uint32_t)(value & 0xFFFFFFFFULL)); + value = low_part & 0xffffffffULL; + value = (value<<32) | high_part; + value = (uint64_t)(((uint64_t)low_part) << 32) | high_part; + return (uint64_t)(((uint64_t)low_part) << 32) | high_part; + } else + { + return value; + } +} + +/** + * Host order, byte swap function for 64bits size numbers + * + * @param value - the 64 bits number to swap, depending on the architecture + * @return the same value in the host order + */ +uint64_t ntohll(uint64_t value) +{ + + // Check the endianness + if (!isBigEndian()) + { + uint32_t high_part = ntohl((uint32_t)(value >> 32)); + uint32_t low_part = ntohl((uint32_t)(value & 0xFFFFFFFFULL)); + + return (uint64_t)(((uint64_t)low_part) << 32) | high_part; + } else + { + return value; + } +} + +#endif /* SPECTRA_H_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectraClient b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectraClient new file mode 100644 index 0000000000000000000000000000000000000000..bfe90331b32567183631127cf5842b8cde1c9fac Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectraClient differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_client.c b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_client.c new file mode 100644 index 0000000000000000000000000000000000000000..7adc6aa44d1caab2cd7a43ba912e4f7a7fb8bde5 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_client.c @@ -0,0 +1,140 @@ +/* + * spectra_client.c + * + * Client part of the spectra demo control system. It is + * supposed to run on the OpenAirInterface machine that + * will send commands to configure Embb and collect the + * answers sent by the Embb "server", running on the + * board. + * + * Considering this compiled with gcc -o spectraClient spectra_client.c. + * + * Use: spectraClient [server IP] [port] [nSamples] + * e.g: spectraClient 127.0.0.1 4546 10 + * - This will tell the client to try to find the server in the + * address 127.0.0.1 at the port 4546 and that the score should be + * calculated over 10 samples. + * + * Created on: Dec 11, 2013 + * Author: camara + * Copyright (c) 2008-2014 Institut Mines-Telecom / Telecom ParisTech + */ +#include <stdio.h> // printf +#include <string.h> // strlen +#include <unistd.h> // close +#include <sys/socket.h> // socket +#include <arpa/inet.h> // inet_addr +#include <stdlib.h> // malloc +#include "spectra_client.h" // The types and generic functions of spectra + +/** + * Creates the test message + * + * @param id messsage identificatior + * @return example message + */ +SpectraMessage createTestMessageClient(int id, int type, int nSamples) { + SpectraMessage msg; + + msg.type = type; + msg.function = EnergyDetection; + msg.messageID = id; + msg.numberOfparameters = 1; + + msg.parameters = malloc(msg.numberOfparameters * sizeof(uint32_t)); + msg.parameters[0] = nSamples; + + return msg; +} + +/** + * Creates the socket and connects to the server + * + * @return - the socket descriptor + */ +int createsSocket(int argc, char *argv[]) { + int sock; + struct sockaddr_in server; + //Create socket + + sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); + if (sock == -1) { + printf("Could not create socket"); + return -1; + } + + puts("Socket created"); + + // treats the parameters + if (argc >= 2) + server.sin_addr.s_addr = inet_addr(argv[1]); + else + server.sin_addr.s_addr = inet_addr(SERVER_ADDRESS); + + server.sin_family = AF_INET; + + if (argc >= 3) + server.sin_port = htons(atoi(argv[2])); + else + server.sin_port = htons(CONNECTION_PORT); + + printf("Target address: %s, port %d : addr\n", inet_ntoa(server.sin_addr), + server.sin_port); + + //Connect to remote server + if (connect(sock, (struct sockaddr *) &server, sizeof(server)) < 0) { + perror("connect failed. Error"); + return -2; + } + + puts("Connected\n"); + return sock; +} + +/** + * Main function for the client + * + * @param argc + * @param argv + * @return + */ +int main(int argc, char *argv[]) { + char server_reply[MAX_PACKET_SIZE]; + int nSamples = 10; + + int sock = createsSocket(argc, argv), id; + + if (argc >= 4) + nSamples = atoi(argv[3]); + + if (sock < 0) + return sock; + + int type = Classifying; + // Just for test purposes send 2 packets to the server, changing only the packet id + for (id = 33; id < 36; ++id) { + printf("-->\nSending message ID : %d\n", id); + + if (id == 35) + type = EndProcessing; + + SpectraMessage message = createTestMessageClient(id, type, nSamples); + + //Send some data + if (sendClientSpectraPacket(sock, message) != 0) { + return 1; + } + + //Receive a reply from the server + if (recv(sock, server_reply, MAX_PACKET_SIZE, 0) < 0) { + puts("recv failed"); + break; + } + + printf("<--\nServer reply :\n"); + printReturnValueMessage(assembleMessage(&server_reply)); + } + + close(sock); + return 0; +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_client.h b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_client.h new file mode 100644 index 0000000000000000000000000000000000000000..deba8b4cf4ff48745892f3d5169915023896dbf8 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_client.h @@ -0,0 +1,240 @@ +/* + * spectra_client.h + * + * Generic functions and types for the spectra demonstration + * control system. These functions are used by the client side. + * + * Created on: Dec 11, 2013 + * Author: camara + * Copyright (c) 2008-2014 Institut Mines-Telecom / Telecom ParisTech + */ + +#ifndef SPECTRA_H_ +#define SPECTRA_H_ + +/** + * The IP address of the Embb server (e.g the Embb board) + */ +//#define SERVER_ADDRESS "192.168.0.2" +#define SERVER_ADDRESS "127.0.0.1" + +/** + * Connection port + */ +#define CONNECTION_PORT 4546 + +/** + * Alinges packet, internally, into 32 bits, or 4 bytes + */ +#define PACKET_ALIGNEMENT_BYTES 4 + +/** + * Maximum packet size, dependent on the size of the + * number of elements field on the header, at the present 8 bits + * i.e. (256 * 4 bytes + the header that is 4 bytes) + */ +#define MAX_PACKET_SIZE 1028 + +/** + * Which is the type of the message, if it is, for example, an start sensing, + * or an answer to a previously requested score + */ +typedef enum { + Sense, //!< Sense + Classifying, //!< Classifying + ScoreResponse, //!< ScoreResponse + EndProcessing +} messageType; + +/** + * Which is the functions that should be performed by the embb + */ +//! An enum. +/*! More detailed enum description. */ +typedef enum { + EnergyDetection, //!< EnergyDetection + WelchPeriodograms //!< WelchPeriodograms +} functionToBePerformed; + +/** + * Definition of the spectra control packets + */ +typedef struct { + uint16_t messageID; //!< Operation ID, used to identify an request and its answer + messageType type :4; //!< 16 different types of message possible + functionToBePerformed function :4; //!< Embb function to be performed 16 possible different functions + unsigned char numberOfparameters; //!< up to 255 parameters per message + uint32_t * parameters; //!< variable number of parameters, depending + //!< on what we are doing and the default values +}__attribute__((packed)) SpectraMessage; + +/** + * \typedef SpectraMsgHelper + * Helper structure to handle control packets + */ +typedef struct { + uint16_t messageID; //!< Operation ID + unsigned char type_function; //!< Assembled type and function for easier treatment + unsigned char numberOfparameters; //!< Number of parameters +// uint32_t parameter; //!< Parameters in sequence + uint32_t parameters[256]; //!< Parameters in sequence +}__attribute__((packed)) SpectraMsgHelper; + +/** + * Helper union structure to make easier to pack and unpack messages + */ +union uSpectraPackets { + SpectraMessage spectraMsg; //!< spectraMsg + SpectraMsgHelper spectraMsgHelper; //!< spectraMsgHelper + unsigned char spectraMsgStream[MAX_PACKET_SIZE]; //!< spectraMsgStream +}; + +/** + * Verifies if the machine architecture is big endian + * + * @return true if the machine architecture is big endian + */ +int isBigEndian(void) +{ + union { + uint32_t i; + char c[4]; + } e = { 0x01000000 }; + + return e.c[0]; +} + +/** + * Network order, byte swap function for 64bits size numbers + * + * @param value - the 64 bits number to swap, depending on the architecture + * @return the same value in the network order + */ +uint64_t htonll(uint64_t value) +{ + // Check the endianess + if (!isBigEndian()) + { + uint32_t high_part = htonl((uint32_t)(value >> 32)); + uint32_t low_part = htonl((uint32_t)(value & 0xFFFFFFFFULL)); + return (uint64_t)(((uint64_t)low_part) << 32) | high_part; + } else + { + return value; + } +} + +/** + * Host order, byte swap function for 64bits size numbers + * + * @param value - the 64 bits number to swap, depending on the architecture + * @return the same value in the host order + */ +uint64_t ntohll(uint64_t value) +{ + + // Check the endianess + if (!isBigEndian()) + { + uint32_t high_part = ntohl((uint32_t)(value >> 32)); + uint32_t low_part = ntohl((uint32_t)(value & 0xFFFFFFFFULL)); + + return (uint64_t)(((uint64_t)low_part) << 32) | high_part; + } else + { + return value; + } +} + + +/** + * Serialize and send packet over the indicated socket + * + * @param sock - in use socket descriptor + * @param msg - message to be transmitted + * @return error status 0 if OK, 1 if something went wrong + */ +int sendClientSpectraPacket(int sock, SpectraMessage msg) { + int i; + union uSpectraPackets toTransmitPacket; + toTransmitPacket.spectraMsg = msg; + + // Puts the values bigger than one byte on Internet format + toTransmitPacket.spectraMsgHelper.messageID = htons(msg.messageID); + + toTransmitPacket.spectraMsgHelper.type_function = (msg.type << 4) | msg.function; + for (i = 0; i < msg.numberOfparameters; i++) { + toTransmitPacket.spectraMsgHelper.parameters[i] = htonl(msg.parameters[i]); + printf("Value %d=%d\n", i, htonl(msg.parameters[i])); + } + + // sends the package + if ( (i = send(sock, toTransmitPacket.spectraMsgStream, PACKET_ALIGNEMENT_BYTES + (msg.numberOfparameters * sizeof(uint32_t)), 0)) < 0) { + printf("Send failed (%i)\n", i); + // to avoid memory leaking + free(msg.parameters); + return -1; + } + + int k = (int)(PACKET_ALIGNEMENT_BYTES + (msg.numberOfparameters * sizeof(uint32_t))); + printf("Sent message size: %d\n >", k); + + for (i=0; i<k;i++){ + printf("%x ", (toTransmitPacket.spectraMsgStream[i] & 0xff)); + } + printf("<\n"); + // to avoid memory leaking + free(msg.parameters); + return 0; +} + +/** + * Assemble the packet from the raw input packet + * @param var - void pointer to the memory area the message is stored + * @return a formated Spectra message + */ +SpectraMessage assembleMessage(void * var) { + int i; + SpectraMessage returnMessage; + char nParameters = (*(SpectraMessage*) var).numberOfparameters; + + // Alocates enough memory area for the variable parameters field + returnMessage.parameters = malloc(nParameters * sizeof(uint32_t)); + + returnMessage.messageID = ntohs((*(SpectraMessage*) var).messageID); + returnMessage.type = ((*(SpectraMsgHelper*) var).type_function) >>4; + returnMessage.function = (((*(SpectraMsgHelper*) var).type_function << 4)>>4) & 0xF; + returnMessage.numberOfparameters = nParameters; + + for (i = 0; i < nParameters; i++) { + returnMessage.parameters[i] = (*(SpectraMsgHelper*) var).parameters[i]; + } + + return returnMessage; +} + +/** + * Prints the test message, only for debugging purposes + * + * @param msg - message to print the components + */ +void printReturnValueMessage(SpectraMessage msg) { +// int i; + + printf("msg.messageID: %d\n", msg.messageID); + printf("msg.type: %d\n", msg.type); + printf("msg.function: %d\n", msg.function); +// printf("msg.numberOfparameters: %d\n", msg.numberOfparameters); +// for (i = 0; i < msg.numberOfparameters; i++) { +// printf("[%d] parameters: %d\n", i, msg.parameters[i]); +// } + + // To transform the 2x32bits in on 64bits value + uint64_t realValue = (uint64_t)(((uint64_t)msg.parameters[0]) << 32) | msg.parameters[1]; + + printf("packet value: 0x%016llx\n", (long long int)realValue); + printf("Network to host value: 0x%016llx\n", (long long int)ntohll(realValue)); + printf("SCORE: %lld\n", (long long int)ntohll(realValue)); +} + +#endif /* SPECTRA_H_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_deamon.c b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_deamon.c new file mode 100644 index 0000000000000000000000000000000000000000..c4e75d456b74f819b9bacd2fd21fb91b0f66fbac --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/client/spectra_deamon.c @@ -0,0 +1,225 @@ +/* + * spectra_deamon.c + * + * Server part of the spectra demo control system. + * + * The real code is supposed to run on the board waiting from the + * OpenAirInterface configuration messages. This is just a "stub" + * version of the real server. It emulates the behavior of the + * server for tests purposes only. It has a list of scores that + * it plays over and over. + * + * Created on: Dec 11, 2013 + * Author: camara + * Copyright (c) 2008-2014 Institut Mines-Telecom / Telecom ParisTech + */ + +#include <stdio.h> +#include <string.h> //strlen +#include <stdlib.h> //malloc +#include <sys/socket.h> +#include <arpa/inet.h> //inet_addr +#include <unistd.h> //write +#include <pthread.h> //for threading , link with lpthread +#include "spectra.h" // The types and generic functions of spectra + +/** + * List of fake scores that will be sent to the application + */ +uint64_t scoresList[] = { + 81886716, + 94073410, + 147421016, + 117364836, + 104805562, + 142811370, + 107986264, + 100032514 + }; + +/** + * Index of the score to send + */ +int scoreIndex = 0; + +/** + * Creates the return packet + * + * Fills the packet that will be returned to the OpenAir interface with the + * score or with the end of connection + * + * @param request - The original request message + * @param score - The score value + * @param request - The type of this message + * + * @return - A new spectra message filled with the right fields + */ +SpectraMsgHelper createReturnPacket(SpectraMessage* request, uint64_t score, + messageType type) { + SpectraMsgHelper returnMessage; + + returnMessage.messageID = htons(request->messageID); // copy the request id + returnMessage.type_function = (type << 4) | request->function; + returnMessage.numberOfparameters = 2; // it is a 64bits number (2 x uint32_t) + score = htonll(score); + returnMessage.parameters[0] = score >> 32; // the score number to send back but it is a 64 bit + returnMessage.parameters[1] = score & 0xFFFFFFFFULL; + + return returnMessage; +} + +/** + * Assemble the packet from the raw input packet + * @param var - void pointer to the memory area the message is stored + * @return a formated Spectra message + */ +SpectraMessage assembleMessage(void * var) { + int i; + SpectraMessage returnMessage; + char nParameters = (*(SpectraMessage*) var).numberOfparameters; + + // Alocates enough memory area for the variable parameters field + returnMessage.parameters = malloc(nParameters * sizeof(uint32_t)); + + returnMessage.messageID = ntohs((*(SpectraMessage*) var).messageID); + returnMessage.type = ((*(SpectraMsgHelper*) var).type_function) >> 4; + returnMessage.function = (((*(SpectraMsgHelper*) var).type_function << 4)>> 4) & 0xF; + returnMessage.numberOfparameters = nParameters; + + for (i = 0; i < nParameters; i++) { + returnMessage.parameters[i] = (*(SpectraMsgHelper*) var).parameters[i]; + } + + return returnMessage; +} + +/** + * Responsible for handling the client connections + * + * @param sock - socket descriptor + */ +void *connection_handler(int fd) { + SpectraMessage request; // received message in the spectra format + SpectraMsgHelper returnMessage; // message in the format we want to send + char rawPacket[MAX_PACKET_SIZE]; // raw packet + struct sockaddr remaddr; // remote address + socklen_t addrlen = sizeof(remaddr); // length of addresses + int recvlen; // # bytes received + int i=0; + + /* now loop, receiving data and printing what we received */ + for (;;) { + printf("---\n Waiting on port %d\n", PORT); + recvlen = recvfrom(fd, rawPacket, MAX_PACKET_SIZE, 0, + (struct sockaddr *) &remaddr, &addrlen); + printf("Received %d bytes > ", recvlen); + + for (i=0; i<recvlen;i++){ + printf("%x ", (rawPacket[i] & 0xff)); + } + printf("<\n"); + + + // prints the packet + if (recvlen <= 0) { + perror("Receive failed!!!!\n"); + } else { + + request = assembleMessage(&rawPacket); + printf(" > Received spectra packet %i of type %i for function %i with %i parameters\n", + request.messageID, + (uint32_t) request.type, (uint32_t) request.function, + (uint32_t) request.numberOfparameters); + + // Process the requested operation + if (request.type == EndProcessing) { + + returnMessage = createReturnPacket(&request, 0, EndProcessing); + printf("End processing!!!\n"); + fflush(stdout); + } else { + if (request.function == EnergyDetection) { + + // circular loop over the score list + scoreIndex = (scoreIndex < (sizeof(scoresList)/ sizeof(scoresList[0]))) ? + scoreIndex : 0; + + printf("Score value: %16llx (%lld)\n", + (long long int)scoresList[scoreIndex], + (long long int)scoresList[scoreIndex]); + printf("Score host to network value: 0x%016llx (%lld)\n", + (long long int)htonll(scoresList[scoreIndex]), + (long long int)htonll(scoresList[scoreIndex])); + + returnMessage = createReturnPacket(&request, + scoresList[scoreIndex++], ScoreResponse); + } + } + + // Sends the answer + int retError = returnSpectraPacket(returnMessage, fd, remaddr); + if (retError == -1) { + perror("Send failed!!!\n"); + } + + // If the command was an end processing exits!! + if (request.type == EndProcessing) { + break; + } + free(request.parameters); + } + + } + //Free the socket pointer + close(fd); + return 0; +} + +/** + * Creates the socket and binds it to the defined port + * + * @return the socket descriptor + */ +int socketBind() { + + struct sockaddr_in myaddr; /* our address */ +// struct sockaddr_in remaddr; /* remote address */ + int fd; /* our socket */ + + /* create a UDP socket */ + + if ((fd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { + perror("cannot create socket\n"); + return 0; + } + + /* bind the socket to any valid IP address and a specific port */ + memset((char *) &myaddr, 0, sizeof(myaddr)); + myaddr.sin_family = AF_INET; + myaddr.sin_addr.s_addr = htonl(INADDR_ANY); + myaddr.sin_port = htons(PORT); + + if (bind(fd, (struct sockaddr *) &myaddr, sizeof(myaddr)) < 0) { + perror("bind failed"); + return 0; + } + puts("bind done"); + return fd; +} + +/** + * The main function for the server + * + * @param argc + * @param argv + * @return + */ +int main(int argc, char *argv[]) { +// int client_sock; + int socket_desc; + + socket_desc = socketBind(); + connection_handler(socket_desc); + + return 0; +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/clientToFatma.zip b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/clientToFatma.zip new file mode 100644 index 0000000000000000000000000000000000000000..a9539173e6330b9bcca6b4b86c62f6e144644926 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/clientSensing/clientToFatma.zip differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/0001-SPECTRA-patch-for-ODTONE-v0.1.patch b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/0001-SPECTRA-patch-for-ODTONE-v0.1.patch new file mode 100644 index 0000000000000000000000000000000000000000..73c6bf9a8c7957c3456b824c730140f42c528369 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/0001-SPECTRA-patch-for-ODTONE-v0.1.patch @@ -0,0 +1,17622 @@ +From e4048b1089cb2dff78898358d6de7638554bf204 Mon Sep 17 00:00:00 2001 +From: nikaia <nikaia@nikaia.(none)> +Date: Wed, 17 Sep 2014 17:27:20 +0200 +Subject: [PATCH] SPECTRA patch for ODTONE v0.1 + +--- + app/lte_test_user/CRMClient.cpp | 260 +++++ + app/lte_test_user/CRMClient.hpp | 94 ++ + app/lte_test_user/Jamfile | 44 + + app/lte_test_user/Makefile | 19 + + app/lte_test_user/eNB_lte_user_tcs.cpp | 1412 ++++++++++++++++++++++++++ + app/lte_test_user/enb_lte_user.conf | 40 + + app/lte_test_user/enb_lte_user.cpp | 1695 ++++++++++++++++++++++++++++++++ + app/lte_test_user/libcrmclient.so.1.0 | Bin 0 -> 2273132 bytes + app/lte_test_user/ue_lte_user.conf | 42 + + app/lte_test_user/ue_lte_user.cpp | 1399 ++++++++++++++++++++++++++ + boost-build.jam | 1 + + inc/odtone/conf.hpp | 670 +++++++++++++ + inc/odtone/mih/detail/archive.hpp | 142 +++ + inc/odtone/mih/types/address.hpp | 19 +- + inc/odtone/mih/types/bin_query.hpp | 1 + + inc/odtone/mih/types/link.hpp | 419 ++++++-- + lib/odtone/CMakeLists.txt | 1 + + lib/odtone/Jamfile | 3 +- + lib/odtone/conf.cpp | 43 + + src/mihf/command_service.cpp | 1 + + src/mihf/dst_transaction.cpp | 7 + + src/mihf/event_service.cpp | 16 +- + src/mihf/link_book.cpp | 59 +- + src/mihf/main.cpp | 32 +- + src/mihf/service_access_controller.cpp | 6 +- + src/mihf/service_management.cpp | 1 + + src/mihf/src_transaction.cpp | 7 + + src/mihf/transaction_pool.cpp | 7 + + 28 files changed, 6366 insertions(+), 74 deletions(-) + create mode 100644 app/lte_test_user/CRMClient.cpp + create mode 100644 app/lte_test_user/CRMClient.hpp + create mode 100644 app/lte_test_user/Jamfile + create mode 100644 app/lte_test_user/Makefile + create mode 100644 app/lte_test_user/eNB_lte_user_tcs.cpp + create mode 100644 app/lte_test_user/enb_lte_user.conf + create mode 100644 app/lte_test_user/enb_lte_user.cpp + create mode 100644 app/lte_test_user/libcrmclient.so.1.0 + create mode 100644 app/lte_test_user/ue_lte_user.conf + create mode 100644 app/lte_test_user/ue_lte_user.cpp + create mode 100644 boost-build.jam + create mode 100644 inc/odtone/conf.hpp + create mode 100644 lib/odtone/conf.cpp + +diff --git a/app/lte_test_user/CRMClient.cpp b/app/lte_test_user/CRMClient.cpp +new file mode 100644 +index 0000000..8e384e7 +--- /dev/null ++++ b/app/lte_test_user/CRMClient.cpp +@@ -0,0 +1,260 @@ ++#include <cpprest/http_client.h> ++#include <cpprest/json.h> ++#include <iostream> ++#include <ostream> ++#include <sstream> ++#include <fstream> ++#include "cpprest/basic_types.h" ++#include "cpprest/asyncrt_utils.h" ++#include "cpprest/uri.h" ++#include <string> ++ ++#include "CRMClient.hpp" ++ ++using namespace std; ++using namespace web; ++using namespace web::http; ++using namespace web::http::client; ++using namespace utility; ++ ++ ++CRMClient::CRMClient(char* url) ++{ ++ m_url = (char*) (malloc (200* sizeof (char))); ++ m_url = url; ++} ++ ++CRMClient::CRMClient() ++{ ++} ++ ++CRMClient::~CRMClient() ++{ ++} ++ ++void CRMClient::SetUrl(char* url) ++{ ++ m_url = url; ++} ++ ++ ++/****************************************************** ++* Retrieves a JSON value from an HTTP request ++* The JSON value is stored in the file "outputGET.txt" ++*******************************************************/ ++ ++pplx::task<void> CRMClient::RequestJSONValueAsync() ++{ ++ http_client client (U(m_url)); ++ return client.request(methods::GET).then([](http_response response) -> pplx::task<json::value> ++ { ++ if(response.status_code() == status_codes::OK) ++ { ++ return response.extract_json(); ++ } ++ ++ return pplx::task_from_result(json::value()); ++ }) ++ .then([](pplx::task<json::value> previousTask) ++ { ++ try ++ { ++ ofstream myfile; ++ myfile.open ("outputGET.txt"); ++ const json::value& v = previousTask.get(); ++ string_t jsonString = v.to_string(); ++ cout << U("Response...") << jsonString <<endl; ++ ++//Parsing Begin ++ ++ cout << U("Start Parsing...") << endl; ++ for(auto iterArray = v.cbegin(); iterArray != v.cend(); ++iterArray) ++ { ++ const json::value &arrayValue = iterArray->second; ++ ++ for(auto iterInner = arrayValue.cbegin(); iterInner != arrayValue.cend(); ++iterInner) ++ { ++ const json::value &propertyValue = iterInner->second; ++ for(auto iterlast = propertyValue.cbegin(); iterlast != propertyValue.cend(); ++iterlast) ++ { ++ const json::value &Name = iterlast->first; ++ const json::value &Value = iterlast->second; ++ cout<< U("Parameter: ") << Name.to_string()<<endl; ++ cout<< U("Value: ") << Value.to_string()<< std::endl; ++ } ++ } ++ cout << std::endl; ++ } ++ ++//parsing End ++ myfile << jsonString.c_str(); ++ myfile.close(); ++ } ++ catch (const http_exception& e) ++ { ++ wostringstream ss; ++ ss << e.what() << endl; ++ wcout << ss.str(); ++ } ++ }); ++ ++} ++ ++/******************************************************* ++* Stores a JSON value (a Policy) using a HTTP request ++********************************************************/ ++ ++pplx::task<void> CRMClient::StoreJSONValuePolicies(char * param1, char * param2, char * param3, char * param4) ++{ ++ http_client client (U(m_url)); ++ json::value::field_map putvalue; ++ ++ putvalue.push_back(make_pair(json::value("pid"), json::value(param1))); ++ putvalue.push_back(make_pair(json::value("name"), json::value(param2))); ++ putvalue.push_back(make_pair(json::value("description"), json::value(param3))); ++ putvalue.push_back(make_pair(json::value("value"), json::value(param4))); ++ ++ const string_t& s = "/"; ++ json::value object = json::value::object(putvalue); ++ return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> ++ { ++ if(response.status_code() == status_codes::OK) ++ { ++ return response.extract_json(); ++ } ++ ++ return pplx::task_from_result(json::value()); ++ }) ++ .then([](pplx::task<json::value> previousTask) ++ { ++ try ++ { ++ const json::value& v = previousTask.get(); ++ string_t jsonString = v.to_string(); ++ cout << U("Response...") << jsonString <<endl; ++ } ++ catch (const http_exception& e) ++ { ++ wostringstream ss; ++ ss << e.what() << endl; ++ wcout << ss.str(); ++ } ++ }); ++} ++ ++ ++/********************************************************** ++* Stores a JSON value (a Measurement) using a HTTP request ++***********************************************************/ ++ ++pplx::task<void> CRMClient::StoreJSONValuemeasurements(char * param1, char * param2, char * param3, char * param4, char * param5, char * param6) ++{ ++ ++ http_client client (U(m_url)); ++ json::value::field_map putvalue; ++ ++ putvalue.push_back(make_pair(json::value("key"), json::value(param1))); ++ putvalue.push_back(make_pair(json::value("name"), json::value(param2))); ++ putvalue.push_back(make_pair(json::value("type"), json::value(param3))); ++ putvalue.push_back(make_pair(json::value("unit"), json::value(param4))); ++ putvalue.push_back(make_pair(json::value("value"), json::value(param5))); ++ putvalue.push_back(make_pair(json::value("time"), json::value(param6))); ++ ++ const string_t& s = ""; ++ json::value object = json::value::object(putvalue); ++ return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> ++ { ++ if(response.status_code() == status_codes::OK) ++ { ++ return response.extract_json(); ++ } ++ ++ return pplx::task_from_result(json::value()); ++ }) ++ .then([](pplx::task<json::value> previousTask) ++ { ++ try ++ { ++ const json::value& v = previousTask.get(); ++ string_t jsonString = v.to_string(); ++ cout << U("Response...") << jsonString <<endl; ++ } ++ catch (const http_exception& e) ++ { ++ wostringstream ss; ++ ss << e.what() << endl; ++ wcout << ss.str(); ++ } ++ }); ++ ++} ++ ++/******************************************************** ++* Stores a JSON value (a Decision) using a HTTP request ++*********************************************************/ ++ ++pplx::task<void> CRMClient::StoreJSONValuedecisions(char * param1, char * param2, char * param3, char * param4, char * param5) ++{ ++ ++ http_client client (U(m_url)); ++ json::value::field_map putvalue; ++ ++ putvalue.push_back(make_pair(json::value("did"), json::value(param1))); ++ putvalue.push_back(make_pair(json::value("name"), json::value(param2))); ++ putvalue.push_back(make_pair(json::value("description"), json::value(param3))); ++ putvalue.push_back(make_pair(json::value("value"), json::value(param4))); ++ putvalue.push_back(make_pair(json::value("time"), json::value(param5))); ++ ++ const string_t& s = "/"; ++ json::value object = json::value::object(putvalue); ++ return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> ++ { ++ if(response.status_code() == status_codes::OK) ++ { ++ return response.extract_json(); ++ } ++ ++ return pplx::task_from_result(json::value()); ++ }) ++ .then([](pplx::task<json::value> previousTask) ++ { ++ try ++ { ++ const json::value& v = previousTask.get(); ++ string_t jsonString = v.to_string(); ++ cout << U("Response...") << jsonString <<endl; ++ } ++ catch (const http_exception& e) ++ { ++ wostringstream ss; ++ ss << e.what() << endl; ++ wcout << ss.str(); ++ } ++ }); ++ ++} ++ ++ ++/******************************************************** ++* Deletes entities using a HTTP request ++*********************************************************/ ++ ++pplx::task<void> CRMClient::Delete() ++{ ++ char* url = m_url; ++ return pplx::create_task([url] ++ { ++ http_client client (U(url)); ++ ++ return client.request(methods::DEL); ++ ++ }).then([](http_response response) ++ { ++ if(response.status_code() == status_codes::OK) ++ { ++ auto body = response.extract_string(); ++ ++ std::wcout << L"Deleted: " << body.get().c_str() << std::endl; ++ } ++ }); ++} +diff --git a/app/lte_test_user/CRMClient.hpp b/app/lte_test_user/CRMClient.hpp +new file mode 100644 +index 0000000..a49f2e6 +--- /dev/null ++++ b/app/lte_test_user/CRMClient.hpp +@@ -0,0 +1,94 @@ ++//============================================================================== ++// Brief : MIH-User ++// Authors : FATMA HRIZI <hrizi@eurecom.fr> ++//------------------------------------------------------------------------------ ++// ODTONE - Open Dot Twenty One ++// ++// Copyright (C) 2009-2012 Universidade Aveiro ++// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++// ++// This software is distributed under a license. The full license ++// agreement can be found in the file LICENSE in this distribution. ++// This software may not be copied, modified, sold or distributed ++// other than expressed in the named license agreement. ++// ++// This software is distributed without any warranty. ++//============================================================================== ++ ++#ifndef ODTONE_CRM_CLIENT_HPP ++#define ODTONE_CRM_CLIENT_HPP ++ ++ ++#include <cpprest/http_client.h> ++#include <cpprest/json.h> ++#include <iostream> ++#include <ostream> ++#include <sstream> ++#include "cpprest/basic_types.h" ++#include "cpprest/asyncrt_utils.h" ++#include "cpprest/uri.h" ++#include <string> ++ ++ ++/** ++ * CRM Client Class ++ * ++ * Defines the required functions to Get/Store Data from/in the CRRM ++ * ++ **/ ++class CRMClient { ++ ++public: ++ ++ CRMClient(); ++ CRMClient(char* url); ++ ~CRMClient(); ++ ++ void SetUrl(char* url); ++ ++ /** ++ * Get the data from the CRRM ++ * ++ * Returns JSON Format ++ */ ++ pplx::task<void> RequestJSONValueAsync(); ++ ++ /** ++ * Store Policies data from the CRRM ++ * ++ * Takes 4 parameters: P_ID, Name, Description, Value ++ * Returns the JSON format of the stored entity ++ */ ++ pplx::task<void> StoreJSONValuePolicies(char * param1, char * param2, char * param3, char * param4); ++ ++ /** ++ * Store Measurements data from the CRRM ++ * ++ * Takes 5 parameters: M_ID, Name, Type, Unit, Value, Time ++ * Returns the JSON format of the stored entity ++ */ ++ pplx::task<void> StoreJSONValuemeasurements(char * param1, char * param2, char * param3, char * param4, char * param5, char * param6); ++ ++ /** ++ * Store Decisions data from the CRRM ++ * ++ * Takes 4 parameters: D_ID, Name, Description, Value ++ * Returns the JSON format of the stored entity ++ */ ++ pplx::task<void> StoreJSONValuedecisions(char * param1, char * param2, char * param3, char * param4, char * param5); ++ ++ ++ ++ /** ++ * Delete Entities from the CRRM ++ * ++ * Returns the JSON format of the deleted entity ++ */ ++ pplx::task<void> Delete(); ++ ++ ++private: ++ char * m_url; ++ ++}; ++#endif +diff --git a/app/lte_test_user/Jamfile b/app/lte_test_user/Jamfile +new file mode 100644 +index 0000000..c2a6aec +--- /dev/null ++++ b/app/lte_test_user/Jamfile +@@ -0,0 +1,44 @@ ++#=============================================================================== ++# Brief : MIH-User SAP Application Sample Project Build ++# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> ++# Bruno Santos <bsantos@av.it.pt> ++#------------------------------------------------------------------------------- ++# ODTONE - Open Dot Twenty One ++# ++# Copyright (C) 2009-2012 Universidade Aveiro ++# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++# ++# This software is distributed under a license. The full license ++# agreement can be found in the file LICENSE in this distribution. ++# This software may not be copied, modified, sold or distributed ++# other than expressed in the named license agreement. ++# ++# This software is distributed without any warranty. ++#=============================================================================== ++ ++project enb_lte_user ++ ; ++ ++exe enb_lte_user ++ : enb_lte_user.cpp ++ ../../lib/odtone//odtone ++ /boost//program_options ++ ; ++ ++install install ++ : enb_lte_user ++ enb_lte_user.conf ++ ue_lte_user ++ ue_lte_user.conf ++ : <location>../../dist ++ ; ++ ++project ue_lte_user ++ ; ++ ++exe ue_lte_user ++ : ue_lte_user.cpp ++ ../../lib/odtone//odtone ++ /boost//program_options ++ ; ++ +diff --git a/app/lte_test_user/Makefile b/app/lte_test_user/Makefile +new file mode 100644 +index 0000000..b6837ad +--- /dev/null ++++ b/app/lte_test_user/Makefile +@@ -0,0 +1,19 @@ ++main: main.cpp ++ g++-4.8 -Wall main.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -L. -std=c++11 -lcrmclient -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem -o CRMClientmain ++ ++lib: CRMClient.cpp ++ ++# 64 bits compilation ++# g++-4.8 -Wall -fPIC -c CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -m64 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem ++# g++-4.8 -shared -m64 -Wl,-soname,libcrmclient.so.1 -o libcrmclient.so.1.0 CRMClient.o ++# ln -sf libcrmclient.so.1.0 libcrmclient.so ++# ln -sf libcrmclient.so.1.0 libcrmclient.so.1 ++ ++# 32 bits compilation ++ g++-4.8 -Wall -fPIC -c CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem ++ g++-4.8 -shared -Wl,-soname,libcrmclient.so.1 -o libcrmclient.so.1.0 CRMClient.o ++ ln -sf libcrmclient.so.1.0 libcrmclient.so ++ ln -sf libcrmclient.so.1.0 libcrmclient.so.1 ++ ++clean: ++ rm *.o *.so* +diff --git a/app/lte_test_user/eNB_lte_user_tcs.cpp b/app/lte_test_user/eNB_lte_user_tcs.cpp +new file mode 100644 +index 0000000..72d46e7 +--- /dev/null ++++ b/app/lte_test_user/eNB_lte_user_tcs.cpp +@@ -0,0 +1,1412 @@ ++//============================================================================== ++// Brief : MIH-User ++// Authors : Bruno Santos <bsantos@av.it.pt> ++// Fatma HRIZI EURECOM <hrizi@eurecom>fr> ++//------------------------------------------------------------------------------ ++// ODTONE - Open Dot Twenty One ++// ++// Copyright (C) 2009-2012 Universidade Aveiro ++// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++// ++// This software is distributed under a license. The full license ++// agreement can be found in the file LICENSE in this distribution. ++// This software may not be copied, modified, sold or distributed ++// other than expressed in the named license agreement. ++// ++// This software is distributed without any warranty. ++//============================================================================== ++ ++//This file is the implementation of the MIH user in the eNB ++ ++#include <odtone/base.hpp> ++#include <odtone/debug.hpp> ++#include <odtone/logger.hpp> ++#include <odtone/mih/request.hpp> ++#include <odtone/mih/response.hpp> ++#include <odtone/mih/indication.hpp> ++#include <odtone/mih/confirm.hpp> ++#include <odtone/mih/tlv_types.hpp> ++#include <odtone/sap/user.hpp> ++ ++#include <boost/utility.hpp> ++#include <boost/bind.hpp> ++#include <boost/tokenizer.hpp> ++#include <boost/foreach.hpp> ++#include <boost/format.hpp> ++ ++#include <iostream> ++#include <map> ++#include <time.h> ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++// Definition of the scenario to execute ++#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests ++//#define SCENARIO_1 // Sequentially activate and deactivate each resource ++#define SCENARIO_2 // Activate all resources, then deactivate all resources ++#define NUM_PARM_REPORT 10 ++ ++/////////////////////////////////////////////////////////////////////////////// ++// The scenario coded in this MIH-USER is the following (with eRALlteDummy and NASRGDummy executables) ++// +--------+ +-----+ ++// |MIH_USER| |MIH-F| ++// +---+----+ +--+--+ ++// | | _current_link_action_request = 0 ++// |---------- User_Register.indication ---------------->| (supported_commands) Handler next msg=user_reg_handler ++// | | ++// |---------- Capability_Discover.request ------------->| Handler next msg=receive_MIH_Capability_Discover_confirm ++// |<--------- Capability_Discover.confirm --------------| (success) ++// | | ++// |---------- Event_Subscribe.request ----------------->| Handler next msg=receive_MIH_Event_Subscribe_confirm ++// |<--------- Event_Subscribe.confirm ------------------| (success) ++// | | ++// ------------------------------------------------------------------------------------------------------------------------ ++// Scenario 1: Sequentially activate and deactivate each resource ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | ++// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) ++// | | Handler next msg=receive_MIH_Link_Actions_confirm ++// |<--------- Link_Actions.confirm ---------------------| (success) ++// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) ++// | | Handler next msg=receive_MIH_Link_Actions_confirm ++// | | _current_link_action_request = _current_link_action_request + 1 ++// |<--------- Link_Actions.confirm ---------------------| (success) ++// | . | ++// | . | ++// | . | ++// | | ++// ------------------------------------------------------------------------------------------------------------------------ ++// Scenario 2: Activate all resources, then deactivate all resources ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | ++// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) ++// | | Handler next msg=receive_MIH_Link_Actions_confirm ++// | | _current_link_action_request = _current_link_action_request + 1 ++// |<--------- Link_Actions.confirm ---------------------| (success) ++// | . | ++// | . | ++// | . | _current_link_action_request = 0 ++// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) ++// | | Handler next msg=receive_MIH_Link_Actions_confirm ++// | | _current_link_action_request = _current_link_action_request + 1 ++// |<--------- Link_Actions.confirm ---------------------| (success) ++// | . | ++// | . | ++// | . | ++// | | ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | ++// |---------- Event_Unsubscribe.request --------------->| Handler next msg=receive_MIH_Event_Unsubscribe_confirm ++// |<--------- Event_Subscribe.confirm ------------------| (success) ++// | | ++// | | ++/////////////////////////////////////////////////////////////////////////////// ++ ++static const char* const kConf_MIH_Commands = "user.commands"; ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++namespace po = boost::program_options; ++ ++using odtone::uint; ++using odtone::ushort; ++using odtone::sint8; ++ ++odtone::logger log_("[mih_usr]", std::cout); ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++//----------------------------------------------------------------------------- ++void __trim(odtone::mih::octet_string &str, const char chr) ++//----------------------------------------------------------------------------- ++{ ++ str.erase(std::remove(str.begin(), str.end(), chr), str.end()); ++} ++//----------------------------------------------------------------------------- ++template <class T> std::string StringOf(T object) { ++//----------------------------------------------------------------------------- ++ std::ostringstream os; ++ os << object; ++ return(os.str()); ++} ++//----------------------------------------------------------------------------- ++std::string getTimeStamp4Log() ++//----------------------------------------------------------------------------- ++{ ++ std::stringstream ss (std::stringstream::in | std::stringstream::out); ++ struct timespec time_spec; ++ unsigned int time_now_micros; ++ unsigned int time_now_s; ++ clock_gettime (CLOCK_REALTIME, &time_spec); ++ time_now_s = (unsigned int) time_spec.tv_sec % 3600; ++ time_now_micros = (unsigned int) time_spec.tv_nsec/1000; ++ ss << time_now_s << ':' << time_now_micros; ++ return ss.str(); ++} ++//----------------------------------------------------------------------------- ++std::string status2string(odtone::mih::status statusP){ ++//----------------------------------------------------------------------------- ++ switch (statusP.get()) { ++ case odtone::mih::status_success: return "SUCCESS";break; ++ case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; ++ case odtone::mih::status_rejected: return "REJECTED";break; ++ case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; ++ case odtone::mih::status_network_error: return "NETWORK_ERROR";break; ++ default: return "UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ ++//----------------------------------------------------------------------------- ++ switch (reasonP.get()) { ++ case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; ++ case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; ++ case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; ++ case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; ++ case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; ++ case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; ++ default: return "DN_REASON_UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ ++//----------------------------------------------------------------------------- ++ switch (reasonP.get()) { ++ case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; ++ case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; ++ case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; ++ case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; ++ default: return "GD_REASON_UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string evt2string(odtone::mih::mih_evt_list evtP){ ++//----------------------------------------------------------------------------- ++ std::string s; ++ if(evtP.get(odtone::mih::mih_evt_link_detected)) s = std::string("DETECTED "); ++ if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; ++ if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; ++ if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; ++ if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; ++ if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; ++ if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; ++ if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ ++//----------------------------------------------------------------------------- ++ std::string s; ++ if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s = std::string("Link_Get_Parameters "); ++ if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; ++ if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; ++ if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; ++ if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string link_type2string(const odtone::mih::link_type& lt) ++//----------------------------------------------------------------------------- ++{ ++ switch (lt.get()) { ++ case odtone::mih::link_type_gsm: return "GSM"; break; ++ case odtone::mih::link_type_gprs: return "GPRS"; break; ++ case odtone::mih::link_type_edge: return "EDGE"; break; ++ case odtone::mih::link_type_ethernet: return "Ethernet"; break; ++ case odtone::mih::link_type_wireless_other: return "Other"; break; ++ case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; ++ case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; ++ case odtone::mih::link_type_umts: return "UMTS"; break; ++ case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; ++ case odtone::mih::link_type_lte: return "LTE"; break; ++ case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; ++ case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; ++ case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; ++ default: break; ++ } ++ return "Unknown link type"; ++} ++//----------------------------------------------------------------------------- ++std::string link_addr2string(const odtone::mih::link_addr *addr) ++//----------------------------------------------------------------------------- ++{ ++ if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { ++ return la->address(); ++ } ++ else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { ++ char plmn[16]; ++ sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); ++ return str(boost::format("%s %d") % plmn % la->_cell_id); ++ } ++ else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { ++ char plmn[16]; ++ sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); ++ return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); ++ } ++ else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { ++ return la->value; ++ } ++ else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { ++ return la->value; ++ } ++ else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { ++ return la->value; ++ } ++ return "null"; ++} ++//----------------------------------------------------------------------------- ++std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) ++//----------------------------------------------------------------------------- ++{ ++ char buffer[256]; ++ int index = 0; ++ ++ index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); ++ index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); ++ return buffer; ++} ++//----------------------------------------------------------------------------- ++std::string link_id2string(odtone::mih::link_id linkP) ++//----------------------------------------------------------------------------- ++{ ++ std::string s; ++ s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ switch (ip_addrP.type()) { ++ case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; ++ case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; ++ default: s = "Unkown type "; ++ } ++ s += ip_addrP.address(); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { ++//----------------------------------------------------------------------------- ++ char buffer[128]; ++ std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); ++ return buffer; ++} ++//----------------------------------------------------------------------------- ++std::string ip_proto2string(odtone::mih::proto ip_protoP) { ++//----------------------------------------------------------------------------- ++ switch (ip_protoP.get()) { ++ case odtone::mih::proto_tcp: return "TCP"; ++ case odtone::mih::proto_udp: return "UDP"; ++ default: break; ++ } ++ return "Unknown IP protocol"; ++} ++// TEMP : next 2 functions are commented to restore flow_id as a uint32 ++// full structure will be updated later ++/*//----------------------------------------------------------------------------- ++std::string flow_id2string(odtone::mih::flow_id flowP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ odtone::mih::ip_tuple ip; ++ ip = flowP.src; ++ s = "SRC = " + ip_tuple2string(flowP.src); ++ s += ", DST = " + ip_tuple2string(flowP.dst); ++ s += ", PROTO = " + ip_proto2string(flowP.transport); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { ++//----------------------------------------------------------------------------- ++ if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { ++ return flow_id2string(res->fid); ++ } ++ else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { ++ return flow_id2string(flow->id); ++ } ++ return "null"; ++}*/ ++//----------------------------------------------------------------------------- ++std::string link_ac_result2string(odtone::mih::link_ac_result resultP) ++//----------------------------------------------------------------------------- ++{ ++ switch (resultP.get()) { ++ case odtone::mih::link_ac_success: return "SUCCESS"; break; ++ case odtone::mih::link_ac_failure: return "FAILURE"; break; ++ case odtone::mih::link_ac_refused: return "REFUSED"; break; ++ case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; ++ default: break; ++ } ++ return "Unknown action result"; ++} ++//----------------------------------------------------------------------------- ++std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ ++ s = link_id2string(link_act_reqP.id); ++ ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; ++ ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; ++ ++ s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ odtone::mih::link_id link_id; ++ ++ for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) ++ { ++ link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ link_id.addr = i->addr; ++ if (i != ntalP->begin()) { ++ s += " / "; ++ } ++ s += link_id2string(link_id); ++ } ++ ++ return s; ++} ++ ++/** ++ * Parse supported commands. ++ * ++ * @param cfg Configuration options. ++ * @return An optional list of supported commands. ++ */ ++boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) ++{ ++ using namespace boost; ++ ++ odtone::mih::mih_cmd_list commands; ++ ++ std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; ++ enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; ++ enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; ++ enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; ++ enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; ++ enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; ++ enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; ++ enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; ++ enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; ++ enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; ++ enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; ++ enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; ++ ++ std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); ++ __trim(tmp, ' '); ++ ++ char_separator<char> sep1(","); ++ tokenizer< char_separator<char> > list_tokens(tmp, sep1); ++ ++ BOOST_FOREACH(std::string str, list_tokens) { ++ if(enum_map.find(str) != enum_map.end()) { ++ commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); ++ } ++ } ++ ++ return commands; ++} ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * This class provides an implementation of an IEEE 802.21 MIH-User. ++ */ ++class mih_user : boost::noncopyable { ++public: ++ /** ++ * Construct the MIH-User. ++ * ++ * @param cfg Configuration options. ++ * @param io The io_service object that the MIH-User will use to ++ * dispatch handlers for any asynchronous operations performed on the socket. ++ */ ++ mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); ++ ++ /** ++ * Destruct the MIH-User. ++ */ ++ ~mih_user(); ++ ++protected: ++ /** ++ * User registration handler. ++ * ++ * @param cfg Configuration options. ++ * @param ec Error Code. ++ */ ++ void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); ++ /** ++ * Default MIH event handler. ++ * ++ * @param msg Received event notification. ++ * @param ec Error code. ++ */ ++ void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); ++ /** ++ * MIH receive message handler. ++ * ++ * @param msg Received message. ++ * @param ec Error code. ++ */ ++ void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++ void send_MIH_User_Register_indication(const odtone::mih::config& cfg); ++ ++ void send_MIH_Capability_Discover_request(const odtone::mih::config& cfg); ++ void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); ++ void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Event_Unsubscribe_request(void); ++ void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); ++ void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); ++ void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); ++ void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++ void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); ++ void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++private: ++ odtone::sap::user _mihf; /**< User SAP helper. */ ++ odtone::mih::id _mihfid; /**< MIHF destination ID. */ ++ odtone::mih::id _mihuserid; /**< MIH_USER ID. */ ++ ++ odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ ++ odtone::mih::port _mihf_lport; /**< MIHF local port number */ ++ ++ odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ ++ odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ ++ ++ odtone::mih::link_ac_type _last_link_action_type; ++ odtone::uint _current_link_action_request, _nb_of_link_action_requests; ++ odtone::uint link_threshold_request, link_measures_request, link_measures_counter; ++ odtone::mih::link_id rcv_link_id; ++ ++ static const odtone::uint _max_link_action_requests = 4; ++ ++ void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Up_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Down_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); ++ ++}; ++ ++//----------------------------------------------------------------------------- ++mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) ++ : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), ++ _last_link_action_type(odtone::mih::link_ac_type_none), ++ _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES) ++//----------------------------------------------------------------------------- ++{ ++ ++ odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); ++ _mihuserid.assign(user_id.c_str()); ++ ++ odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); ++ _mihfid.assign(dest_id.c_str()); ++ ++ odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); ++ boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); ++ if (ip.is_v4()) { ++ odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); ++ _mihf_ip = ip_addr; ++ } ++ else if (ip.is_v6()) { ++ odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); ++ _mihf_ip = ip_addr; ++ } ++ ++ _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); ++ ++ //_nb_of_link_action_requests = NB_OF_RESOURCES; ++ if (_nb_of_link_action_requests > _max_link_action_requests) { ++ _nb_of_link_action_requests = _max_link_action_requests; ++ } ++ ++ _link_id_list.clear(); ++ _subs_evt_list.clear(); ++ link_threshold_request = 0; ++ link_measures_request =0; ++ link_measures_counter =0; ++ log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); ++ ++ // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F ++ mih_user::send_MIH_User_Register_indication(cfg); ++} ++ ++//----------------------------------------------------------------------------- ++mih_user::~mih_user() ++//----------------------------------------------------------------------------- ++{ ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH-User register result: ", ec.message(), "\n"); ++ log_(0, ""); ++ ++ // ++ // Local Capability Discover Request ++ // ++ odtone::mih::message msg; ++ _mihfid.assign("mihf2"); ++ msg << odtone::mih::request(odtone::mih::request::capability_discover, _mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Capability_Discover.request --->]["+msg.destination().to_string()+"]\n"); ++ _mihf.async_send(msg, boost::bind(&mih_user::receive_MIH_Capability_Discover_confirm, this, _1)); ++ log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)"); ++ log_(0, ""); ++ ++ //****eNB***** ++ // Remote Capability Discover Request ++ // ++ mih_user::send_MIH_Capability_Discover_request(cfg); ++ ++ //Trigger Link_Configure_Thresholds Request ++ // odtone::mih::message m; ++ //m.destination(msg.source()); ++ //mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ //****eNB***** ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ if (ec) { ++ log_(0, __FUNCTION__, " error: ", ec.message()); ++ return; ++ } ++ ++ switch (msg.mid()) { ++ case odtone::mih::indication::link_detected: ++ mih_user::receive_MIH_Link_Detected_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_up: ++ mih_user::receive_MIH_Link_Up_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_down: ++ mih_user::receive_MIH_Link_Down_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_going_down: ++ mih_user::receive_MIH_Link_Going_Down_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_handover_imminent: ++ log_(0, "MIH-User has received a local event \"link_handover_imminent\""); ++ break; ++ ++ case odtone::mih::indication::link_handover_complete: ++ log_(0, "MIH-User has received a local event \"link_handover_complete\""); ++ break; ++ ++ case odtone::mih::indication::link_parameters_report: ++ //log_(0, "MIH-User has received a local event \"link_parameters_report\""); ++ mih_user::receive_MIH_Link_Parameters_Report(msg, ec); ++ if (link_threshold_request == 0){ ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ link_threshold_request =1; ++ } else if (link_threshold_request == 1){ ++ link_measures_counter ++; ++ // Stop measures after 5 reports ++ if (link_measures_counter == NUM_PARM_REPORT){ ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ } ++ } ++ break; ++ ++ case odtone::mih::indication::link_pdu_transmit_status: ++ log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); ++ break; ++ ++ case odtone::mih::confirm::link_configure_thresholds: ++ mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); ++ break; ++ ++ default: ++ log_(0, "MIH-User has received UNKNOWN local event"); ++ break; ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ if (ec) { ++ log_(0, __FUNCTION__, " error: ", ec.message()); ++ return; ++ } ++ ++ switch (msg.mid()) { ++ ++ case odtone::mih::confirm::capability_discover: ++ mih_user::receive_MIH_Capability_Discover_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::event_subscribe: ++ mih_user::receive_MIH_Event_Subscribe_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::event_unsubscribe: ++ mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::link_actions: ++ mih_user::receive_MIH_Link_Actions_confirm(msg); ++ ++ break; ++ default: ++ log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); ++ break; ++ } ++} ++/* ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_det_info_list ldil; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_det_info_list(ldil); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ // TODO: for each link_det_info in the list {display LINK_DET_INFO} ++ ++ log_(0, "MIH_Link_Detected.indication - End\n"); ++} ++*/ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ ++ log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_det_info_list ldil; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_det_info_list(ldil); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); ++ ++boost::system::error_code ec; ++ for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) ++ { ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(i->id).c_str()); ++ log_(0, " - Network ID: ", i->network_id); ++ log_(0, " - SINR: ", i->sinr); ++ log_(0, " - Data_rate: ", i->data_rate); ++ } ++ ++ send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ ++ ++ log_(0, "MIH_Link_Detected.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++// odtone::mih::tlv_old_access_router oldAR; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link); ++// & odtone::mih::tlv_old_access_router(oar); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); ++ ++ ++ log_(0, "MIH_Link_Up.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::link_addr> addr; ++ odtone::mih::link_dn_reason ldr; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_old_access_router(addr) ++ & odtone::mih::tlv_link_dn_reason(ldr); ++ ++// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); ++ ++ //Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); ++// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Down.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++ odtone::mih::link_gd_reason lgd; ++ odtone::mih::link_ac_ex_time ex_time; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_time_interval(ex_time) ++ & odtone::mih::tlv_link_gd_reason(lgd); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - Time Interval:", (ex_time/256)); ++ log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Going_Down.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id link; ++ odtone::mih::link_param_rpt_list lprl; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_link_param_rpt_list(lprl); ++ ++ log_(0, ""); ++ log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ ++ ++ for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) ++ { ++ if (const odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { ++ ++ log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); ++ //link_action if free channel is available ++ } ++ else{ ++ log_(0, " -Regular Report for BW Threshold "); ++ //update MEAS ++ } ++ } ++ ++ log_(0, "MIH_Link_Parameters_Report.indication - End"); ++} ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); ++ ++ m << odtone::mih::indication(odtone::mih::indication::user_register) ++ & odtone::mih::tlv_command_list(supp_cmd); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); ++ ++ log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Capability_Discover_request(const odtone::mih::config& cfg) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ //boost::optional<odtone::mih::net_type_addr_list> ntal; ++ //boost::optional<odtone::mih::mih_evt_list> evt; ++ boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); ++ ++ m << odtone::mih::request(odtone::mih::request::capability_discover) ++ //& odtone::mih::tlv_net_type_addr_list(ntal) ++ //& odtone::mih::tlv_event_list(evt) ++ & odtone::mih::tlv_command_list(supp_cmd); ++ m.source(_mihuserid); ++ _mihfid.assign("mihf2"); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH_Capability_Discover.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, ""); ++ log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ boost::optional<odtone::mih::net_type_addr_list> ntal; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ boost::optional<odtone::mih::mih_cmd_list> cmd; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_net_type_addr_list(ntal) ++ & odtone::mih::tlv_event_list(evt) ++ & odtone::mih::tlv_command_list(cmd); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ if (cmd) { ++ log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); ++ } ++ if (ntal) { ++ log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); ++ //Store link address ++ for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) ++ { ++ rcv_link_id.addr = i->addr; ++ rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ } ++ } ++ log_(0, ""); ++ ++ // ++ // event subscription ++ // ++ // For every interface the MIHF sent in the ++ // Capability_Discover.response send an Event_Subscribe.request ++ // for all availabe events ++ // ++ if (ntal && evt) { ++ _subs_evt_list = evt.get(); // save the list of subscribed link events ++ for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { ++ if (i->nettype.link.which() == 1) ++ { ++ odtone::mih::link_tuple_id li; ++ ++ li.addr = i->addr; ++ li.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ _link_id_list.push_back(li); // save the link identifier of the network interface ++ ++ mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); ++ } ++ } ++ } ++ ++ log_(0, "MIH_Capability_Discover.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ m << odtone::mih::request(odtone::mih::request::event_subscribe) ++ & odtone::mih::tlv_link_identifier(li) ++ & odtone::mih::tlv_event_list(evt); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ ++ "\\nLink="+link_id2string(li).c_str()+ ++ "\\nEvent list="+evt2string(evt).c_str()+ ++ " --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); ++ ++ log_(0, "MIH_Event_Subscribe.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_event_list(evt); ++ ++ if (evt) { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } else { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ log_(0, ""); ++ ++ //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); ++ log_(0, "TEMP : Resource scenario deactivated\n"); ++ ++ log_(0, "MIH_Event_Subscribe.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Unsubscribe_request(void) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id li; ++ ++ // For every interface the MIH user received in the ++ // Capability_Discover.confirm, send an Event_Unsubscribe.request ++ // for all subscribed events ++ for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { ++ li.type = i->type; ++ li.addr = i->addr; ++ mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ m << odtone::mih::request(odtone::mih::request::event_unsubscribe) ++ & odtone::mih::tlv_link_identifier(li) ++ & odtone::mih::tlv_event_list(evt); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ ++ "\\nLink="+link_id2string(li).c_str()+ ++ "\\nEvent list="+evt2string(evt).c_str()+ ++ " --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); ++ ++ log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_event_list(evt); ++ ++ if (evt) { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } else { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ log_(0, ""); ++ ++ log_(0, "MIH_Event_Unsubscribe.confirm - End"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::link_action_list lal; ++ odtone::mih::link_action_req link_act_req; ++ ++ link_act_req.id = link; ++ link_act_req.action.type = type; ++ ++ _last_link_action_type = type; ++ ++ // Initialize resource parameters ++ odtone::mih::resource_desc res; ++ ++ res.lid = link; // Link identifier ++ res.data_rate = 128000; // bit rate ++ res.jumbo = false; // jumbo disable ++ res.multicast = false; // multicast disable ++ ++ odtone::mih::qos qos; // Class Of Service ++ qos.value = 56; ++ res.qos_val = qos; ++ res.fid = 555 + _current_link_action_request; ++ ++// // Flow identifier ++// res.fid.src.ip = _mihf_ip; ++// res.fid.src.port_val = _mihf_lport; ++// ++// if (mih_user::_current_link_action_request == 0) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY ++// } ++// else if (mih_user::_current_link_action_request == 1) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY ++// } ++// else if (mih_user::_current_link_action_request == 2) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY ++// res.multicast = true; ++// } ++// else if (mih_user::_current_link_action_request == 3) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY ++// } ++// res.fid.dst.port_val = 1235; // DUMMY ++// res.fid.transport = odtone::mih::proto_udp; ++ ++ link_act_req.action.param.param = res; ++ ++ link_act_req.ex_time = 0; ++ ++ lal.push_back(link_act_req); ++ ++ m << odtone::mih::request(odtone::mih::request::link_actions) ++ & odtone::mih::tlv_link_action_list(lal); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - FLOW_ID - Flow identifier: ", res.fid); ++//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); ++ log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); ++ ++ log_(0, "MIH_Link_Actions.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ boost::optional<odtone::mih::link_action_rsp_list> larl; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_action_rsp_list(larl); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ if (larl) { ++ log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); ++ for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) ++ { ++ log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), ++ ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); ++ } ++ } ++ log_(0, ""); ++ ++ // 1st scenario: Sequentially activate and deactivate each resource ++#ifdef SCENARIO_1 ++ if (larl) { ++ odtone::mih::link_action_rsp *rsp = &larl->front(); ++ if (_current_link_action_request < _nb_of_link_action_requests) { ++ if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++ if (rsp->result.get() == odtone::mih::link_ac_success) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++ _current_link_action_request += 1; ++ } ++ } ++ else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); ++ } ++ } ++ else { // Ends the scenario ++ mih_user::send_MIH_Event_Unsubscribe_request(); ++ } ++ } ++#endif // SCENARIO_1 ++ ++#ifdef SCENARIO_2 ++ // 2nd scenario: Activate all resources, then deactivate all resources ++ if (larl.get().size() > 0) { ++ odtone::mih::link_action_rsp *rsp = &larl->front(); ++ if (++_current_link_action_request < _nb_of_link_action_requests) { ++ if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); ++ } ++ else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++ } ++ } ++ else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++ _current_link_action_request = 0; ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++ } ++ else { // Ends the scenario ++ mih_user::send_MIH_Event_Unsubscribe_request(); ++ } ++ } ++#endif // SCENARIO_2 ++ ++ log_(0, "MIH_Link_Actions.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ odtone::mih::threshold th; ++ std::vector<odtone::mih::threshold> thl; ++ ++ odtone::mih::link_tuple_id lti; ++// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; ++ ++ ++ odtone::mih::link_cfg_param_list lcpl; ++ odtone::mih::link_cfg_param lcp; ++ odtone::mih::link_param_lte lp; ++ ++ ++ log_(0,""); ++ log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); ++ ++ ++ odtone::mih::link_det_info_list ldil; ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_det_info_list(ldil); ++ for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) ++ { ++ ++ ++ lti.type = i->id.type; ++ lti.addr = i->id.addr; ++ lp = odtone::mih::link_param_lte_bandwidth; ++ lcp.type = lp; ++ if ( link_measures_request ==0){ ++ lcp.timer_interval = 10; ++ lcp.action = odtone::mih::th_action_normal; ++ link_measures_request = 1; ++ } ++ else{ ++ lcp.timer_interval = 0; ++ lcp.action = odtone::mih::th_action_cancel; ++ link_measures_request = 0; ++ } ++ th.threshold_val = 5; ++ th.threshold_x_dir = odtone::mih::threshold::above_threshold; ++ thl.push_back(th); ++ lcp.threshold_list = thl; ++ lcpl.push_back(lcp); ++ ++ m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) ++ & odtone::mih::tlv_link_identifier(lti) ++ & odtone::mih::tlv_link_cfg_param_list(lcpl); ++ ++ m.destination(msg.source()); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ ++ link_id2string(lti).c_str()+ ++ " --->]["+_mihfid.to_string()+"]\n"); ++ _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); ++ ++ ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); ++ ++ log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); ++ ++ } ++ ++/* ++ if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} ++ ++ log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); ++ ++ if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} ++ if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} ++ if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} ++ ++ log_(0, "\t Threshold value: ", th.threshold_val); ++ ++ if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} ++ if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} ++ ++ log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); ++*/ ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, ""); ++ log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); ++ ++ // T odtone::uint iter; ++ // T odtone::mih::status st; ++ ++ //boost::optional<odtone::mih::link_cfg_status_list> lcsl; ++ // Todtone::mih::link_cfg_status_list lcsl; ++ // Todtone::mih::link_cfg_status lcp; ++ //odtone::mih::link_param_gen lp; ++ ++ // T odtone::mih::link_tuple_id lti; ++ ++ //msg >> odtone::mih::confirm() ++ // & odtone::mih::tlv_status(st) ++ // & odtone::mih::tlv_link_identifier(lti) ++ // & odtone::mih::tlv_link_cfg_status_list(lcsl); ++ ++ ++ log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); ++ log_(0,""); ++} ++ ++//----------------------------------------------------------------------------- ++int main(int argc, char** argv) ++//----------------------------------------------------------------------------- ++{ ++ odtone::setup_crash_handler(); ++ ++ try { ++ boost::asio::io_service ios; ++ ++ // declare MIH Usr available options ++ po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); ++ desc.add_options() ++ ("help", "Display configuration options") ++ (odtone::sap::kConf_File, po::value<std::string>()->default_value("lte_user.conf"), "Configuration file") ++ (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") ++ (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1234), "Listening port") ++ (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") ++ (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") ++ (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") ++ (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") ++ (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1"), "MIHF destination"); ++ ++ odtone::mih::config cfg(desc); ++ cfg.parse(argc, argv, odtone::sap::kConf_File); ++ ++ if (cfg.help()) { ++ std::cerr << desc << std::endl; ++ return EXIT_SUCCESS; ++ } ++ ++ mih_user usr(cfg, ios); ++ ++ ios.run(); ++ ++ } catch(std::exception& e) { ++ log_(0, "exception: ", e.what()); ++ } ++} ++ ++// EOF //////////////////////////////////////////////////////////////////////// ++ +diff --git a/app/lte_test_user/enb_lte_user.conf b/app/lte_test_user/enb_lte_user.conf +new file mode 100644 +index 0000000..b9bbf27 +--- /dev/null ++++ b/app/lte_test_user/enb_lte_user.conf +@@ -0,0 +1,40 @@ ++#=============================================================================== ++# Brief : MIH-User configuration file ++# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> ++# Bruno Santos <bsantos@av.it.pt> ++#------------------------------------------------------------------------------- ++# ODTONE - Open Dot Twenty One ++# ++# Copyright (C) 2009-2012 Universidade Aveiro ++# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++# ++# This software is distributed under a license. The full license ++# agreement can be found in the file LICENSE in this distribution. ++# This software may not be copied, modified, sold or distributed ++# other than expressed in the named license agreement. ++# ++# This software is distributed without any warranty. ++#=============================================================================== ++ ++## ++## User id ++## ++[user] ++id = user_enb ++ ++## ++## Commands supported by the MIH-User ++## ++commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete ++ ++## ++## Port used for communication with MIHF ++## ++[conf] ++port = 1635 ++ ++## ++## MIHF configuration. For the default demonstration leave as is. ++## ++[mihf] ++local_port = 1025 +diff --git a/app/lte_test_user/enb_lte_user.cpp b/app/lte_test_user/enb_lte_user.cpp +new file mode 100644 +index 0000000..80fbfe7 +--- /dev/null ++++ b/app/lte_test_user/enb_lte_user.cpp +@@ -0,0 +1,1695 @@ ++//============================================================================== ++// Brief : MIH-User ++// Authors : Bruno Santos <bsantos@av.it.pt> ++//------------------------------------------------------------------------------ ++// ODTONE - Open Dot Twenty One ++// ++// Copyright (C) 2009-2012 Universidade Aveiro ++// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++// ++// This software is distributed under a license. The full license ++// agreement can be found in the file LICENSE in this distribution. ++// This software may not be copied, modified, sold or distributed ++// other than expressed in the named license agreement. ++// ++// This software is distributed without any warranty. ++//============================================================================== ++ ++#include <odtone/base.hpp> ++#include <odtone/debug.hpp> ++#include <odtone/logger.hpp> ++#include <odtone/mih/request.hpp> ++#include <odtone/mih/response.hpp> ++#include <odtone/mih/indication.hpp> ++#include <odtone/mih/confirm.hpp> ++#include <odtone/mih/tlv_types.hpp> ++#include <odtone/sap/user.hpp> ++ ++#include <boost/utility.hpp> ++#include <boost/bind.hpp> ++#include <boost/tokenizer.hpp> ++#include <boost/foreach.hpp> ++#include <boost/format.hpp> ++ ++#include <iostream> ++#include <string> ++#include <map> ++#include <stdio.h> ++#include <time.h> ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++// Definition of the scenario to execute ++#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests ++//#define SCENARIO_1 // Sequentially activate and deactivate each resource ++#define SCENARIO_2 // Activate all resources, then deactivate all resources ++#define NUM_PARM_REPORT 10 ++ ++//The thresholds for the Energy Detection Sensing algorithm respectively for 10 and 100 samples ++#define ED_THRESHOLD_10 23695432 ++#define ED_THRESHOLD_100 230445932 ++/////////////////////////////////////////////////////////////////////////////// ++// The scenario coded in this MIH-USER (of the eNB) for the demo of the Scenario 2 of SPECTRA project is the following ++// +--------+ +-----+ +---------+ ++// |MIH_USER| |MIH-F| |LINK_SAP | ++// +---+----+ +--+--+ +----+----+ ++// | | | ++// ------------------------------------------------------------------------------------------------------------------------ ++// Initiallization of the MIH-USER and the MIHF ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | | ++// ... (start of MIH-F here) ... ... ++// | |---------- Link_Capability_Discover.request --------X| ++// ... (start of LINK_SAP here) ... ... ++// | |<--------- Link_Register.indication -----------------| ++// | |---------- Link_Capability_Discover.request -------->| ++// | |<--------- Link_Capability_Discover.confirm ---------| ++// | | | ++// ... (start of MIH USER here) ... ... ++// |---------- MIH_User_Register.indication ------------>| (supported_commands) | ++ ++// ------------------------------------------------------------------------------------------------------------------------ ++// Locally send the Capability Discover and the Event Subscribe primitives ++// (from MIH USER to local MIHF) ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | | ++// |---------- MIH_Capability_Discover.request --------->| | ++// |<--------- MIH_Capability_Discover.confirm ----------| | ++// | | | ++// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| ++// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| ++// | | | ++// | | | ++ ++// ------------------------------------------------------------------------------------------------------------------------ ++// Remotely send the Capability Discover and the Event Subscribe primitives ++// (from MIH USER to remote MIHF of the UE) ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | | ++// |---------- MIH_Capability_Discover.request --------->| | ++// |<--------- MIH_Capability_Discover.confirm ----------| | ++// | | | ++// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| ++// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| ++// | | | ++// | | | ++// ------------------------------------------------------------------------------------------------------------------------ ++// Detect the connection of the UE to the eNB + Start the measurement report process ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | | ++// | | | ++// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| ++// | | (RRC Connection reconfiguration notification) | ++// | | | ++// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| ++// | | | ++// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| ++// | | | ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| ++// | (Success) | | ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... | | ++// | | | ++// ------------------------------------------------------------------------------------------------------------------------ ++// Activate the TVWS link (After running the cognitive algorithms) ++// ------------------------------------------------------------------------------------------------------------------------ ++// | | | ++// |-------------- MIH_Link_Actions.request ------------>|--------------- MIH_Link_Actions.request ----------->| ++// | | | ++// |<------------- MIH_Link_Actions.confirm -------------|<-------------- MIH_Link_Actions.confirm ------------| ++// | | | ++ ++//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ++ ++ ++static const char* const kConf_MIH_Commands = "user.commands"; ++ ++///////////////////////////////////////////////////////////////////////////////////////// ++ ++std::string exec(char* cmd) { ++ FILE* pipe = popen(cmd, "r"); ++ if (!pipe) return "ERROR"; ++ char buffer[128]; ++ std::string result = ""; ++ while(!feof(pipe)) { ++ if(fgets(buffer, 128, pipe) != NULL) ++ result += buffer; ++ } ++ pclose(pipe); ++ return result; ++} ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++namespace po = boost::program_options; ++ ++using odtone::uint; ++using odtone::ushort; ++using odtone::sint8; ++ ++odtone::logger log_("[mih_usr]", std::cout); ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++//----------------------------------------------------------------------------- ++void __trim(odtone::mih::octet_string &str, const char chr) ++//----------------------------------------------------------------------------- ++{ ++ str.erase(std::remove(str.begin(), str.end(), chr), str.end()); ++} ++//----------------------------------------------------------------------------- ++template <class T> std::string StringOf(T object) { ++//----------------------------------------------------------------------------- ++ std::ostringstream os; ++ os << object; ++ return(os.str()); ++} ++//----------------------------------------------------------------------------- ++std::string getTimeStamp4Log() ++//----------------------------------------------------------------------------- ++{ ++ std::stringstream ss (std::stringstream::in | std::stringstream::out); ++ struct timespec time_spec; ++ unsigned int time_now_micros; ++ unsigned int time_now_s; ++ clock_gettime (CLOCK_REALTIME, &time_spec); ++ time_now_s = (unsigned int) time_spec.tv_sec % 3600; ++ time_now_micros = (unsigned int) time_spec.tv_nsec/1000; ++ ss << time_now_s << ':' << time_now_micros; ++ return ss.str(); ++} ++//----------------------------------------------------------------------------- ++std::string status2string(odtone::mih::status statusP){ ++//----------------------------------------------------------------------------- ++ switch (statusP.get()) { ++ case odtone::mih::status_success: return "SUCCESS";break; ++ case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; ++ case odtone::mih::status_rejected: return "REJECTED";break; ++ case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; ++ case odtone::mih::status_network_error: return "NETWORK_ERROR";break; ++ default: return "UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ ++//----------------------------------------------------------------------------- ++ switch (reasonP.get()) { ++ case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; ++ case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; ++ case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; ++ case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; ++ case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; ++ case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; ++ default: return "DN_REASON_UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ ++//----------------------------------------------------------------------------- ++ switch (reasonP.get()) { ++ case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; ++ case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; ++ case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; ++ case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; ++ default: return "GD_REASON_UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string evt2string(odtone::mih::mih_evt_list evtP){ ++//----------------------------------------------------------------------------- ++ std::string s=std::string(" "); ++ if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; ++ if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; ++ if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; ++ if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; ++ if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; ++ if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; ++ if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; ++ if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ ++//----------------------------------------------------------------------------- ++ std::string s=std::string(" "); ++ if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; ++ if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; ++ if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; ++ if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; ++ if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string link_type2string(const odtone::mih::link_type& lt) ++//----------------------------------------------------------------------------- ++{ ++ switch (lt.get()) { ++ case odtone::mih::link_type_gsm: return "GSM"; break; ++ case odtone::mih::link_type_gprs: return "GPRS"; break; ++ case odtone::mih::link_type_edge: return "EDGE"; break; ++ case odtone::mih::link_type_ethernet: return "Ethernet"; break; ++ case odtone::mih::link_type_wireless_other: return "Other"; break; ++ case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; ++ case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; ++ case odtone::mih::link_type_umts: return "UMTS"; break; ++ case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; ++ case odtone::mih::link_type_lte: return "LTE"; break; ++ case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; ++ case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; ++ case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; ++ default: break; ++ } ++ return "Unknown link type"; ++} ++//----------------------------------------------------------------------------- ++std::string link_addr2string(const odtone::mih::link_addr *addr) ++//----------------------------------------------------------------------------- ++{ ++ if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { ++ return la->address(); ++ } ++ else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { ++ char plmn[16]; ++ sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); ++ return str(boost::format("%s %d") % plmn % la->_cell_id); ++ } ++ else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { ++ char plmn[16]; ++ sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); ++ return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); ++ } ++ else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { ++ return la->value; ++ } ++ else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { ++ return la->value; ++ } ++ else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { ++ return la->value; ++ } ++ return "null"; ++} ++//----------------------------------------------------------------------------- ++std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) ++//----------------------------------------------------------------------------- ++{ ++ char buffer[256]; ++ int index = 0; ++ ++ index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); ++ index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); ++ return buffer; ++} ++//----------------------------------------------------------------------------- ++std::string link_id2string(odtone::mih::link_id linkP) ++//----------------------------------------------------------------------------- ++{ ++ std::string s; ++ s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ switch (ip_addrP.type()) { ++ case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; ++ case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; ++ default: s = "Unkown type "; ++ } ++ s += ip_addrP.address(); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { ++//----------------------------------------------------------------------------- ++ char buffer[128]; ++ std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); ++ return buffer; ++} ++//----------------------------------------------------------------------------- ++std::string ip_proto2string(odtone::mih::proto ip_protoP) { ++//----------------------------------------------------------------------------- ++ switch (ip_protoP.get()) { ++ case odtone::mih::proto_tcp: return "TCP"; ++ case odtone::mih::proto_udp: return "UDP"; ++ default: break; ++ } ++ return "Unknown IP protocol"; ++} ++// TEMP : next 2 functions are commented to restore flow_id as a uint32 ++// full structure will be updated later ++/*//----------------------------------------------------------------------------- ++std::string flow_id2string(odtone::mih::flow_id flowP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ odtone::mih::ip_tuple ip; ++ ip = flowP.src; ++ s = "SRC = " + ip_tuple2string(flowP.src); ++ s += ", DST = " + ip_tuple2string(flowP.dst); ++ s += ", PROTO = " + ip_proto2string(flowP.transport); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { ++//----------------------------------------------------------------------------- ++ if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { ++ return flow_id2string(res->fid); ++ } ++ else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { ++ return flow_id2string(flow->id); ++ } ++ return "null"; ++}*/ ++//----------------------------------------------------------------------------- ++std::string link_ac_result2string(odtone::mih::link_ac_result resultP) ++//----------------------------------------------------------------------------- ++{ ++ switch (resultP.get()) { ++ case odtone::mih::link_ac_success: return "SUCCESS"; break; ++ case odtone::mih::link_ac_failure: return "FAILURE"; break; ++ case odtone::mih::link_ac_refused: return "REFUSED"; break; ++ case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; ++ default: break; ++ } ++ return "Unknown action result"; ++} ++//----------------------------------------------------------------------------- ++std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ ++ s = link_id2string(link_act_reqP.id); ++ ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; ++ ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; ++ ++ s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; ++ return s; ++} ++ ++ ++/*//----------------------------------------------------------------------------- ++std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ odtone::mih::link_id link_id; ++ ++ for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) ++ { ++ link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ link_id.addr = i->addr; ++ if (i != ntalP->begin()) { ++ s += " / "; ++ } ++ s += link_id2string(link_id); ++ } ++ ++ return s; ++} ++*/ ++ ++//Updated from UE code ++//----------------------------------------------------------------------------- ++std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ std::ostringstream stream; ++ odtone::mih::net_type_addr net_type_addr; ++ ++ for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) ++ { ++ net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); ++ stream << net_type_addr; ++ if (i != ntalP->begin()) { ++ stream << " / "; ++ } ++ } ++ s = stream.str(); ++ return s; ++} ++ ++/** ++ * Parse supported commands. ++ * ++ * @param cfg Configuration options. ++ * @return An optional list of supported commands. ++ */ ++boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) ++{ ++ using namespace boost; ++ ++ odtone::mih::mih_cmd_list commands; ++ ++ std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; ++ enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; ++ enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; ++ enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; ++ enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; ++ enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; ++ enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; ++ enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; ++ enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; ++ enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; ++ enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; ++ enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; ++ ++ std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); ++ __trim(tmp, ' '); ++ ++ char_separator<char> sep1(","); ++ tokenizer< char_separator<char> > list_tokens(tmp, sep1); ++ ++ BOOST_FOREACH(std::string str, list_tokens) { ++ if(enum_map.find(str) != enum_map.end()) { ++ commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); ++ } ++ } ++ ++ return commands; ++} ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * This class provides an implementation of an IEEE 802.21 MIH-User. ++ */ ++class mih_user : boost::noncopyable { ++public: ++ /** ++ * Construct the MIH-User. ++ * ++ * @param cfg Configuration options. ++ * @param io The io_service object that the MIH-User will use to ++ * dispatch handlers for any asynchronous operations performed on the socket. ++ */ ++ mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); ++ ++ /** ++ * Destruct the MIH-User. ++ */ ++ ~mih_user(); ++ ++protected: ++ /** ++ * User registration handler. ++ * ++ * @param cfg Configuration options. ++ * @param ec Error Code. ++ */ ++ void user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec); ++ /** ++ * Default MIH event handler. ++ * ++ * @param msg Received event notification. ++ * @param ec Error code. ++ */ ++ void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); ++ /** ++ * MIH receive message handler. ++ * ++ * @param msg Received message. ++ * @param ec Error code. ++ */ ++ void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++ void send_MIH_User_Register_indication(const odtone::mih::config& cfg); ++ ++ void send_MIH_Capability_Discover_request(void); ++ void send_MIH_Capability_Discover_request_remote(void); ++ void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); ++ void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Event_Unsubscribe_request(void); ++ void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); ++ void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); ++ void send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link); ++ void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); ++ ++ //Measurements report methods ++ void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); ++ int receive_Sensing_Report(); ++ void receive_CRRM_Data(); ++ ++ void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); ++ void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); ++ void forward_Parameters_Report_indication(odtone::mih::message& m); ++ ++private: ++ odtone::sap::user _mihf; /**< User SAP helper. */ ++ odtone::mih::id _mihfid; /**< MIHF destination ID. */ ++ odtone::mih::id _mihuserid; /**< MIH_USER ID. */ ++ ++ odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ ++ odtone::mih::port _mihf_lport; /**< MIHF local port number */ ++ ++ odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ ++ odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ ++ ++ odtone::mih::link_ac_type _last_link_action_type; ++ odtone::uint _current_link_action_request, _nb_of_link_action_requests; ++ odtone::uint link_threshold_request, link_measures_request, link_measures_counter; ++ odtone::mih::link_id rcv_link_id; ++ ++ static const odtone::uint _max_link_action_requests = 4; ++ odtone::uint _num_thresholds_request; ++ odtone::uint second_link_activated; ++ odtone::uint count; ++ odtone::uint sensing_done; ++ odtone::uint sensing_score; ++ void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Up_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Down_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); ++ ++}; ++ ++//----------------------------------------------------------------------------- ++mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) ++ : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), ++ _last_link_action_type(odtone::mih::link_ac_type_none), ++ _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); ++ _mihuserid.assign(user_id.c_str()); ++ ++ odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); ++ _mihfid.assign(dest_id.c_str()); ++ ++ odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); ++ boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); ++ if (ip.is_v4()) { ++ odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); ++ _mihf_ip = ip_addr; ++ } ++ else if (ip.is_v6()) { ++ odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); ++ _mihf_ip = ip_addr; ++ } ++ ++ _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); ++ ++ //_nb_of_link_action_requests = NB_OF_RESOURCES; ++ if (_nb_of_link_action_requests > _max_link_action_requests) { ++ _nb_of_link_action_requests = _max_link_action_requests; ++ } ++ ++ _link_id_list.clear(); ++ _subs_evt_list.clear(); ++ link_threshold_request = 0; ++ second_link_activated = 0; ++ count = 0; ++ sensing_done = 0; ++ sensing_score = 0; ++ link_measures_request =0; ++ link_measures_counter =0; ++ log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); ++ ++ // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F ++ mih_user::send_MIH_User_Register_indication(cfg); ++} ++ ++//----------------------------------------------------------------------------- ++mih_user::~mih_user() ++//----------------------------------------------------------------------------- ++{ ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH-User register result: ", ec.message(), "\n"); ++ ++ // ++ // Let's fire a capability discover request to get things moving ++ // ++ mih_user::send_MIH_Capability_Discover_request(); ++ ++ //send a capability discover request to the remote UE ++ mih_user::send_MIH_Capability_Discover_request_remote(); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ if (ec) { ++ log_(0, __FUNCTION__, " error: ", ec.message()); ++ return; ++ } ++ ++ switch (msg.mid()) { ++ case odtone::mih::indication::link_detected: ++ mih_user::receive_MIH_Link_Detected_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_up: ++ mih_user::receive_MIH_Link_Up_indication(msg); ++ if (_num_thresholds_request == 0) { ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ _num_thresholds_request += 1; ++ } ++ break; ++ ++ case odtone::mih::indication::link_down: ++ mih_user::receive_MIH_Link_Down_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_going_down: ++ mih_user::receive_MIH_Link_Going_Down_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_handover_imminent: ++ log_(0, "MIH-User has received a local event \"link_handover_imminent\""); ++ break; ++ ++ case odtone::mih::indication::link_handover_complete: ++ log_(0, "MIH-User has received a local event \"link_handover_complete\""); ++ break; ++ ++ case odtone::mih::indication::link_parameters_report: ++ //log_(0, "MIH-User has received a local event \"link_parameters_report\""); ++ mih_user::receive_MIH_Link_Parameters_Report(msg, ec); ++ /*if (link_threshold_request == 0){ ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ link_threshold_request =1; ++ } else if (link_threshold_request == 1){ ++ link_measures_counter ++; ++ // Stop measures after 5 reports ++ if (link_measures_counter == NUM_PARM_REPORT){ ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ } ++ }*/ ++ break; ++ ++ case odtone::mih::indication::link_pdu_transmit_status: ++ log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); ++ break; ++ ++ case odtone::mih::confirm::link_configure_thresholds: ++ mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); ++ break; ++ ++ default: ++ log_(0, "MIH-User has received UNKNOWN local event"); ++ break; ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ if (ec) { ++ log_(0, __FUNCTION__, " error: ", ec.message()); ++ return; ++ } ++ ++ switch (msg.mid()) { ++ ++ case odtone::mih::confirm::capability_discover: ++ mih_user::receive_MIH_Capability_Discover_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::event_subscribe: ++ mih_user::receive_MIH_Event_Subscribe_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::event_unsubscribe: ++ mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::link_actions: ++ mih_user::receive_MIH_Link_Actions_confirm(msg); ++ break; ++ ++ default: ++ log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); ++ break; ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); ++ odtone::mih::link_det_info ldi; ++ odtone::mih::link_det_info_list ldil; ++ odtone::mih::link_det_info_list::iterator it_ldil; ++ odtone::mih::link_id lid; ++ boost::system::error_code ec; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_det_info_list(ldil); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); ++ for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { ++ ldi = *it_ldil; ++ log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); ++ log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); ++ log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); ++ log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); ++ log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); ++ log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); ++ log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); ++ log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); ++ ++ } ++ ++ // Display message parameters ++ // TODO: for each link_det_info in the list {display LINK_DET_INFO} ++// mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ ++ log_(0, "MIH_Link_Detected.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++// odtone::mih::tlv_old_access_router oldAR; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link); ++// & odtone::mih::tlv_old_access_router(oar); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Up.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::link_addr> addr; ++ odtone::mih::link_dn_reason ldr; ++ ++ log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_old_access_router(addr) ++ & odtone::mih::tlv_link_dn_reason(ldr); ++ ++// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); ++ ++ //Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); ++// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Down.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++ odtone::mih::link_gd_reason lgd; ++ odtone::mih::link_ac_ex_time ex_time; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_time_interval(ex_time) ++ & odtone::mih::tlv_link_gd_reason(lgd); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - Time Interval:", (ex_time/256)); ++ log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Going_Down.indication - End\n"); ++} ++ ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); ++ ++ m << odtone::mih::indication(odtone::mih::indication::user_register) ++ & odtone::mih::tlv_command_list(supp_cmd); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, /*boost::cref(cfg),*/ _2)); ++ ++ log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Capability_Discover_request(void) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ m << odtone::mih::request(odtone::mih::request::capability_discover); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Capability_Discover_request_remote(void) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ m << odtone::mih::request(odtone::mih::request::capability_discover); ++ m.source(_mihuserid); ++ odtone::mih::id mid_ue; ++ mid_ue.assign("mihf2_ue"); ++ m.destination(mid_ue); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ boost::optional<odtone::mih::net_type_addr_list> ntal; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ boost::optional<odtone::mih::mih_cmd_list> cmd; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_net_type_addr_list(ntal) ++ & odtone::mih::tlv_event_list(evt) ++ & odtone::mih::tlv_command_list(cmd); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ if (cmd) { ++ log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); ++ } ++ if (ntal) { ++ log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); ++ //Store link address ++ //for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) ++ //{ ++ // rcv_link_id.addr = i->addr; ++ // rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ //} ++ } ++ log_(0, ""); ++ ++ // ++ // event subscription ++ // ++ // For every interface the MIHF sent in the ++ // Capability_Discover.response send an Event_Subscribe.request ++ // for all availabe events ++ // ++ ++ if (ntal && evt) { ++ _subs_evt_list = evt.get(); // save the list of subscribed link events ++ ++ std::cout<<"NTALL "<<ntal.get()[0].addr<<std::endl; ++ for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { ++ if (i->nettype.link.which() == 1) ++ { ++ odtone::mih::link_tuple_id li; ++// std::ostringstream stream; ++// std::stringstream st; ++// odtone::mih::net_type_addr net_type_addr; ++// net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); ++// stream << net_type_addr; ++// std::string s; ++// std::stringstream st (stream.str()); ++// while (getline (st, s, '\n')) ++// { ++// std::stringstream ss(s); ++// getline (ss, name, ':'); ++// getline (ss, value, '\n'); ++// // ss>>name>>value; ++// std::cout<<" name "<<name<<" value "<<value<<std::endl; ++// } ++/* ++ st<<i->addr; ++ getline (st, s, ' '); ++ std::cout<<"LINK_TUPLE_ID s "<<s<<std::endl; ++ ++ odtone::mih::l2_3gpp_addr add; ++ add.value = s;*/ ++ li.addr = /*add*/ /*boost::get<odtone::mih::l2_3gpp_addr>(*/i->addr/*)*/; ++ li.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ if (std::find(_link_id_list.begin(), _link_id_list.end(), li)==_link_id_list.end()) ++ _link_id_list.push_back(li); // save the link identifier of the network interface ++ ++ std::cout<<"LINK_TUPLE_ID - Link identifier:after "<<link_addr2string(&li.addr).c_str() <<" "<<_link_id_list[_link_id_list.size()-1]<<std::endl; ++ mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); ++ } ++ } ++ } ++ ++ log_(0, "MIH_Capability_Discover.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ m << odtone::mih::request(odtone::mih::request::event_subscribe) ++ & odtone::mih::tlv_link_identifier(li) ++ & odtone::mih::tlv_event_list(evt); ++ m.source(_mihuserid); ++ m.destination(dest); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ ++ "\\nLink="+link_id2string(li).c_str()+ ++ "\\nEvent list="+evt2string(evt).c_str()+ ++ " --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); ++ std::cout<<"LINK_TUPLE_ID - Link identifier: "<<li<<std::endl; ++// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); ++ ++ log_(0, "MIH_Event_Subscribe.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_event_list(evt); ++ ++ if (evt) { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } else { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ log_(0, ""); ++ ++ //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); ++ //log_(0, "TEMP : Resource scenario deactivated\n"); ++ ++ log_(0, "MIH_Event_Subscribe.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Unsubscribe_request(void) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id li; ++ ++ // For every interface the MIH user received in the ++ // Capability_Discover.confirm, send an Event_Unsubscribe.request ++ // for all subscribed events ++ for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { ++ li.type = i->type; ++ li.addr = i->addr; ++ mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ m << odtone::mih::request(odtone::mih::request::event_unsubscribe) ++ & odtone::mih::tlv_link_identifier(li) ++ & odtone::mih::tlv_event_list(evt); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ ++ "\\nLink="+link_id2string(li).c_str()+ ++ "\\nEvent list="+evt2string(evt).c_str()+ ++ " --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); ++ ++ log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_event_list(evt); ++ ++ if (evt) { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } else { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ log_(0, ""); ++ ++ log_(0, "MIH_Event_Unsubscribe.confirm - End"); ++} ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::link_action_list lal; ++ odtone::mih::link_action_req link_act_req; ++ //struct null n; ++ ++// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { ++// std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; ++ link_act_req.id = link; ++// } ++ ++ link_act_req.action.type = odtone::mih::link_ac_type_power_up; ++ link_act_req.action.attr.clear(); ++ link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); ++ ++ link_act_req.ex_time = 5000; // in ms ++ ++ lal.push_back(link_act_req); ++ std::cout<<"_link received from parameters report "<<link<<std::endl; ++ for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { ++ std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; ++ } ++ ++ m << odtone::mih::request(odtone::mih::request::link_actions) ++ & odtone::mih::tlv_link_action_list(lal); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); ++ ++ log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::link_action_list lal; ++ odtone::mih::link_action_req link_act_req; ++ ++ link_act_req.id = link; ++ link_act_req.action.type = type; ++ ++ _last_link_action_type = type; ++ ++ // Initialize resource parameters ++ odtone::mih::resource_desc res; ++ ++ res.lid = link; // Link identifier ++ res.data_rate = 128000; // bit rate ++ res.jumbo = false; // jumbo disable ++ res.multicast = false; // multicast disable ++ ++ odtone::mih::qos qos; // Class Of Service ++ qos.value = 56; ++ res.qos_val = qos; ++ res.fid = 555 + _current_link_action_request; ++ ++// // Flow identifier ++// res.fid.src.ip = _mihf_ip; ++// res.fid.src.port_val = _mihf_lport; ++// ++// if (mih_user::_current_link_action_request == 0) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY ++// } ++// else if (mih_user::_current_link_action_request == 1) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY ++// } ++// else if (mih_user::_current_link_action_request == 2) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY ++// res.multicast = true; ++// } ++// else if (mih_user::_current_link_action_request == 3) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY ++// } ++// res.fid.dst.port_val = 1235; // DUMMY ++// res.fid.transport = odtone::mih::proto_udp; ++ ++// link_act_req.action.param.param = res; ++ ++// link_act_req.ex_time = 0; ++ link_act_req.ex_time = 5000; ++ ++ lal.push_back(link_act_req); ++ ++ m << odtone::mih::request(odtone::mih::request::link_actions) ++ & odtone::mih::tlv_link_action_list(lal); ++ m.source(_mihuserid); ++ ++// odtone::mih::id mid_ue; ++// mid_ue.assign("mihf2_ue"); ++ m.destination(/*mid_ue*/_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - FLOW_ID - Flow identifier: ", res.fid); ++//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); ++ log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); ++ ++ log_(0, "MIH_Link_Actions.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ boost::optional<odtone::mih::link_action_rsp_list> larl; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st); ++// & odtone::mih::tlv_link_action_rsp_list(larl); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++// if (larl) { ++// log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); ++// for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) ++// { ++// log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), ++// ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); ++// } ++// } ++ log_(0, ""); ++ ++ // 1st scenario: Sequentially activate and deactivate each resource ++// #ifdef SCENARIO_1 ++// if (larl) { ++// odtone::mih::link_action_rsp *rsp = &larl->front(); ++// if (_current_link_action_request < _nb_of_link_action_requests) { ++// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++// if (rsp->result.get() == odtone::mih::link_ac_success) { ++// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++// _current_link_action_request += 1; ++// } ++// } ++// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { ++// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); ++// } ++// } ++// else { // Ends the scenario ++// mih_user::send_MIH_Event_Unsubscribe_request(); ++// } ++// } ++// #endif // SCENARIO_1 ++// ++// #ifdef SCENARIO_2 ++// // 2nd scenario: Activate all resources, then deactivate all resources ++// if (larl.get().size() > 0) { ++// odtone::mih::link_action_rsp *rsp = &larl->front(); ++// if (++_current_link_action_request < _nb_of_link_action_requests) { ++// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); ++// } ++// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { ++// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++// } ++// } ++// else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++// _current_link_action_request = 0; ++// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++// } ++// else { // Ends the scenario ++// mih_user::send_MIH_Event_Unsubscribe_request(); ++// } ++// } ++// #endif // SCENARIO_2 ++ ++ log_(0, "MIH_Link_Actions.confirm - End\n"); ++} ++ ++void mih_user::receive_CRRM_Data() ++{ ++ ++ //CRRM data report ++ std::string result = exec ("./CRMClientmain 0 1"); ++} ++ ++//----------------------------------------------------------------------------- ++int mih_user::receive_Sensing_Report() ++//----------------------------------------------------------------------------- ++{ ++ std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); ++ std::string tmp; ++ int value; ++ std::stringstream ss(result); ++ ss >> tmp >> value; ++ std::cout<<"Result of Sensing "<<value<<std::endl; ++ sensing_done = 1; ++ return value; ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id link, link1; ++ odtone::mih::link_param_rpt_list lprl; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_link_param_rpt_list(lprl); ++ ++ log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); ++ log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ std::cout<<"LINK_TUPLE_ID - Link identifier: "<<link<<std::endl; ++ ++ for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) ++ { ++ log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); ++ if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) ++ { ++ log_(0, "Meausrement Value: ", (short) *value ); ++ } ++ } ++ log_(0, "MIH_Link_Parameters_Report.indication - End"); ++ ++ ++ //eNB1: Forward the message to UE 2 ++ forward_Parameters_Report_indication(msg); ++ ++ //eNB2 : Action Power Up the TVWS Link after running the cognitive algorithm ++ ++ //First Phase: Collect the data as input fot the cognitive algorithm ++ //Sensing data ++ if (sensing_done == 0) ++ sensing_score = receive_Sensing_Report(); ++ ++ //CRRM Data ++ receive_CRRM_Data(); ++ ++ for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) ++ { ++ if (odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { ++// log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); ++ //link_action if free channel is available ++// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { ++// if(*i==link) ++ odtone::mih::l2_3gpp_addr add; ++ odtone::mih::link_type type; ++ type = odtone::mih::link_type_lte; ++// add.value = "l2_3gpp_addr"; ++ link1.addr = add; ++ link1.type = type; ++// send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_power_up); ++ ++ //Send ONLY ONCE RRC_Connection_Reconfiguration request to the UE on LTE link 0 ++ if (second_link_activated==0) ++ { ++ if (count == 20) ++ { ++ send_MIH_Link_Action_Power_Up_request(_link_id_list[1]);//Link LTE ++ second_link_activated = 1; ++ } ++ } ++ ++ count++; ++// continue; ++// send_MIH_Link_Action_Power_Up_request(link); ++// if (link_id2string(link).c_str() == "LTE") ++// } ++ } ++ } ++ ++} ++ ++ ++//Forward the MIH_Link_Parameters_Report to the UE of the CPE ++//----------------------------------------------------------------------------- ++void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id link; ++ odtone::mih::link_param_rpt_list lprl; ++ ++ m >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_link_param_rpt_list(lprl); ++ ++ odtone::mih::message msg; ++ msg.source(_mihuserid); ++ odtone::mih::id mid_ue; ++ mid_ue.assign("mihf3_ue"); ++ msg.destination(mid_ue); ++ ++ //mn_ho_candidate_query is used to constuct/send the message containing parameters reports ++ msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_link_param_rpt_list(lprl); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- before MIH_User_PARAMETERS.indication --->]["+msg.destination().to_string()+"]\n"); ++ ++ ++ _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); ++ ++ log_(0, "MIH_User_PARAMETERS.indication- SENT (towards UE)\n"); ++ ++} ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::threshold th1; ++ odtone::mih::threshold th2; ++ std::vector<odtone::mih::threshold> thl1; ++ std::vector<odtone::mih::threshold> thl2; ++ odtone::mih::link_tuple_id lti; ++ odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; ++ //List of the link threshold parameters ++ odtone::mih::link_cfg_param_list lcpl; ++ odtone::mih::link_cfg_param lcp1; ++ odtone::mih::link_cfg_param lcp2; ++ odtone::mih::link_param_lte lp1; ++ odtone::mih::link_param_lte lp2; ++ //odtone::mih::link_param_gen lp; ++ ++ odtone::mih::link_param_type typr; ++ ++ log_(0,""); ++ log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); ++ ++ //link_tuple_id ++ lti.type = rcv_link_id.type; ++ lti.addr = rcv_link_id.addr; ++ ++ //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); ++ ++ //link_param_gen_data_rate = 0, /**< Data rate. */ ++ //link_param_gen_signal_strength = 1, /**< Signal strength. */ ++ //link_param_gen_sinr = 2, /**< SINR. */ ++ //link_param_gen_throughput = 3, /**< Throughput. */ ++ //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ ++ //lp = odtone::mih::link_param_lte_bandwidth; ++ lp1 = odtone::mih::link_param_lte_rsrp; ++ lp2 = odtone::mih::link_param_lte_rsrq; ++ lcp1.type = lp1; ++ lcp2.type = lp2; ++ ++ link_measures_request = 0; ++ if (link_measures_request ==0){ ++ // Set Timer Interval (in ms) ++ lcp1.timer_interval = 3000; ++ lcp2.timer_interval = 3000; ++ //th_action_normal = 0, /**< Set normal threshold. */ ++ //th_action_one_shot = 1, /**< Set one-shot threshold. */ ++ //th_action_cancel = 2 /**< Cancel threshold. */ ++ lcp1.action = odtone::mih::th_action_normal; ++ lcp2.action = odtone::mih::th_action_normal; ++ link_measures_request = 1; ++ } else if ( link_measures_request==1){ ++ // Set Timer Interval (in ms) ++ lcp1.timer_interval = 0; ++ lcp2.timer_interval = 0; ++ lcp1.action = odtone::mih::th_action_cancel; ++ lcp2.action = odtone::mih::th_action_cancel; ++ link_measures_request = 0; ++ } ++ ++ //above_threshold = 0, /**< Above threshold. */ ++ //below_threshold = 1, /**< Below threshold. */ ++ th1.threshold_val = -105; ++ th2.threshold_val = -19; ++ th1.threshold_x_dir = odtone::mih::threshold::above_threshold; ++ th2.threshold_x_dir = odtone::mih::threshold::above_threshold; ++ ++ thl1.push_back(th1); ++ thl2.push_back(th2); ++ lcp1.threshold_list = thl1; ++ lcp2.threshold_list = thl2; ++ ++ lcpl.push_back(lcp1); ++ lcpl.push_back(lcp2); ++ ++ ++ ++ m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) ++ & odtone::mih::tlv_link_identifier(lti) ++ & odtone::mih::tlv_link_cfg_param_list(lcpl); ++ ++ m.source(_mihuserid); ++ odtone::mih::id mid_ue; ++ mid_ue.assign("mihf2_ue"); ++ m.destination(mid_ue); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ ++ // link_tupple_id2string(lti).c_str() + ++ link_id2string(lti).c_str()+ " --->][ ADDR LINK " ++ + m.destination().to_string() +"]\n"); ++ std::cout<<"LINK TUPLE ID "<<lti<<std::endl; ++ _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); ++ ++ ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); ++ ++ log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); ++ ++ //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} ++ //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} ++ //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} ++ //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} ++ //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} ++ if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} ++ ++ log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); ++ ++ if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} ++ if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} ++ if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} ++ ++ log_(0, "\t Threshold value: ", (short) th1.threshold_val); ++ ++ if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} ++ if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} ++ ++ log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); ++} ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, ""); ++ log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); ++ ++ odtone::uint iter; ++ odtone::mih::status st; ++ ++ //boost::optional<odtone::mih::link_cfg_status_list> lcsl; ++ odtone::mih::link_cfg_status_list lcsl; ++ odtone::mih::link_cfg_status lcp; ++ odtone::mih::link_param_gen lp; ++ ++ odtone::mih::link_tuple_id lti; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(lti); ++// & odtone::mih::tlv_link_cfg_status_list(lcsl); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); ++ log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); ++ ++ log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); ++ ++ for(iter=0; iter<lcsl.size(); iter++) ++ { ++ log_(0, "\t Link Param Type: ", lcsl[iter].type); ++ log_(0, "\t Threshold Val: ", (lcsl[iter].thold.threshold_val/256)); ++ if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} ++ if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} ++ if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} ++ else {log_(0, "\t Config Status: ", lcsl[iter].status);} ++ } ++ ++ log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); ++ log_(0,""); ++} ++ ++ ++//----------------------------------------------------------------------------- ++int main(int argc, char** argv) ++//----------------------------------------------------------------------------- ++{ ++ odtone::setup_crash_handler(); ++ ++ try { ++ boost::asio::io_service ios; ++ ++ // declare MIH Usr available options ++ po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); ++ desc.add_options() ++ ("help", "Display configuration options") ++ (odtone::sap::kConf_File, po::value<std::string>()->default_value("enb_lte_user.conf"), "Configuration file") ++ (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") ++ (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") ++ (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") ++ (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") ++ (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") ++ (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") ++ (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1_enb"), "MIHF destination"); ++ ++ odtone::mih::config cfg(desc); ++ cfg.parse(argc, argv, odtone::sap::kConf_File); ++ ++ if (cfg.help()) { ++ std::cerr << desc << std::endl; ++ return EXIT_SUCCESS; ++ } ++ ++ mih_user usr(cfg, ios); ++ ++ ios.run(); ++ ++ } catch(std::exception& e) { ++ log_(0, "exception: ", e.what()); ++ } ++} ++ ++// EOF //////////////////////////////////////////////////////////////////////// ++ +diff --git a/app/lte_test_user/libcrmclient.so.1.0 b/app/lte_test_user/libcrmclient.so.1.0 +new file mode 100644 +index 0000000000000000000000000000000000000000..c3c182c85c84468692552a29a6755784abc8971f +GIT binary patch +literal 2273132 +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 +zAB~5wK8VlB_x~°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 +z&ioiuZtGRbdg_q=Wpu38m%0A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 +zv(WBDn~S#Lb9gTh+V*G@	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 +z7I=RmJOm&JAuu2e5r{$z;*bCnl8^!myeYy%0D=$#1Hur2D8wKR2{0iEDX_r%GvOfs +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 +z4V(VVtW%|I0Hzd|f28}C4nq	Y0OrPPoHo=*B{Mq|m<?lm)Lzr#a^TBK=vuo9_b6 +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 +E59O{Bxc~qF + +literal 0 +HcmV?d00001 + +diff --git a/app/lte_test_user/ue_lte_user.conf b/app/lte_test_user/ue_lte_user.conf +new file mode 100644 +index 0000000..27d5676 +--- /dev/null ++++ b/app/lte_test_user/ue_lte_user.conf +@@ -0,0 +1,42 @@ ++#=============================================================================== ++# Brief : MIH-User configuration file ++# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> ++# Bruno Santos <bsantos@av.it.pt> ++#------------------------------------------------------------------------------- ++# ODTONE - Open Dot Twenty One ++# ++# Copyright (C) 2009-2012 Universidade Aveiro ++# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++# ++# This software is distributed under a license. The full license ++# agreement can be found in the file LICENSE in this distribution. ++# This software may not be copied, modified, sold or distributed ++# other than expressed in the named license agreement. ++# ++# This software is distributed without any warranty. ++#=============================================================================== ++ ++## ++## User id ++## ++[user] ++id = user_ue ++ ++## ++## Commands supported by the MIH-User ++## ++commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete ++ ++## ++## Port used for communication with MIHF ++## ++[conf] ++port = 1635 ++ ++## ++## MIHF configuration. For the default demonstration leave as is. ++## ++[mihf] ++local_port = 1025 ++ ++ +diff --git a/app/lte_test_user/ue_lte_user.cpp b/app/lte_test_user/ue_lte_user.cpp +new file mode 100644 +index 0000000..896958f +--- /dev/null ++++ b/app/lte_test_user/ue_lte_user.cpp +@@ -0,0 +1,1399 @@ ++//============================================================================== ++// Brief : MIH-User ++// Authors : Bruno Santos <bsantos@av.it.pt> ++//------------------------------------------------------------------------------ ++// ODTONE - Open Dot Twenty One ++// ++// Copyright (C) 2009-2012 Universidade Aveiro ++// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro ++// ++// This software is distributed under a license. The full license ++// agreement can be found in the file LICENSE in this distribution. ++// This software may not be copied, modified, sold or distributed ++// other than expressed in the named license agreement. ++// ++// This software is distributed without any warranty. ++//============================================================================== ++ ++#include <odtone/base.hpp> ++#include <odtone/debug.hpp> ++#include <odtone/logger.hpp> ++#include <odtone/mih/request.hpp> ++#include <odtone/mih/response.hpp> ++#include <odtone/mih/indication.hpp> ++#include <odtone/mih/confirm.hpp> ++#include <odtone/mih/tlv_types.hpp> ++#include <odtone/sap/user.hpp> ++ ++#include <boost/utility.hpp> ++#include <boost/bind.hpp> ++#include <boost/tokenizer.hpp> ++#include <boost/foreach.hpp> ++#include <boost/format.hpp> ++ ++#include <iostream> ++#include <map> ++#include <time.h> ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++// Definition of the scenario to execute ++#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests ++//#define SCENARIO_1 // Sequentially activate and deactivate each resource ++//#define SCENARIO_2 // Activate all resources, then deactivate all resources ++#define NUM_PARM_REPORT 10 ++ ++/////////////////////////////////////////////////////////////////////////////// ++// The scenario coded in this MIH-USER demo is the following ++// +--------+ +-----+ +---------+ ++// |MIH_USER| |MIH-F| |LINK_SAP | ++// +---+----+ +--+--+ +----+----+ ++// | | | ++// ... (start of MIH-F here) ... ... ++// | |---------- Link_Capability_Discover.request --------X| ++// ... (start of LINK_SAP here) ... ... ++// | |<--------- Link_Register.indication -----------------| ++// | |---------- Link_Capability_Discover.request -------->| ++// | |<--------- Link_Capability_Discover.confirm ---------| ++// | | | ++// ... (start of MIH USER here) ... ... ++// |---------- MIH_User_Register.indication ------------>| (supported_commands) | ++// | | | ++// |---------- MIH_Capability_Discover.request --------->| | ++// |<--------- MIH_Capability_Discover.confirm ----------| | ++// | | | ++// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| ++// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| ++// | | | ++// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| ++// | (POWER UP + SCAN) | (POWER UP + SCAN) | ++// ... ... ... ++// | | | ++// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| ++// | | | ++// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| ++// | | RRC Connection Reestablishment notification) | ++// | | | ++// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| ++// | | | ++// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| ++// | | (RRC Connection reconfiguration notification) | ++// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| ++// | | | ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| ++// | (Success) | | ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| ++// ... ... ... ++// etc etc etc ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++static const char* const kConf_MIH_Commands = "user.commands"; ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++namespace po = boost::program_options; ++ ++using odtone::uint; ++using odtone::ushort; ++using odtone::sint8; ++ ++odtone::logger log_("[mih_usr]", std::cout); ++ ++/////////////////////////////////////////////////////////////////////////////// ++ ++//----------------------------------------------------------------------------- ++void __trim(odtone::mih::octet_string &str, const char chr) ++//----------------------------------------------------------------------------- ++{ ++ str.erase(std::remove(str.begin(), str.end(), chr), str.end()); ++} ++//----------------------------------------------------------------------------- ++template <class T> std::string StringOf(T object) { ++//----------------------------------------------------------------------------- ++ std::ostringstream os; ++ os << object; ++ return(os.str()); ++} ++//----------------------------------------------------------------------------- ++std::string getTimeStamp4Log() ++//----------------------------------------------------------------------------- ++{ ++ std::stringstream ss (std::stringstream::in | std::stringstream::out); ++ struct timespec time_spec; ++ unsigned int time_now_micros; ++ unsigned int time_now_s; ++ clock_gettime (CLOCK_REALTIME, &time_spec); ++ time_now_s = (unsigned int) time_spec.tv_sec % 3600; ++ time_now_micros = (unsigned int) time_spec.tv_nsec/1000; ++ ss << time_now_s << ':' << time_now_micros; ++ return ss.str(); ++} ++//----------------------------------------------------------------------------- ++std::string status2string(odtone::mih::status statusP){ ++//----------------------------------------------------------------------------- ++ switch (statusP.get()) { ++ case odtone::mih::status_success: return "SUCCESS";break; ++ case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; ++ case odtone::mih::status_rejected: return "REJECTED";break; ++ case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; ++ case odtone::mih::status_network_error: return "NETWORK_ERROR";break; ++ default: return "UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ ++//----------------------------------------------------------------------------- ++ switch (reasonP.get()) { ++ case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; ++ case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; ++ case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; ++ case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; ++ case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; ++ case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; ++ default: return "DN_REASON_UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ ++//----------------------------------------------------------------------------- ++ switch (reasonP.get()) { ++ case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; ++ case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; ++ case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; ++ case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; ++ default: return "GD_REASON_UNKNOWN"; ++ } ++} ++//----------------------------------------------------------------------------- ++std::string evt2string(odtone::mih::mih_evt_list evtP){ ++//----------------------------------------------------------------------------- ++ std::string s=std::string(" "); ++ if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; ++ if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; ++ if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; ++ if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; ++ if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; ++ if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; ++ if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; ++ if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ ++//----------------------------------------------------------------------------- ++ std::string s=std::string(" "); ++ if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; ++ if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; ++ if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; ++ if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; ++ if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; ++ if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string link_type2string(const odtone::mih::link_type& lt) ++//----------------------------------------------------------------------------- ++{ ++ switch (lt.get()) { ++ case odtone::mih::link_type_gsm: return "GSM"; break; ++ case odtone::mih::link_type_gprs: return "GPRS"; break; ++ case odtone::mih::link_type_edge: return "EDGE"; break; ++ case odtone::mih::link_type_ethernet: return "Ethernet"; break; ++ case odtone::mih::link_type_wireless_other: return "Other"; break; ++ case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; ++ case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; ++ case odtone::mih::link_type_umts: return "UMTS"; break; ++ case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; ++ case odtone::mih::link_type_lte: return "LTE"; break; ++ case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; ++ case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; ++ case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; ++ default: break; ++ } ++ return "Unknown link type"; ++} ++//----------------------------------------------------------------------------- ++std::string link_addr2string(const odtone::mih::link_addr *addr) ++//----------------------------------------------------------------------------- ++{ ++ if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { ++ return la->address(); ++ } ++ else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { ++ char plmn[16]; ++ sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); ++ return str(boost::format("%s %d") % plmn % la->_cell_id); ++ } ++ else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { ++ char plmn[16]; ++ sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); ++ return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); ++ } ++ else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { ++ return la->value; ++ } ++ else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { ++ return la->value; ++ } ++ else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { ++ return la->value; ++ } ++ return "null"; ++} ++//----------------------------------------------------------------------------- ++std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) ++//----------------------------------------------------------------------------- ++{ ++ char buffer[256]; ++ int index = 0; ++ ++ index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); ++ index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); ++ return buffer; ++} ++//----------------------------------------------------------------------------- ++std::string link_id2string(odtone::mih::link_id linkP) ++//----------------------------------------------------------------------------- ++{ ++ std::string s; ++ s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ switch (ip_addrP.type()) { ++ case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; ++ case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; ++ default: s = "Unkown type "; ++ } ++ s += ip_addrP.address(); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { ++//----------------------------------------------------------------------------- ++ char buffer[128]; ++ std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); ++ return buffer; ++} ++//----------------------------------------------------------------------------- ++std::string ip_proto2string(odtone::mih::proto ip_protoP) { ++//----------------------------------------------------------------------------- ++ switch (ip_protoP.get()) { ++ case odtone::mih::proto_tcp: return "TCP"; ++ case odtone::mih::proto_udp: return "UDP"; ++ default: break; ++ } ++ return "Unknown IP protocol"; ++} ++// TEMP : next 2 functions are commented to restore flow_id as a uint32 ++// full structure will be updated later ++/*//----------------------------------------------------------------------------- ++std::string flow_id2string(odtone::mih::flow_id flowP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ odtone::mih::ip_tuple ip; ++ ip = flowP.src; ++ s = "SRC = " + ip_tuple2string(flowP.src); ++ s += ", DST = " + ip_tuple2string(flowP.dst); ++ s += ", PROTO = " + ip_proto2string(flowP.transport); ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { ++//----------------------------------------------------------------------------- ++ if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { ++ return flow_id2string(res->fid); ++ } ++ else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { ++ return flow_id2string(flow->id); ++ } ++ return "null"; ++}*/ ++//----------------------------------------------------------------------------- ++std::string link_ac_result2string(odtone::mih::link_ac_result resultP) ++//----------------------------------------------------------------------------- ++{ ++ switch (resultP.get()) { ++ case odtone::mih::link_ac_success: return "SUCCESS"; break; ++ case odtone::mih::link_ac_failure: return "FAILURE"; break; ++ case odtone::mih::link_ac_refused: return "REFUSED"; break; ++ case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; ++ default: break; ++ } ++ return "Unknown action result"; ++} ++//----------------------------------------------------------------------------- ++std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ ++ s = link_id2string(link_act_reqP.id); ++ ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; ++ if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; ++ ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; ++ if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; ++ ++ s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; ++ return s; ++} ++//----------------------------------------------------------------------------- ++std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { ++//----------------------------------------------------------------------------- ++ std::string s; ++ std::ostringstream stream; ++ odtone::mih::net_type_addr net_type_addr; ++ ++ for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) ++ { ++ net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); ++ stream << net_type_addr; ++ if (i != ntalP->begin()) { ++ stream << " / "; ++ } ++ } ++ s = stream.str(); ++ return s; ++} ++ ++ ++ ++/** ++ * Parse supported commands. ++ * ++ * @param cfg Configuration options. ++ * @return An optional list of supported commands. ++ */ ++boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) ++{ ++ using namespace boost; ++ ++ odtone::mih::mih_cmd_list commands; ++ ++ std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; ++ enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; ++ enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; ++ enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; ++ enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; ++ enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; ++ enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; ++ enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; ++ enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; ++ enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; ++ enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; ++ enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; ++ ++ std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); ++ __trim(tmp, ' '); ++ ++ char_separator<char> sep1(","); ++ tokenizer< char_separator<char> > list_tokens(tmp, sep1); ++ ++ BOOST_FOREACH(std::string str, list_tokens) { ++ if(enum_map.find(str) != enum_map.end()) { ++ commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); ++ } ++ } ++ ++ return commands; ++} ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * This class provides an implementation of an IEEE 802.21 MIH-User. ++ */ ++class mih_user : boost::noncopyable { ++public: ++ /** ++ * Construct the MIH-User. ++ * ++ * @param cfg Configuration options. ++ * @param io The io_service object that the MIH-User will use to ++ * dispatch handlers for any asynchronous operations performed on the socket. ++ */ ++ mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); ++ ++ /** ++ * Destruct the MIH-User. ++ */ ++ ~mih_user(); ++ ++protected: ++ /** ++ * User registration handler. ++ * ++ * @param cfg Configuration options. ++ * @param ec Error Code. ++ */ ++ void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); ++ /** ++ * Default MIH event handler. ++ * ++ * @param msg Received event notification. ++ * @param ec Error code. ++ */ ++ void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); ++ /** ++ * MIH receive message handler. ++ * ++ * @param msg Received message. ++ * @param ec Error code. ++ */ ++ void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++ void send_MIH_User_Register_indication(const odtone::mih::config& cfg); ++ ++ void send_MIH_Capability_Discover_request(void); ++ void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); ++ void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Event_Unsubscribe_request(void); ++ void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); ++ void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); ++ ++ void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); ++ void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); ++ void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++ void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); ++ void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); ++ ++private: ++ odtone::sap::user _mihf; /**< User SAP helper. */ ++ odtone::mih::id _mihfid; /**< MIHF destination ID. */ ++ odtone::mih::id _mihuserid; /**< MIH_USER ID. */ ++ ++ odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ ++ odtone::mih::port _mihf_lport; /**< MIHF local port number */ ++ ++ odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ ++ odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ ++ ++ odtone::mih::link_ac_type _last_link_action_type; ++ odtone::uint _current_link_action_request, _nb_of_link_action_requests; ++ odtone::uint link_threshold_request, link_measures_request, link_measures_counter; ++ odtone::mih::link_id rcv_link_id; ++ ++ static const odtone::uint _max_link_action_requests = 4; ++ odtone::uint _num_thresholds_request; ++ ++ void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); ++ void send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link); ++ void receive_MIH_Link_Up_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Down_indication(odtone::mih::message& msg); ++ ++ void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); ++ ++}; ++ ++//----------------------------------------------------------------------------- ++mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) ++ : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), ++ _last_link_action_type(odtone::mih::link_ac_type_none), ++ _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); ++ _mihuserid.assign(user_id.c_str()); ++ ++ odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); ++ _mihfid.assign(dest_id.c_str()); ++ ++ odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); ++ boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); ++ if (ip.is_v4()) { ++ odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); ++ _mihf_ip = ip_addr; ++ } else if (ip.is_v6()) { ++ odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); ++ _mihf_ip = ip_addr; ++ } ++ ++ _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); ++ ++ //_nb_of_link_action_requests = NB_OF_RESOURCES; ++ if (_nb_of_link_action_requests > _max_link_action_requests) { ++ _nb_of_link_action_requests = _max_link_action_requests; ++ } ++ ++ _link_id_list.clear(); ++ _subs_evt_list.clear(); ++ link_threshold_request = 0; ++ link_measures_request =0; ++ link_measures_counter =0; ++ log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); ++ ++ // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F ++ mih_user::send_MIH_User_Register_indication(cfg); ++} ++ ++//----------------------------------------------------------------------------- ++mih_user::~mih_user() ++//----------------------------------------------------------------------------- ++{ ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH-User register result: ", ec.message(), "\n"); ++ ++ // ++ // Let's fire a capability discover request to get things moving ++ // ++ mih_user::send_MIH_Capability_Discover_request(); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ if (ec) { ++ log_(0, __FUNCTION__, " error: ", ec.message()); ++ return; ++ } ++ ++ switch (msg.mid()) { ++ case odtone::mih::indication::link_detected: ++ mih_user::receive_MIH_Link_Detected_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_up: ++ mih_user::receive_MIH_Link_Up_indication(msg); ++ if (_num_thresholds_request == 0) { ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ _num_thresholds_request += 1; ++ } ++ break; ++ ++ case odtone::mih::indication::link_down: ++ mih_user::receive_MIH_Link_Down_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_going_down: ++ mih_user::receive_MIH_Link_Going_Down_indication(msg); ++ break; ++ ++ case odtone::mih::indication::link_handover_imminent: ++ log_(0, "MIH-User has received a local event \"link_handover_imminent\""); ++ break; ++ ++ case odtone::mih::indication::link_handover_complete: ++ log_(0, "MIH-User has received a local event \"link_handover_complete\""); ++ break; ++ ++ case odtone::mih::indication::link_parameters_report: ++ //log_(0, "MIH-User has received a local event \"link_parameters_report\""); ++ mih_user::receive_MIH_Link_Parameters_Report(msg, ec); ++ /*if (link_threshold_request == 0){ ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ link_threshold_request =1; ++ } else if (link_threshold_request == 1){ ++ link_measures_counter ++; ++ // Stop measures after 5 reports ++ if (link_measures_counter == NUM_PARM_REPORT){ ++ mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); ++ } ++ }*/ ++ break; ++ ++ case odtone::mih::indication::link_pdu_transmit_status: ++ log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); ++ break; ++ ++ case odtone::mih::confirm::link_configure_thresholds: ++ mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); ++ break; ++ ++ default: ++ log_(0, "MIH-User has received UNKNOWN local event"); ++ break; ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ if (ec) { ++ log_(0, __FUNCTION__, " error: ", ec.message()); ++ return; ++ } ++ ++ switch (msg.mid()) { ++ ++ case odtone::mih::confirm::capability_discover: ++ mih_user::receive_MIH_Capability_Discover_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::event_subscribe: ++ mih_user::receive_MIH_Event_Subscribe_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::event_unsubscribe: ++ mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); ++ break; ++ ++ case odtone::mih::confirm::link_actions: ++ mih_user::receive_MIH_Link_Actions_confirm(msg); ++ break; ++ ++ default: ++ log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); ++ break; ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); ++ odtone::mih::link_det_info ldi; ++ odtone::mih::link_det_info_list ldil; ++ odtone::mih::link_det_info_list::iterator it_ldil; ++ odtone::mih::link_id lid; ++ ++ msg >> odtone::mih::indication() & odtone::mih::tlv_link_det_info_list(ldil); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); ++ for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { ++ ldi = *it_ldil; ++ log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); ++ log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); ++ log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); ++ log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); ++ log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); ++ log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); ++ log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); ++ log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); ++ ++ } ++ // Display message parameters ++ // TODO: for each link_det_info in the list {display LINK_DET_INFO} ++ ++ // send Link_Action / Power Up ++ lid.type = odtone::mih::link_type_lte; ++ lid.addr = ldi.id.addr; ++ //send_MIH_Link_Action_Power_Up_plus_scan_request(lid); ++ log_(0, "MIH_Link_Detected.indication - End\n"); ++} ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::link_action_list lal; ++ odtone::mih::link_action_req link_act_req; ++ //struct null n; ++ ++ link_act_req.id = link; ++ link_act_req.action.type = odtone::mih::link_ac_type_power_up; ++ link_act_req.action.attr.clear(); ++ link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); ++ ++ link_act_req.ex_time = 5000; // in ms ++ ++ lal.push_back(link_act_req); ++ ++ m << odtone::mih::request(odtone::mih::request::link_actions) ++ & odtone::mih::tlv_link_action_list(lal); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); ++ ++ log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++// odtone::mih::tlv_old_access_router oldAR; ++ ++ msg >> odtone::mih::indication() & odtone::mih::tlv_link_identifier(link); ++// & odtone::mih::tlv_old_access_router(oar); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Up.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::link_addr> addr; ++ odtone::mih::link_dn_reason ldr; ++ ++ log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_old_access_router(addr) ++ & odtone::mih::tlv_link_dn_reason(ldr); ++ ++// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); ++ ++ //Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); ++// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Down.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); ++ ++ odtone::mih::link_tuple_id link; ++ odtone::mih::link_gd_reason lgd; ++ odtone::mih::link_ac_ex_time ex_time; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_time_interval(ex_time) ++ & odtone::mih::tlv_link_gd_reason(lgd); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - Time Interval:", (ex_time/256)); ++ log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); ++ ++ log_(0, "MIH_Link_Going_Down.indication - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id link; ++ odtone::mih::link_param_rpt_list lprl; ++ ++ msg >> odtone::mih::indication() ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_link_param_rpt_list(lprl); ++ ++ log_(0, ""); ++ log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, "MIH_Link_Parameters_Report.indication - End"); ++} ++ ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); ++ ++ m << odtone::mih::indication(odtone::mih::indication::user_register) ++ & odtone::mih::tlv_command_list(supp_cmd); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); ++ ++ log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Capability_Discover_request(void) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ m << odtone::mih::request(odtone::mih::request::capability_discover); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ boost::optional<odtone::mih::net_type_addr_list> ntal; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ boost::optional<odtone::mih::mih_cmd_list> cmd; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_net_type_addr_list(ntal) ++ & odtone::mih::tlv_event_list(evt) ++ & odtone::mih::tlv_command_list(cmd); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ if (cmd) { ++ log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); ++ } ++ if (ntal) { ++ log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); ++ //Store link address ++ for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) ++ { ++ rcv_link_id.addr = i->addr; ++ rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ } ++ } ++ log_(0, ""); ++ ++ // ++ // event subscription ++ // ++ // For every interface the MIHF sent in the ++ // Capability_Discover.response send an Event_Subscribe.request ++ // for all availabe events ++ // ++ if (ntal && evt) { ++ _subs_evt_list = evt.get(); // save the list of subscribed link events ++ for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { ++ if (i->nettype.link.which() == 1) ++ { ++ odtone::mih::link_tuple_id li; ++ ++ li.addr = i->addr; ++ li.type = boost::get<odtone::mih::link_type>(i->nettype.link); ++ _link_id_list.push_back(li); // save the link identifier of the network interface ++ ++ mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); ++ } ++ } ++ } ++ ++ log_(0, "MIH_Capability_Discover.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ m << odtone::mih::request(odtone::mih::request::event_subscribe) ++ & odtone::mih::tlv_link_identifier(li) ++ & odtone::mih::tlv_event_list(evt); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ ++ "\\nLink="+link_id2string(li).c_str()+ ++ "\\nEvent list="+evt2string(evt).c_str()+ ++ " --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); ++ ++ log_(0, "MIH_Event_Subscribe.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_event_list(evt); ++ ++ if (evt) { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } else { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ log_(0, ""); ++ ++ //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); ++ //log_(0, "TEMP : Resource scenario deactivated\n"); ++ ++ log_(0, "MIH_Event_Subscribe.confirm - End\n"); ++ mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(link); ++ ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Unsubscribe_request(void) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::link_tuple_id li; ++ ++ // For every interface the MIH user received in the ++ // Capability_Discover.confirm, send an Event_Unsubscribe.request ++ // for all subscribed events ++ for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { ++ li.type = i->type; ++ li.addr = i->addr; ++ mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); ++ } ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ ++ m << odtone::mih::request(odtone::mih::request::event_unsubscribe) ++ & odtone::mih::tlv_link_identifier(li) ++ & odtone::mih::tlv_event_list(evt); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ ++ "\\nLink="+link_id2string(li).c_str()+ ++ "\\nEvent list="+evt2string(evt).c_str()+ ++ " --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); ++ ++ log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ odtone::mih::link_tuple_id link; ++ boost::optional<odtone::mih::mih_evt_list> evt; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_identifier(link) ++ & odtone::mih::tlv_event_list(evt); ++ ++ if (evt) { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ "\\nEvent list="+evt2string(evt.get()).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } else { ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ "\\nLink="+link_id2string(link).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ } ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); ++ if (evt) { ++ log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); ++ } ++ log_(0, ""); ++ ++ log_(0, "MIH_Event_Unsubscribe.confirm - End"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::link_action_list lal; ++ odtone::mih::link_action_req link_act_req; ++ ++ link_act_req.id = link; ++ link_act_req.action.type = type; ++ ++ _last_link_action_type = type; ++ ++ // Initialize resource parameters ++ odtone::mih::resource_desc res; ++ ++ res.lid = link; // Link identifier ++ res.data_rate = 128000; // bit rate ++ res.jumbo = false; // jumbo disable ++ res.multicast = false; // multicast disable ++ ++ odtone::mih::qos qos; // Class Of Service ++ qos.value = 56; ++ res.qos_val = qos; ++ res.fid = 555 + _current_link_action_request; ++ ++// // Flow identifier ++// res.fid.src.ip = _mihf_ip; ++// res.fid.src.port_val = _mihf_lport; ++// ++// if (mih_user::_current_link_action_request == 0) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY ++// } ++// else if (mih_user::_current_link_action_request == 1) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY ++// } ++// else if (mih_user::_current_link_action_request == 2) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY ++// res.multicast = true; ++// } ++// else if (mih_user::_current_link_action_request == 3) { ++// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, ++// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY ++// } ++// res.fid.dst.port_val = 1235; // DUMMY ++// res.fid.transport = odtone::mih::proto_udp; ++ ++ link_act_req.action.param.param = res; ++ ++ link_act_req.ex_time = 0; ++ ++ lal.push_back(link_act_req); ++ ++ m << odtone::mih::request(odtone::mih::request::link_actions) ++ & odtone::mih::tlv_link_action_list(lal); ++ m.source(_mihuserid); ++ m.destination(_mihfid); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); ++ ++ log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); ++ log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); ++ log_(0, " - FLOW_ID - Flow identifier: ", res.fid); ++//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); ++ log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); ++ ++ log_(0, "MIH_Link_Actions.request - SENT\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); ++ ++ odtone::mih::status st; ++ boost::optional<odtone::mih::link_action_rsp_list> larl; ++ ++ msg >> odtone::mih::confirm() ++ & odtone::mih::tlv_status(st) ++ & odtone::mih::tlv_link_action_rsp_list(larl); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ ++ "\\nstatus="+status2string(st).c_str()+ ++ " --->]["+msg.destination().to_string()+"]\n"); ++ ++ // Display message parameters ++ log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); ++ if (larl) { ++ log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); ++ for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) ++ { ++ log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), ++ ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); ++ } ++ } ++ log_(0, ""); ++ ++ // 1st scenario: Sequentially activate and deactivate each resource ++#ifdef SCENARIO_1 ++ if (larl) { ++ odtone::mih::link_action_rsp *rsp = &larl->front(); ++ if (_current_link_action_request < _nb_of_link_action_requests) { ++ if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++ if (rsp->result.get() == odtone::mih::link_ac_success) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++ _current_link_action_request += 1; ++ } ++ } ++ else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); ++ } ++ } ++ else { // Ends the scenario ++ mih_user::send_MIH_Event_Unsubscribe_request(); ++ } ++ } ++#endif // SCENARIO_1 ++ ++#ifdef SCENARIO_2 ++ // 2nd scenario: Activate all resources, then deactivate all resources ++ if (larl.get().size() > 0) { ++ odtone::mih::link_action_rsp *rsp = &larl->front(); ++ if (++_current_link_action_request < _nb_of_link_action_requests) { ++ if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); ++ } ++ else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++ } ++ } ++ else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { ++ _current_link_action_request = 0; ++ mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); ++ } ++ else { // Ends the scenario ++ mih_user::send_MIH_Event_Unsubscribe_request(); ++ } ++ } ++#endif // SCENARIO_2 ++ ++ log_(0, "MIH_Link_Actions.confirm - End\n"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ odtone::mih::message m; ++ odtone::mih::threshold th; ++ std::vector<odtone::mih::threshold> thl; ++ odtone::mih::link_tuple_id lti; ++ odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; ++ //List of the link threshold parameters ++ odtone::mih::link_cfg_param_list lcpl; ++ odtone::mih::link_cfg_param lcp; ++ odtone::mih::link_param_lte lp; ++ //odtone::mih::link_param_gen lp; ++ ++ odtone::mih::link_param_type typr; ++ ++ log_(0,""); ++ log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); ++ ++ //link_tuple_id ++ lti.type = rcv_link_id.type; ++ lti.addr = rcv_link_id.addr; ++ ++ //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); ++ ++ //link_param_gen_data_rate = 0, /**< Data rate. */ ++ //link_param_gen_signal_strength = 1, /**< Signal strength. */ ++ //link_param_gen_sinr = 2, /**< SINR. */ ++ //link_param_gen_throughput = 3, /**< Throughput. */ ++ //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ ++ //lp = odtone::mih::link_param_lte_bandwidth; ++ lp = odtone::mih::link_param_lte_rsrp; ++ lcp.type = lp; ++ ++ link_measures_request = 0; ++ if ( link_measures_request ==0){ ++ // Set Timer Interval (in ms) ++ lcp.timer_interval = 3000; ++ //th_action_normal = 0, /**< Set normal threshold. */ ++ //th_action_one_shot = 1, /**< Set one-shot threshold. */ ++ //th_action_cancel = 2 /**< Cancel threshold. */ ++ lcp.action = odtone::mih::th_action_normal; ++ link_measures_request = 1; ++ } else if ( link_measures_request==1){ ++ // Set Timer Interval (in ms) ++ lcp.timer_interval = 0; ++ lcp.action = odtone::mih::th_action_cancel; ++ link_measures_request = 0; ++ } ++ ++ //above_threshold = 0, /**< Above threshold. */ ++ //below_threshold = 1, /**< Below threshold. */ ++ th.threshold_val = -105; ++ th.threshold_x_dir = odtone::mih::threshold::above_threshold; ++ ++ thl.push_back(th); ++ lcp.threshold_list = thl; ++ lcpl.push_back(lcp); ++ ++ m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) ++ & odtone::mih::tlv_link_identifier(lti) ++ & odtone::mih::tlv_link_cfg_param_list(lcpl); ++ ++ m.destination(msg.source()); ++ ++ log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ ++ // link_tupple_id2string(lti).c_str() + ++ link_id2string(lti).c_str()+ ++ " --->]["+_mihfid.to_string()+"]\n"); ++ ++ _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); ++ ++ log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); ++ ++ log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); ++ ++ //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} ++ //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} ++ //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} ++ //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} ++ //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} ++ if(lp == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} ++ ++ log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); ++ ++ if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} ++ if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} ++ if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} ++ ++ log_(0, "\t Threshold value: ", th.threshold_val); ++ ++ if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} ++ if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} ++ ++ log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); ++} ++ ++//----------------------------------------------------------------------------- ++void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) ++//----------------------------------------------------------------------------- ++{ ++ log_(0, ""); ++ log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); ++ ++ // T odtone::uint iter; ++ // T odtone::mih::status st; ++ ++ //boost::optional<odtone::mih::link_cfg_status_list> lcsl; ++ // Todtone::mih::link_cfg_status_list lcsl; ++ // Todtone::mih::link_cfg_status lcp; ++ //odtone::mih::link_param_gen lp; ++ ++ // T odtone::mih::link_tuple_id lti; ++ ++ //msg >> odtone::mih::confirm() ++ // & odtone::mih::tlv_status(st) ++ // & odtone::mih::tlv_link_identifier(lti) ++ // & odtone::mih::tlv_link_cfg_status_list(lcsl); ++ ++ ++ log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); ++ log_(0,""); ++} ++ ++//----------------------------------------------------------------------------- ++int main(int argc, char** argv) ++//----------------------------------------------------------------------------- ++{ ++ odtone::setup_crash_handler(); ++ ++ try { ++ boost::asio::io_service ios; ++ ++ // declare MIH Usr available options ++ po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); ++ desc.add_options() ++ ("help", "Display configuration options") ++ (odtone::sap::kConf_File, po::value<std::string>()->default_value("ue_lte_user.conf"), "Configuration file") ++ (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") ++ (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") ++ (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") ++ (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") ++ (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") ++ (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") ++ (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf2_ue"), "MIHF destination"); ++ ++ odtone::mih::config cfg(desc); ++ cfg.parse(argc, argv, odtone::sap::kConf_File); ++ ++ if (cfg.help()) { ++ std::cerr << desc << std::endl; ++ return EXIT_SUCCESS; ++ } ++ ++ mih_user usr(cfg, ios); ++ ++ ios.run(); ++ ++ } catch(std::exception& e) { ++ log_(0, "exception: ", e.what()); ++ } ++} ++ ++// EOF //////////////////////////////////////////////////////////////////////// ++ +diff --git a/boost-build.jam b/boost-build.jam +new file mode 100644 +index 0000000..f5b963e +--- /dev/null ++++ b/boost-build.jam +@@ -0,0 +1 @@ ++boost-build /home/nikaia/DEMO_SPECTRA/boost_1_49_0//tools/build/v2 ; +diff --git a/inc/odtone/conf.hpp b/inc/odtone/conf.hpp +new file mode 100644 +index 0000000..61b773c +--- /dev/null ++++ b/inc/odtone/conf.hpp +@@ -0,0 +1,670 @@ ++//======================================================================================================= ++// Brief : Configuration DSL ++// Authors : Bruno Santos <bsantos@av.it.pt> ++// ------------------------------------------------------------------------------------------------------ ++// ODTONE - Open Dot Twenty One ++// ++// Copyright (C) 2009-2013 Universidade Aveiro ++// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro ++// ++// This software is distributed under a license. The full license ++// agreement can be found in the file LICENSE in this distribution. ++// This software may not be copied, modified, sold or distributed ++// other than expressed in the named license agreement. ++// ++// This software is distributed without any warranty. ++//======================================================================================================= ++ ++#ifndef ODTONE_PMIP_CONF__HPP_ ++#define ODTONE_PMIP_CONF__HPP_ ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++#include <odtone/base.hpp> ++#include <odtone/net/ip/address.hpp> ++#include <boost/variant.hpp> ++#include <boost/function.hpp> ++#include <boost/mpl/for_each.hpp> ++#include <boost/mpl/vector.hpp> ++#include <boost/mpl/at.hpp> ++#include <boost/mpl/back.hpp> ++#include <boost/mpl/deref.hpp> ++#include <boost/mpl/prior.hpp> ++#include <boost/function_types/result_type.hpp> ++#include <boost/function_types/parameter_types.hpp> ++#include <boost/function_types/function_arity.hpp> ++#include <boost/spirit/include/qi.hpp> ++#include <boost/spirit/include/phoenix_core.hpp> ++#include <boost/spirit/include/phoenix_operator.hpp> ++#include <boost/spirit/include/phoenix_bind.hpp> ++#include <boost/spirit/include/phoenix_fusion.hpp> ++#include <boost/spirit/include/phoenix_stl.hpp> ++#include <boost/fusion/include/std_pair.hpp> ++#include <boost/mpl/find.hpp> ++#include <algorithm> ++#include <cstring> ++#include <string> ++#include <map> ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++namespace odtone { namespace conf { ++ ++namespace qi = boost::spirit::qi; ++namespace ph = boost::phoenix; ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++enum error_reason { ++ invalid_syntax, ++ invalid_command, ++ invalid_arg_type, ++ to_many_args, ++ invalid_prop, ++ invalid_prop_arg_type, ++ to_many_prop_args, ++ ++ error_reason_size ++}; ++ ++static char const* const error_reason_string[error_reason_size] = { ++ "invalid syntax", ++ "invalid command", ++ "invalid argument type", ++ "to many arguments", ++ "invalid property", ++ "invalid property argument type", ++ "to many property arguments" ++}; ++ ++typedef boost::variant<uint, ++ sint, ++ double, ++ std::string, ++ net::ip::address_v4, ++ net::ip::address_v6 ++ > arg_type; ++ ++typedef std::vector<arg_type> args_type; ++typedef std::pair<std::string, args_type> prop_type; ++typedef std::map<std::string, args_type> pset_type; ++ ++template<class T> ++class type_id_ { ++ typedef typename boost::remove_cv<typename boost::remove_reference<T>::type>::type tt; ++ typedef typename boost::mpl::find<arg_type::types, tt>::type ft; ++ typedef typename boost::mpl::distance<boost::mpl::begin<arg_type::types>::type, ft>::type tp; ++ ++ ODTONE_STATIC_ASSERT( ++ (!boost::is_same<typename boost::mpl::end<arg_type::types>::type, ft>::value), ++ "Type not supported" ++ ); ++ ++public: ++ static const uint value = tp::value; ++}; ++ ++struct property_class { ++ const char* name; //property name ++ uint type_id; //property argument type ++ uint count; //maximum number of arguments it can take ++}; ++ ++class function { ++ template<class F, size_t N> ++ class at_ { ++ typedef typename boost::function_types::parameter_types<F>::type ps; ++ typedef typename boost::mpl::at_c<ps, N>::type tp; ++ ++ public: ++ typedef typename boost::remove_cv<typename boost::remove_reference<tp>::type>::type type; ++ }; ++ ++ template<class F, size_t N> ++ static const typename at_<F, N>::type& get_(arg_type const& arg) ++ { ++ typedef typename at_<F, N>::type type; ++ ++ return boost::get<type>(arg); ++ } ++ ++ template<class F, class WithPropSet, size_t N> ++ struct adaptor_; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::true_, 1> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), pset); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::false_, 1> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0])); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::true_, 2> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), pset); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::false_, 2> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1])); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::true_, 3> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), pset); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::false_, 3> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2])); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::true_, 4> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), ++ get_<F, 3>(args[3]), pset); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::false_, 4> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), ++ get_<F, 3>(args[3])); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::true_, 5> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), ++ get_<F, 3>(args[3]), get_<F, 4>(args[4]), pset); ++ } ++ }; ++ ++ template<class F> ++ struct adaptor_<F, boost::mpl::false_, 5> { ++ F f; ++ ++ adaptor_(F f_) : f(f_) ++ { } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), ++ get_<F, 3>(args[3]), get_<F, 4>(args[4])); ++ } ++ }; ++ ++ template<class F, class WithPropSet> ++ class traits_; ++ ++ template<class F> ++ class traits_<F, boost::mpl::true_> { ++ typedef typename boost::function_types::parameter_types<F>::type tmp; ++ typedef typename boost::mpl::prior<typename boost::mpl::end<tmp>::type>::type last; ++ ++ ODTONE_STATIC_ASSERT( ++ (boost::is_same<void, ++ typename boost::function_types::result_type<F>::type>::value), ++ "Result type must be void" ++ ); ++ ++ ODTONE_STATIC_ASSERT( ++ (boost::is_same<pset_type const&, ++ typename boost::mpl::deref<last>::type>::value), ++ "Last argument must a const reference to a pset_type" ++ ); ++ ++ public: ++ typedef typename boost::mpl::erase<tmp, last>::type arg_types; ++ ++ static const size_t arg_count = boost::mpl::size<arg_types>::value; ++ }; ++ ++ template<class F> ++ class traits_<F, boost::mpl::false_> { ++ ODTONE_STATIC_ASSERT( ++ (boost::is_same<void, ++ typename boost::function_types::result_type<F>::type>::value), ++ "Result type must be void" ++ ); ++ ++ public: ++ typedef typename boost::function_types::parameter_types<F>::type arg_types; ++ ++ static const size_t arg_count = boost::mpl::size<arg_types>::value; ++ }; ++ ++ struct push_arg_id_ { ++ push_arg_id_(std::vector<uint>& ais) ++ : args_id(ais) ++ { } ++ ++ template<class T> ++ void operator()(T) ++ { ++ const uint id = type_id_<T>::value; ++ ++ args_id.push_back(id); ++ } ++ ++ std::vector<uint>& args_id; ++ }; ++ ++public: ++ function() ++ : _pclass(0), _pcsize(0) ++ {} ++ ++ template<class F> ++ function(F f) ++ : _pclass(0), _pcsize(0) ++ { ++ typedef typename traits_<F, boost::mpl::false_>::arg_types types; ++ ++ boost::mpl::for_each< ++ types, ++ boost::remove_cv< boost::remove_reference<boost::mpl::_1> > ++ >(push_arg_id_(_args)); ++ ++ _func = adaptor_<F, boost::mpl::false_, traits_<F, boost::mpl::false_>::arg_count>(f); ++ } ++ ++ template<class F, size_t N> ++ function(F f, property_class const (&pc)[N]) ++ : _pclass(pc), _pcsize(N) ++ { ++ typedef typename traits_<F, boost::mpl::true_>::arg_types types; ++ ++ boost::mpl::for_each< ++ types, ++ boost::remove_cv< boost::remove_reference<boost::mpl::_1> > ++ >(push_arg_id_(_args)); ++ ++ _func = adaptor_<F, boost::mpl::true_, traits_<F, boost::mpl::true_>::arg_count>(f); ++ } ++ ++ void operator()(args_type const& args, pset_type const& pset) const ++ { ++ _func(args, pset); ++ } ++ ++ size_t max_args() const ++ { ++ return _args.size(); ++ } ++ ++ uint arg_id(uint n) const ++ { ++ return _args[n]; ++ } ++ ++ property_class const* get_prop_class(std::string const& name) const ++ { ++ return std::find_if(_pclass, _pclass + _pcsize, ++ !ph::bind(&std::strcmp, name.c_str(), ++ ph::bind(&property_class::name, ph::arg_names::arg1))); ++ } ++ ++private: ++ std::vector<uint> _args; ++ property_class const* _pclass; ++ size_t _pcsize; ++ ++ boost::function<void(args_type const& args, pset_type const& pset)> _func; ++}; ++ ++typedef std::map<std::string, function> functions; ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++template<class Iterator> ++struct skipper_grammar : qi::grammar<Iterator> { ++ skipper_grammar() ++ : skipper_grammar::base_type(sk, "skipper") ++ { ++ sk = ++ qi::blank ++ | ('#' >> *(qi::standard::char_ - qi::eol) > &qi::eol) ++ | ('\\' >> qi::eol) ++ ; ++ } ++ ++ qi::rule<Iterator> sk; ++}; ++ ++template<class Iterator> ++struct string_grammar : qi::grammar<Iterator, std::string()> { ++ string_grammar() ++ : string_grammar::base_type(str, "string") ++ { ++ esc.add("\\\"", '\"') ++ ("\\\\", '\\') ++ ("\\t", '\t') ++ ("\\n", '\n') ++ ("\\r", '\r') ++ ; ++ ++ str = ++ '"' ++ >> qi::as_string[*((esc | qi::standard::char_) - '"')][qi::_val = qi::_1] ++ > '"' ++ ; ++ } ++ ++ qi::symbols<const char, const char> esc; ++ qi::rule<Iterator, std::string()> str; ++}; ++ ++template<class Iterator> ++struct ip4_grammar : qi::grammar<Iterator, net::ip::address_v4()> { ++ ++ static net::ip::address_v4 conv(std::string const& str) ++ { ++ return net::ip::address_v4::from_string(str); ++ } ++ ++ ip4_grammar() ++ : ip4_grammar::base_type(ip4, "ip4") ++ { ++ ip4 = qi::as_string ++ [ ++ qi::raw ++ [ ++ qi::repeat(3)[u8 >> '.'] >> u8 ++ ] ++ ] ++ [qi::_val = ph::bind(&ip4_grammar::conv, qi::_1)]; ++ } ++ ++ qi::rule<Iterator, net::ip::address_v4()> ip4; ++ qi::uint_parser<uint8, 10, 1, 3> u8; ++}; ++ ++template<class Iterator> ++struct ip6_grammar : qi::grammar<Iterator, qi::locals<uint>, net::ip::address_v6()> { ++ typedef qi::locals<uint> locals; ++ ++ static net::ip::address_v6 conv(std::string const& str) ++ { ++ return net::ip::address_v6::from_string(str); ++ } ++ ++ ip6_grammar() ++ : ip6_grammar::base_type(ip6, "ip6") ++ { ++ ip6 = qi::as_string ++ [ ++ qi::raw ++ [ ++ ( qi::repeat(1, 6)[h16 >> ':'] >> ((h16 >> (':' >> (h16 | ':'))) | ':' | ip4) ) ++ | ( "::" >> -(qi::repeat(0, 5)[h16 >> ':'] >> ((h16 >> -(':' >> (h16 ))) | ip4)) ) ++ | ( ++ qi::eps[qi::_a = 0] ++ >> +(h16[qi::_pass = ++qi::_a < 7u] >> ':') ++ >> +(':' >> h16[qi::_pass = ++qi::_a < 8u]) ++ ) ++ ] ++ ] ++ [qi::_val = ph::bind(&ip6_grammar::conv, qi::_1)]; ++ } ++ ++ qi::rule<Iterator, locals, net::ip::address_v6()> ip6; ++ qi::uint_parser<uint16, 16, 1, 4> h16; ++ ip4_grammar<Iterator> ip4; ++}; ++ ++template<class Iterator> ++struct parser_grammar : qi::grammar<Iterator, ++ skipper_grammar<Iterator> ++ > { ++ typedef skipper_grammar<Iterator> skipper; ++ ++ struct act_on_name_ { ++ template<class, class, class, class> ++ struct result { typedef bool type; }; ++ ++ bool operator()(function const*& f, functions const& cm, std::string const& name, error_reason& er) const ++ { ++ functions::const_iterator it = cm.find(name); ++ ++ if (it == cm.end()) { ++ er = invalid_command; ++ return false; ++ } ++ ++ f = &it->second; ++ return true; ++ } ++ }; ++ ++ struct act_on_arg_ { ++ template<class, class, class, class> ++ struct result { typedef bool type; }; ++ ++ bool operator()(function const& f, args_type& args, arg_type& arg, error_reason& er) const ++ { ++ const uint n = args.size(); ++ ++ if (n >= f.max_args()) { ++ er = to_many_args; ++ return false; ++ } ++ ++ if (static_cast<uint>(arg.which()) != f.arg_id(n)) { ++ er = invalid_arg_type; ++ return false; ++ } ++ ++ args.push_back(arg); ++ return true; ++ } ++ }; ++ ++ struct act_on_prop_ { ++ template<class, class, class, class> ++ struct result { typedef bool type; }; ++ ++ bool operator()(property_class const*& pc, function const& f, std::string const& name, error_reason& er) const ++ { ++ pc = f.get_prop_class(name); ++ ++ if (!pc) { ++ er = invalid_prop; ++ return false; ++ } ++ return true; ++ } ++ ++ bool operator()(property_class const* pc, arg_type const& arg, uint& count, error_reason& er) const ++ { ++ if (++count > pc->count) { ++ er = to_many_prop_args; ++ return false; ++ } ++ if (pc->type_id != uint(arg.which())) { ++ er = invalid_prop_arg_type; ++ return false; ++ } ++ return true; ++ } ++ }; ++ ++ struct run_cmd_ { ++ template<class, class, class> ++ struct result { typedef void type; }; ++ ++ void operator()(function const& f, args_type const& args, pset_type const& pset) const ++ { ++ f(args, pset); ++ } ++ }; ++ ++ struct error_handler_ { ++ template<class, class, class, class, class> ++ struct result { typedef void type; }; ++ ++ void operator()(Iterator begin, Iterator end, Iterator pos, qi::info const& what, error_reason er) const ++ { ++ Iterator eol = std::find(begin, end, '\n'); ++ ++ std::cout << "error: " ++ << error_reason_string[er] ++ << ", expecting \'" ++ << what ++ << "\':\n" ++ << std::string(begin, eol) ++ << std::endl ++ << std::string(std::distance(begin, pos), '~') ++ << '^' ++ << std::string(std::distance(pos, eol), '~') ++ << std::endl; ++ } ++ }; ++ ++ parser_grammar(functions const& cm) ++ : parser_grammar::base_type(start), cmds(cm), ereason(invalid_syntax) ++ { ++ start = qi::eol | cmd; ++ ++ cmd = ( ++ name [qi::_pass = act_on_name(qi::_a, ph::cref(cmds), qi::_1, ph::ref(ereason))] ++ > *arg [qi::_pass = act_on_arg(*qi::_a, qi::_b, qi::_1, ph::ref(ereason))] ++ > -pset(*qi::_a) [qi::_c = qi::_1] ++ > qi::eol ++ ) ++ [run_cmd(*qi::_a, qi::_b, qi::_c)]; ++ ++ name %= qi::lexeme[qi::alpha >> *(qi::alnum | '-')]; ++ ++ arg %= ++ ip4 ++ | ip6 ++ | qi::uint_ ++ | qi::int_ ++ | qi::double_ ++ | str ++ ; ++ ++ pset = ++ qi::lit('{') ++ > qi::eol ++ > +prop(qi::_r1) ++ > qi::lit('}') ++ ; ++ ++ prop = ++ name[qi::_val = qi::_1, qi::_a = 0, ++ qi::_pass = act_on_prop(qi::_a, qi::_r1, qi::_1, ph::ref(ereason))] ++ > (*arg[qi::_val = qi::_1, ++ qi::_pass = act_on_prop(qi::_a, qi::_1, qi::_b, ph::ref(ereason))]) ++ > qi::eol ++ ; ++ ++ cmd.name("command"); ++ name.name("identifier"); ++ arg.name("argument"); ++ pset.name("property-set"); ++ prop.name("property"); ++ ++ qi::on_error<qi::fail>(start, error_handler(qi::_1, qi::_2, qi::_3, qi::_4, ph::ref(ereason))); ++ } ++ ++ typedef qi::locals<function const*, args_type, pset_type> cmd_locals; ++ typedef qi::locals<property_class const*, uint> prop_locals; ++ ++ qi::rule<Iterator, skipper> start; ++ qi::rule<Iterator, cmd_locals, skipper> cmd; ++ qi::rule<Iterator, std::string(), skipper> name; ++ qi::rule<Iterator, arg_type(), skipper> arg; ++ qi::rule<Iterator, pset_type(function const&), skipper> pset; ++ qi::rule<Iterator, prop_type(function const&), prop_locals, skipper> prop; ++ ++ string_grammar<Iterator> str; ++ ip4_grammar<Iterator> ip4; ++ ip6_grammar<Iterator> ip6; ++ ++ functions const& cmds; ++ error_reason ereason; ++ ++ ph::function<act_on_name_> act_on_name; ++ ph::function<act_on_arg_> act_on_arg; ++ ph::function<act_on_prop_> act_on_prop; ++ ph::function<run_cmd_> run_cmd; ++ ph::function<error_handler_> error_handler; ++}; ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++bool exec(std::string::const_iterator& begin, std::string::const_iterator end, functions const& cm); ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++} /* namespace conf */ } /* namespace odtone */ ++ ++// EOF ////////////////////////////////////////////////////////////////////////////////////////////////// ++#endif /* ODTONE_PMIP_CONF__HPP_ */ +diff --git a/inc/odtone/mih/detail/archive.hpp b/inc/odtone/mih/detail/archive.hpp +index 42925f1..f2225b9 100644 +--- a/inc/odtone/mih/detail/archive.hpp ++++ b/inc/odtone/mih/detail/archive.hpp +@@ -1146,6 +1146,148 @@ struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T1 + }; + + ++template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8, class T9, class T10, class T11, class T12, class T13> ++struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13> > { ++ void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13>& val) const ++ { ++ octet selector; ++ ++ ar & selector; ++ switch (selector) { ++ case 0: ++ val = T1(); ++ ar & boost::get<T1>(val); ++ break; ++ ++ case 1: ++ val = T2(); ++ ar & boost::get<T2>(val); ++ break; ++ ++ case 2: ++ val = T3(); ++ ar & boost::get<T3>(val); ++ break; ++ ++ case 3: ++ val = T4(); ++ ar & boost::get<T4>(val); ++ break; ++ ++ case 4: ++ val = T5(); ++ ar & boost::get<T5>(val); ++ break; ++ ++ case 5: ++ val = T6(); ++ ar & boost::get<T6>(val); ++ break; ++ ++ case 6: ++ val = T7(); ++ ar & boost::get<T7>(val); ++ break; ++ ++ case 7: ++ val = T8(); ++ ar & boost::get<T8>(val); ++ break; ++ ++ case 8: ++ val = T9(); ++ ar & boost::get<T9>(val); ++ break; ++ ++ case 9: ++ val = T10(); ++ ar & boost::get<T10>(val); ++ break; ++ ++ case 10: ++ val = T11(); ++ ar & boost::get<T11>(val); ++ break; ++ ++ case 11: ++ val = T12(); ++ ar & boost::get<T12>(val); ++ break; ++ ++ case 12: ++ val = T13(); ++ ar & boost::get<T13>(val); ++ break; ++ ++ // default: ++ // ODTONE_NEVER_HERE; ++ } ++ } ++ ++ void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13>& val) const ++ { ++ octet selector = val.which(); ++ ++ ar & selector; ++ switch (selector) { ++ case 0: ++ ar & boost::get<T1>(val); ++ break; ++ ++ case 1: ++ ar & boost::get<T2>(val); ++ break; ++ ++ case 2: ++ ar & boost::get<T3>(val); ++ break; ++ ++ case 3: ++ ar & boost::get<T4>(val); ++ break; ++ ++ case 4: ++ ar & boost::get<T5>(val); ++ break; ++ ++ case 5: ++ ar & boost::get<T6>(val); ++ break; ++ ++ case 6: ++ ar & boost::get<T7>(val); ++ break; ++ ++ case 7: ++ ar & boost::get<T8>(val); ++ break; ++ ++ case 8: ++ ar & boost::get<T9>(val); ++ break; ++ ++ case 9: ++ ar & boost::get<T10>(val); ++ break; ++ ++ case 10: ++ ar & boost::get<T11>(val); ++ break; ++ ++ case 11: ++ ar & boost::get<T12>(val); ++ break; ++ ++ case 12: ++ ar & boost::get<T13>(val); ++ break; ++ ++ // default: ++ // ODTONE_NEVER_HERE; ++ } ++ } ++}; ++ + /////////////////////////////////////////////////////////////////////////////// + } /* namspace detail */ + +diff --git a/inc/odtone/mih/types/address.hpp b/inc/odtone/mih/types/address.hpp +index 18e586c..18fac4c 100644 +--- a/inc/odtone/mih/types/address.hpp ++++ b/inc/odtone/mih/types/address.hpp +@@ -20,6 +20,7 @@ + + /////////////////////////////////////////////////////////////////////////////// + #include <odtone/mih/types/base.hpp> ++#include <iostream> + + /////////////////////////////////////////////////////////////////////////////// + namespace odtone { namespace mih { +@@ -236,6 +237,7 @@ struct l2_3gpp_2g_cell_id { + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp_2g_cell_id& addr) + { ++ out << "l2_3gpp_2g_cell_id: CI:" << addr._ci << " LAC:" << addr._lac << " PLMN id:" << addr.plmn_id[0]<< addr.plmn_id[1]<< addr.plmn_id[2]; + return out; + } + +@@ -247,6 +249,12 @@ struct l2_3gpp_2g_cell_id { + */ + bool operator==(const l2_3gpp_2g_cell_id& other) const + { ++ std::cerr << "l2_3gpp_2g_cell_id()==" << std::endl; ++ std::cerr << "_lac " << _lac << " " << other._lac << std::endl; ++ std::cerr << "_ci " << _ci << " " << other._ci << std::endl; ++ std::cerr << "plmn_id " << plmn_id[0] << " " << other.plmn_id[0] << std::endl; ++ std::cerr << "plmn_id " << plmn_id[1] << " " << other.plmn_id[1] << std::endl; ++ std::cerr << "plmn_id " << plmn_id[2] << " " << other.plmn_id[2] << std::endl; + return ((_lac == other._lac) + && (_ci == other._ci) + && (plmn_id[0] == other.plmn_id[0]) +@@ -285,6 +293,7 @@ struct l2_3gpp_3g_cell_id { + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp_3g_cell_id& addr) + { ++ out << "l2_3gpp_3g_cell_id: Cell id:" << addr._cell_id << " PLMN id:" << addr.plmn_id[0]<< addr.plmn_id[1]<< addr.plmn_id[2]; + return out; + } + +@@ -296,6 +305,11 @@ struct l2_3gpp_3g_cell_id { + */ + bool operator==(const l2_3gpp_3g_cell_id& other) const + { ++ std::cerr << "l2_3gpp_3g_cell_id()==" << std::endl; ++ std::cerr << "_cell_id " << _cell_id << " " << other._cell_id << std::endl; ++ std::cerr << "plmn_id " << plmn_id[0] << " " << other.plmn_id[0] << std::endl; ++ std::cerr << "plmn_id " << plmn_id[1] << " " << other.plmn_id[1] << std::endl; ++ std::cerr << "plmn_id " << plmn_id[2] << " " << other.plmn_id[2] << std::endl; + return ((_cell_id == other._cell_id) + && (plmn_id[0] == other.plmn_id[0]) + && (plmn_id[1] == other.plmn_id[1]) +@@ -328,6 +342,7 @@ struct l2_3gpp_addr { + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp_addr& addr) + { ++ out << "l2_3gpp_addr " << addr.value; + return out; + } + +@@ -339,7 +354,7 @@ struct l2_3gpp_addr { + */ + bool operator==(const l2_3gpp_addr& other) const + { +- return ((value == other.value)); ++ return ((value.compare(other.value) == 0)); + } + + octet_string value; /**< L2_3GPP_ADDR data type. */ +@@ -370,6 +385,7 @@ struct l2_3gpp2_addr { + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp2_addr& addr) + { ++ out << "l2_3gpp2_addr:" << addr.value; + return out; + } + +@@ -412,6 +428,7 @@ struct other_l2_addr { + */ + friend std::ostream& operator<<(std::ostream& out, const other_l2_addr& addr) + { ++ out << "other_l2_addr"; + return out; + } + +diff --git a/inc/odtone/mih/types/bin_query.hpp b/inc/odtone/mih/types/bin_query.hpp +index f418acf..06e6c6b 100644 +--- a/inc/odtone/mih/types/bin_query.hpp ++++ b/inc/odtone/mih/types/bin_query.hpp +@@ -50,6 +50,7 @@ enum net_type_inc_enum { + net_type_inc_ieee802_16 = 9, /**< IEEE 802.16 */ + net_type_inc_ieee802_20 = 10, /**< IEEE 802.20 */ + net_type_inc_ieee802_22 = 11, /**< IEEE 802.22 */ ++ net_type_inc_lte = 12, /**< LTE */ + }; + + /** +diff --git a/inc/odtone/mih/types/link.hpp b/inc/odtone/mih/types/link.hpp +index 80381b4..f9c8718 100644 +--- a/inc/odtone/mih/types/link.hpp ++++ b/inc/odtone/mih/types/link.hpp +@@ -73,6 +73,74 @@ typedef boost::variant<sint8, percentage> sig_strength; + */ + typedef bool link_res_status; + ++/** ++ * MAX_DELAY data type. ++ */ ++typedef uint16 max_delay; ++ ++/** ++ * BITRATE data type. ++ */ ++typedef uint32 bitrate; ++ ++/** ++ * JITTER data type. ++ */ ++typedef uint16 jitter; ++ ++/** ++ * PKT_LOSS_RATE data type. ++ */ ++typedef uint16 pkt_loss_rate; ++ ++/** ++ * COS data type. ++ */ ++typedef uint16 cos; ++ ++/** ++ * DROP_ELIGIBILITY data type. ++ */ ++typedef bool drop_eligibility; ++ ++/** ++ * MULTICAST_ENABLE data type. ++ */ ++typedef bool multicast_enable; ++ ++/** ++ * JUMBO_ENABLE data type. ++ */ ++typedef bool jumbo_enable; ++ ++/** ++ * PORT data type. ++ */ ++typedef uint16 port; ++ ++/** ++ * MARK data type. ++ */ ++typedef uint8 mark; ++ ++/** ++ * FLOW_ID data type. ++ */ ++typedef uint32 flow_id; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * PROTO data type enumeration. ++ */ ++enum proto_enum { ++ proto_tcp = 0, /**< TCP. */ ++ proto_udp = 1 /**< UDP. */ ++}; ++ ++/** ++ * PROTO data type. ++ */ ++typedef enumeration<proto_enum> proto; + /////////////////////////////////////////////////////////////////////////////// + /** + * OP_MODE data type enumeration. +@@ -181,6 +249,248 @@ struct threshold { + uint16 threshold_val; /**< Threshold value. */ + enumeration<type_ip_enum> threshold_x_dir; /**< Threshold Direction. */ + }; ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * LINK_TYPE data type enumeration. ++ */ ++enum link_type_enum { ++ link_type_gsm = 1, /**< Wireless - GSM. */ ++ link_type_gprs = 2, /**< Wireless - GPRS. */ ++ link_type_edge = 3, /**< Wireless - EDGE. */ ++ ++ link_type_ethernet = 15, /**< Ethernet. */ ++ ++ link_type_wireless_other = 18, /**< Wireless - Other. */ ++ link_type_802_11 = 19, /**< Wireless - IEEE 802.11. */ ++ ++ link_type_cdma2000 = 22, /**< Wireless - CDMA-2000. */ ++ link_type_umts = 23, /**< Wireless - UMTS. */ ++ link_type_cdma2000_hrpd = 24, /**< Wireless - CDMA-2000-HRPD. */ ++ link_type_lte = 25, /**< Wireless - LTE. */ ++ ++ link_type_802_16 = 27, /**< Wireless - IEEE 802.16. */ ++ link_type_802_20 = 28, /**< Wireless - IEEE 802.20. */ ++ link_type_802_22 = 29 /**< Wireless - IEEE 802.22. */ ++}; ++ ++/** ++ * LINK_TYPE data type. ++ */ ++typedef enumeration<link_type_enum> link_type; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * LINK_ID data type. ++ */ ++struct link_id { ++ /** ++ * Construct an empty LINK_ID data type. ++ */ ++ link_id() : type(link_type_enum(0)) ++ { } ++ ++ /** ++ * Serialize/deserialize the LINK_ID data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & type; ++ ar & addr; ++ } ++ ++ /** ++ * Check if the LINK_ID data type is equal to another LINK_ID. ++ * ++ * @param other The LINK_ID to compare to. ++ * @return True if they are equal or false otherwise. ++ */ ++ bool operator==(const link_id& other) const ++ { ++ return ((type == other.type) && (addr == other.addr)); ++ } ++ ++ /** ++ * LINK_ID data type output. ++ * ++ * @param out ostream. ++ * @param addr LINK_ID data type. ++ * @return ostream reference. ++ */ friend std::ostream& operator<<(std::ostream& out, const link_id& lid) ++ { ++ out << "link id " << lid.type << " " << lid.addr; ++ return out; ++ } ++ ++ ++ link_type type; /**< Link address type. */ ++ link_addr addr; /**< Link address. */ ++}; ++ ++/** ++ * LIST(LINK_ID) data type. ++ */ ++typedef std::vector<link_id> link_id_list; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * IP_TUPLE data type. ++ */ ++struct ip_tuple { ++ /** ++ * Serialize/deserialize the IP_TUPLE data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & ip; ++ ar & port_val; ++ } ++ ++ ip_addr ip; /**< IP address.*/ ++ port port_val; /**< Port. */ ++}; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * Auxiliar QOS data type. ++ */ ++struct qos_sequence { ++ /** ++ * Serialize/deserialize the auxiliar QOS data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & delay_val; ++ ar & bitrate_val; ++ ar & jitter_val; ++ ar & pkt_loss_val; ++ } ++ ++ max_delay delay_val; /**< Maximum delay. */ ++ bitrate bitrate_val; /**< Bitrate. */ ++ jitter jitter_val; /**< Jitter. */ ++ pkt_loss_rate pkt_loss_val; /**< Packet loss. */ ++}; ++ ++/** ++ * QOS data type. ++ */ ++struct qos { ++ /** ++ * Serialize/deserialize the QOS data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & value; ++ } ++ ++ boost::variant<qos_sequence, cos> value; /**< QoS value. */ ++}; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * Auxiliar MARK_QoS data type. ++ */ ++struct mark_qos_sequence { ++ /** ++ * Serialize/deserialize the auxiliar FLOW_ATTRIBUTE data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & mark_val; ++ ar & qos_val; ++ } ++ ++ mark mark_val; /**< Mark value. */ ++ qos qos_val; /**< QoS value. */ ++}; ++ ++/** ++ * Auxiliar MARK_DROP_ELIG data type. ++ */ ++struct mark_drop_elig_sequence { ++ /** ++ * Serialize/deserialize the auxiliar FLOW_ATTRIBUTE data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & mark_val; ++ ar & drop_val; ++ } ++ ++ mark mark_val; /**< Mark value. */ ++ drop_eligibility drop_val; /**< Drop eligibility value. */ ++}; ++ ++/** ++ * FLOW_ATTRIBUTE data type. ++ */ ++struct flow_attribute { ++ /** ++ * Serialize/deserialize the FLOW_ATTRIBUTE data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & id; ++ ar & multicast; ++ ar & qos_val; ++ ar & drop_elig_val; ++ } ++ ++ flow_id id; /**< Flow ID. */ ++ boost::variant<null, multicast_enable> multicast; /**< Multicast enable. */ ++ boost::variant<null, mark_qos_sequence> qos_val; /**< Mask/QoS value. */ ++ boost::variant<null, mark_drop_elig_sequence> drop_elig_val; /**< Mask/drop eligibility value. */ ++}; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** ++ * RESOURCE_DESC data type. ++ */ ++struct resource_desc { ++ /** ++ * Serialize/deserialize the RESOURCE_DESC data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & lid; ++ ar & fid; ++ ar & data_rate; ++ ar & qos_val; ++ ar & jumbo; ++ ar & multicast; ++ } ++ ++ link_id lid; /**< Link ID. */ ++ flow_id fid; /**< Flow ID. */ ++ boost::variant<null, uint32> data_rate; /**< Link data rate. */ ++ boost::variant<null, qos> qos_val; /**< QoS value. */ ++ boost::variant<null, jumbo_enable> jumbo; /**< Jumbo enable. */ ++ boost::variant<null, multicast_enable> multicast; /**< Multicast enable. */ ++}; + + /////////////////////////////////////////////////////////////////////////////// + /** +@@ -208,6 +518,9 @@ enum link_ac_type_enum { + link_ac_type_low_power = 2, /**< Link low power. */ + link_ac_type_power_down = 3, /**< Link power down. */ + link_ac_type_power_up = 4, /**< Link power up. */ ++ link_ac_type_flow_attr = 5, /**< Flow Attribute. */ ++ link_ac_type_link_activate_resources = 6, /**< Link Activate Resources. */ ++ link_ac_type_link_deactivate_resources = 7, /**< Link Deactivate Resources. */ + }; + + /** +@@ -232,11 +545,31 @@ typedef bitmap<8, link_ac_attr_enum> link_ac_attr; + + /////////////////////////////////////////////////////////////////////////////// + /** ++ * LINK_AC_PARAM data type. ++ */ ++struct link_ac_param { ++ /** ++ * Serialize/deserialize the LINK_AC_PARAM data type. ++ * ++ * @param ar The archive to/from where serialize/deserialize the data type. ++ */ ++ template<class ArchiveT> ++ void serialize(ArchiveT& ar) ++ { ++ ar & param; ++ } ++ ++ boost::variant<null, flow_attribute, resource_desc> param; /**< Link action parameter.*/ ++}; ++ ++/////////////////////////////////////////////////////////////////////////////// ++/** + * LINK_ACTION data type. + */ + struct link_action { + link_ac_type type; /**< Link action type. */ + link_ac_attr attr; /**< Link action attribute. */ ++ link_ac_param param;/**< Link action parameter. */ + + /** + * Serialize/deserialize the LINK_ACTION data type. +@@ -248,6 +581,7 @@ struct link_action { + {; + ar & type; + ar & attr; ++ ar & param; + } + }; + +@@ -266,33 +600,6 @@ enum th_action_enum { + */ + typedef enumeration<th_action_enum> th_action; + +-/////////////////////////////////////////////////////////////////////////////// +-/** +- * LINK_TYPE data type enumeration. +- */ +-enum link_type_enum { +- link_type_gsm = 1, /**< Wireless - GSM. */ +- link_type_gprs = 2, /**< Wireless - GPRS. */ +- link_type_edge = 3, /**< Wireless - EDGE. */ +- +- link_type_ethernet = 15, /**< Ethernet. */ +- +- link_type_wireless_other = 18, /**< Wireless - Other. */ +- link_type_802_11 = 19, /**< Wireless - IEEE 802.11. */ +- +- link_type_cdma2000 = 22, /**< Wireless - CDMA-2000. */ +- link_type_umts = 23, /**< Wireless - UMTS. */ +- link_type_cdma2000_hrpd = 24, /**< Wireless - CDMA-2000-HRPD. */ +- +- link_type_802_16 = 27, /**< Wireless - IEEE 802.16. */ +- link_type_802_20 = 28, /**< Wireless - IEEE 802.20. */ +- link_type_802_22 = 29 /**< Wireless - IEEE 802.22. */ +-}; +- +-/** +- * LINK_TYPE data type. +- */ +-typedef enumeration<link_type_enum> link_type; + + /////////////////////////////////////////////////////////////////////////////// + /** +@@ -505,46 +812,29 @@ typedef enumeration<link_param_802_22_enum> link_param_802_22; + + /////////////////////////////////////////////////////////////////////////////// + /** +- * LINK_ID data type. +- */ +-struct link_id { +- /** +- * Construct an empty LINK_ID data type. +- */ +- link_id() : type(link_type_enum(0)) +- { } +- +- /** +- * Serialize/deserialize the LINK_ID data type. +- * +- * @param ar The archive to/from where serialize/deserialize the data type. +- */ +- template<class ArchiveT> +- void serialize(ArchiveT& ar) +- { +- ar & type; +- ar & addr; +- } +- +- /** +- * Check if the LINK_ID data type is equal to another LINK_ID. +- * +- * @param other The LINK_ID to compare to. +- * @return True if they are equal or false otherwise. +- */ +- bool operator==(const link_id& other) const +- { +- return ((type == other.type) && (addr == other.addr)); +- } +- +- link_type type; /**< Link address type. */ +- link_addr addr; /**< Link address. */ ++ * LINK_PARAM_lte data type enumeration. ++ */ ++enum link_param_lte_enum { ++ link_param_lte_rsrp = 0, /**< RSSI. */ ++ link_param_lte_rsrq = 1, /**< No QoS resource available. */ ++ link_param_lte_cqi = 2, /**< Multicast packet loss rate.*/ ++ link_param_lte_bandwidth = 3, /**< System Load. */ ++ link_param_lte_pkt_delay = 4, /**< Number of registered users. */ ++ link_param_lte_pkt_loss = 5, /**< Number of active users. */ ++ link_param_lte_l2_buffer = 6, /**< Congestion windows of users. */ ++ link_param_lte_MN_cap = 7, /**< Congestion windows of users. */ ++ link_param_lte_embms = 8, /**< Congestion windows of users. */ ++ link_param_lte_jumbo_feasibility = 9, /**< Congestion windows of users. */ ++ link_param_lte_jumbo_setup = 10, /**< Congestion windows of users. */ ++ link_param_lte_active_embms = 11, /**< Transmission rate of users. */ ++ link_param_lte_link_congestion = 12, /**< Link congestion. */ + }; + ++/////////////////////////////////////////////////////////////////////////////// + /** +- * LIST(LINK_ID) data type. ++ * LINK_PARAM_LTE data type. + */ +-typedef std::vector<link_id> link_id_list; ++typedef enumeration<link_param_lte_enum> link_param_lte; + + /////////////////////////////////////////////////////////////////////////////// + /** +@@ -746,7 +1036,8 @@ typedef boost::variant<link_param_gen, + link_param_hrpd, + link_param_802_16, + link_param_802_20, +- link_param_802_22> link_param_type; ++ link_param_802_22, ++ link_param_lte> link_param_type; + + /** + * LIST(LINK_PARAM_TYPE) data type. +diff --git a/lib/odtone/CMakeLists.txt b/lib/odtone/CMakeLists.txt +index 0023be9..8cd9f73 100644 +--- a/lib/odtone/CMakeLists.txt ++++ b/lib/odtone/CMakeLists.txt +@@ -10,6 +10,7 @@ set(libodtone_SOVERSION "${libodtone_MAJOR_VERSION}.${libodtone_MINOR_VERSION}") + + set(libodtone_SRC + strutil.cpp ++conf.cpp + mih/types/address.cpp + mih/config.cpp + mih/archive.cpp +diff --git a/lib/odtone/Jamfile b/lib/odtone/Jamfile +index 341d777..3f37d29 100644 +--- a/lib/odtone/Jamfile ++++ b/lib/odtone/Jamfile +@@ -32,7 +32,8 @@ odtone.explicit-alias source-list + ; + + lib odtone +- : debug.cpp ++ : conf.cpp ++ debug.cpp + strutil.cpp + logger.cpp + win32.cpp +diff --git a/lib/odtone/conf.cpp b/lib/odtone/conf.cpp +new file mode 100644 +index 0000000..e99fd91 +--- /dev/null ++++ b/lib/odtone/conf.cpp +@@ -0,0 +1,43 @@ ++//======================================================================================================= ++// Brief : Configuration DSL ++// Authors : Bruno Santos <bsantos@av.it.pt> ++// ------------------------------------------------------------------------------------------------------ ++// ODTONE - Open Dot Twenty One ++// ++// Copyright (C) 2009-2013 Universidade Aveiro ++// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro ++// ++// This software is distributed under a license. The full license ++// agreement can be found in the file LICENSE in this distribution. ++// This software may not be copied, modified, sold or distributed ++// other than expressed in the named license agreement. ++// ++// This software is distributed without any warranty. ++//======================================================================================================= ++ ++#include <odtone/conf.hpp> ++#include <boost/spirit/include/qi_parse_attr.hpp> ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++namespace odtone { namespace conf { ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++bool exec(std::string::const_iterator& begin, std::string::const_iterator end, functions const& cm) ++{ ++ skipper_grammar<std::string::const_iterator> skipper; ++ parser_grammar<std::string::const_iterator> parser(cm); ++ std::string::const_iterator pos = begin; ++ ++ while (begin != end) { ++ if (!qi::phrase_parse(begin, end, parser, skipper) || pos == begin) ++ return false; ++ pos = begin; ++ } ++ ++ return true; ++} ++ ++///////////////////////////////////////////////////////////////////////////////////////////////////////// ++} /* namespace conf */ } /* namespace odtone */ ++ ++// EOF ////////////////////////////////////////////////////////////////////////////////////////////////// +diff --git a/src/mihf/command_service.cpp b/src/mihf/command_service.cpp +index 84557a0..516801b 100644 +--- a/src/mihf/command_service.cpp ++++ b/src/mihf/command_service.cpp +@@ -614,6 +614,7 @@ bool command_service::link_actions_request(meta_message_ptr &in, + // For each Link_ID in request message + std::vector<mih::link_action_req>::iterator lar; + for(lar = lal.begin(); lar != lal.end(); lar++) { ++ std::cout<<"ADDR in MIHF "<<(*lar).id.addr<<std::endl; + out->destination(mih::id(_link_abook.search_interface((*lar).id.type, (*lar).id.addr))); + // If the Link SAP it is known send message + if (out->destination().to_string().compare("") != 0) { +diff --git a/src/mihf/dst_transaction.cpp b/src/mihf/dst_transaction.cpp +index 692f1ea..f6d7d86 100644 +--- a/src/mihf/dst_transaction.cpp ++++ b/src/mihf/dst_transaction.cpp +@@ -16,6 +16,7 @@ + //============================================================================== + + /////////////////////////////////////////////////////////////////////////////// ++#include "log.hpp" + #include "dst_transaction.hpp" + #include "utils.hpp" + /////////////////////////////////////////////////////////////////////////////// +@@ -50,6 +51,7 @@ void dst_transaction_t::run() + + _init_lbl_: + { ++ ODTONE_LOG(1, "(dst_transaction_t) init tid ", in->tid()); + transaction_status = ONGOING; + opcode = in->opcode(); + tid = in->tid(); +@@ -75,6 +77,7 @@ void dst_transaction_t::run() + + _wait_response_prm_lbl_: + { ++ ODTONE_LOG(1, "(dst_transaction_t) wait response tid ", tid); + state = DST_WAIT_RESPONSE_PRM; + + if (transaction_stop_when == 0) +@@ -88,6 +91,7 @@ void dst_transaction_t::run() + + _send_response_begin_lbl_: + { ++ ODTONE_LOG(1, "(dst_transaction_t) send response begin tid ", tid); + state = DST_SEND_RESPONSE; + + start_ack_requestor = out->ackreq(); +@@ -100,6 +104,7 @@ void dst_transaction_t::run() + + _send_response_lbl_: + { ++ ODTONE_LOG(1, "(dst_transaction_t) send response tid ", tid); + if (!start_ack_requestor || ack_requestor_status == SUCCESS) + goto _success_lbl_; + else if (ack_requestor_status == FAILURE) +@@ -110,6 +115,7 @@ void dst_transaction_t::run() + + _failure_lbl_: + { ++ ODTONE_LOG(1, "(dst_transaction_t) failure tid ", tid); + state = DST_FAILURE; + transaction_status = FAILURE; + +@@ -118,6 +124,7 @@ void dst_transaction_t::run() + + _success_lbl_: + { ++ ODTONE_LOG(1, "(dst_transaction_t) success tid ", tid); + state = DST_SUCCESS; + transaction_status = SUCCESS; + } +diff --git a/src/mihf/event_service.cpp b/src/mihf/event_service.cpp +index fc4d08d..9c021ea 100644 +--- a/src/mihf/event_service.cpp ++++ b/src/mihf/event_service.cpp +@@ -40,6 +40,9 @@ extern odtone::uint16 kConf_MIHF_Link_Delete_Value; + + namespace odtone { namespace mihf { + ++ ++ ++ + /** + * Construct the event service. + * +@@ -306,7 +309,9 @@ bool event_service::event_subscribe_response(meta_message_ptr &in, + & mih::tlv_status(st) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); +- ++ ++ ++ std::cout<< "(mies) The link received is "<<link<<std::endl; + // add a subscription + if (st == mih::status_success) { + st = subscribe(mih::id(in->destination().to_string()), link, events.get()); +@@ -659,6 +664,9 @@ void event_service::msg_forward(meta_message_ptr &msg, + for(it = _event_subscriptions.begin(); + it != _event_subscriptions.end(); + it++, i++) { ++ ODTONE_LOG(3, "(mies) msg_forward() comparing event ", event, " with event_subscription it->event ", it->event); ++ ODTONE_LOG(3, "(mies) msg_forward() comparing link_tuple_id.type ", li.type, " with event_subscription it->link.type ", it->link.type); ++ ODTONE_LOG(3, "(mies) msg_forward() comparing link_tuple_id.addr ", li.addr, " with event_subscription it->link.addr ", it->link.addr); + if ((it->event == event) && (it->link == li)) { + ODTONE_LOG(3, "(mies) found registration of user: ", + it->user, " for event type ", event); +@@ -666,6 +674,7 @@ void event_service::msg_forward(meta_message_ptr &msg, + _transmit(msg); + } + } ++ ODTONE_LOG(3, "(mies) msg_forward() end"); + } + + +@@ -680,6 +689,7 @@ void event_service::link_event_forward(meta_message_ptr &msg, + mih::mih_evt_list_enum event) + { + mih::link_tuple_id li; ++ ODTONE_LOG(1, "(mies) link_event_forward()"); + *msg >> mih::indication() + & mih::tlv_link_identifier(li); + +@@ -699,8 +709,10 @@ bool event_service::link_up_indication(meta_message_ptr &in, meta_message_ptr &o + ODTONE_LOG(1, "(mies) received Link_Up.indication from ", + in->source().to_string()); + +- if(in->is_local()) ++ if(in->is_local()) { ++ ODTONE_LOG(1, "(mies) link_up_indication() reset ", in->source().to_string() ," in link_abook"); + _link_abook.reset(in->source().to_string()); ++ } + + link_event_forward(in, mih::mih_evt_link_up); + +diff --git a/src/mihf/link_book.cpp b/src/mihf/link_book.cpp +index 89c7ef1..271112e 100644 +--- a/src/mihf/link_book.cpp ++++ b/src/mihf/link_book.cpp +@@ -23,6 +23,38 @@ + + namespace odtone { namespace mihf { + ++//----------------------------------------------------------------------------- ++static std::string evt2string(mih::link_evt_list evtP){ ++//----------------------------------------------------------------------------- ++ std::string s=std::string(" "); ++ if(evtP.get(mih::evt_link_detected)) s += "DETECTED "; ++ if(evtP.get(mih::evt_link_up)) s += "UP "; ++ if(evtP.get(mih::evt_link_down)) s += "DOWN "; ++ if(evtP.get(mih::evt_link_parameters_report)) s += "PARAMETERS_REPORT "; ++ if(evtP.get(mih::evt_link_going_down)) s += "GOING_DOWN "; ++ if(evtP.get(mih::evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; ++ if(evtP.get(mih::evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; ++ if(evtP.get(mih::evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; ++ return s; ++} ++enum link_cmd_list_enum { ++ cmd_link_event_subscribe = 1, /**< Event subscribe. */ ++ cmd_link_event_unsubscribe = 2, /**< Event unsubscribe. */ ++ cmd_link_get_parameters = 3, /**< Get parameters. */ ++ cmd_link_configure_thresholds = 4, /**< Configure thresholds. */ ++ cmd_link_action = 5, /**< Action. */ ++}; ++//----------------------------------------------------------------------------- ++static std::string cmd2string(mih::link_cmd_list cmdP){ ++//----------------------------------------------------------------------------- ++ std::string s=std::string(" "); ++ if(cmdP.get(mih::cmd_link_event_subscribe)) s += "EVENT_SUBSCRIBE "; ++ if(cmdP.get(mih::cmd_link_event_unsubscribe)) s += "EVENT_UNSUBSCRIBE"; ++ if(cmdP.get(mih::cmd_link_get_parameters)) s += "GET_PARAMETERS "; ++ if(cmdP.get(mih::cmd_link_configure_thresholds)) s += "CONFIGURE_THRESHOLDS "; ++ if(cmdP.get(mih::cmd_link_action)) s += "ACTION "; ++ return s; ++} + /** + * Add a new Link SAP entry in the link book. + * +@@ -47,7 +79,7 @@ void link_book::add(const mih::octet_string &id, + a.status = true; + + _lbook[id] = a; +- ODTONE_LOG(4, "(link_book) added: ", id, " ", ip, " ", port); ++ ODTONE_LOG(4, "(link_book) added: ", id, " ", ip, " ", port, " ", link_id); + } + + /** +@@ -102,8 +134,11 @@ void link_book::update_capabilities(const mih::octet_string &id, + it = _lbook.find(id); + + if (it != _lbook.end()) { ++ ODTONE_LOG(4, "(link_book) update_capabilities() FOUND: ", id, " ", evt2string(event_list), " ", cmd2string(cmd_list)); + it->second.event_list = event_list; + it->second.cmd_list = cmd_list; ++ } else { ++ ODTONE_LOG(4, "(link_book) update_capabilities() NOT FOUND: ", id, " ", evt2string(event_list), " ", cmd2string(cmd_list)); + } + } + +@@ -131,8 +166,10 @@ void link_book::inactive(mih::octet_string &id) + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + +- if (it != _lbook.end()) ++ if (it != _lbook.end()) { ++ ODTONE_LOG(4, "(link_book) inactive() : ", id); + it->second.status = false; ++ } + } + + /** +@@ -183,10 +220,21 @@ const mih::octet_string link_book::search_interface(mih::link_type lt, mih::link + boost::mutex::scoped_lock lock(_mutex); + + mih::octet_string id; +- for(std::map<mih::octet_string, link_entry>::iterator it = _lbook.begin(); it != _lbook.end(); it++) { ++ std::map<mih::octet_string, link_entry>::iterator it; ++ for(it = _lbook.begin(); it != _lbook.end(); it++) { ++ ++ ODTONE_LOG(4, "(link_book) ID LINK TYPE : ", it->first, " ADDRESS ", it->second.link_id.addr, " STATUS", it->second.status); ++ } ++ ++ for(/*std::map<mih::octet_string, link_entry>::iterator*/ it = _lbook.begin(); it != _lbook.end(); it++) { ++ ++ ODTONE_LOG(4, "(link_book) search 1st step ID LINK TYPE comparison type:",it->second.link_id.type, "LT", lt); + if(it->second.link_id.type == lt) { ++ ODTONE_LOG(4, "(link_book) search 2nd step ID LINK TYPE comparison addr:",it->second.link_id.addr, "LA", la); + if(it->second.link_id.addr == la) { +- if(it->second.status) { ++ ODTONE_LOG(4, "(link_book) search 3rd step ID LINK TYPE check status:",it->second.status); ++ if(it->second.status) { ++ ODTONE_LOG(4, "(link_book) search 3rd step ID LINK TYPE : ", it->first, " ADDRESS ", it->second.link_id.addr, " STATUS", it->second.status); + id = it->first; + break; + } +@@ -194,6 +242,7 @@ const mih::octet_string link_book::search_interface(mih::link_type lt, mih::link + } + } + ++ ODTONE_LOG(4, "(link_book) search_interface() : ", lt, " ", la, " Found:", id); + return id; + } + +@@ -214,6 +263,7 @@ uint16 link_book::fail(const mih::octet_string &id) + boost::throw_exception(unknown_link_sap()); + + (it->second.fail)++; ++ ODTONE_LOG(4, "(link_book) fail() : ", id ," num fails:",it->second.fail); + return it->second.fail; + } + +@@ -232,6 +282,7 @@ void link_book::reset(const mih::octet_string &id) + if (it == _lbook.end()) + boost::throw_exception(unknown_link_sap()); + ++ ODTONE_LOG(4, "(link_book) reset() : ", id); + it->second.fail = 0; + it->second.status = true; + } +diff --git a/src/mihf/main.cpp b/src/mihf/main.cpp +index 244d38e..46a9db9 100644 +--- a/src/mihf/main.cpp ++++ b/src/mihf/main.cpp +@@ -289,6 +289,7 @@ void set_links(mih::octet_string &list, link_book &lbook) + enum_map["CDMA2000"] = odtone::mih::link_type_cdma2000; + enum_map["UMTS"] = odtone::mih::link_type_umts; + enum_map["CDMA2000-HRPD"] = odtone::mih::link_type_cdma2000_hrpd; ++ enum_map["LTE"] = odtone::mih::link_type_lte; + enum_map["802_16"] = odtone::mih::link_type_802_16; + enum_map["802_20"] = odtone::mih::link_type_802_20; + enum_map["802_22"] = odtone::mih::link_type_802_22; +@@ -323,7 +324,35 @@ void set_links(mih::octet_string &list, link_book &lbook) + throw "not supported yet"; + break; + ++ case 25: { ++ ++it; ++ mih::octet_string plmn = *it; ++ ++it; ++ mih::octet_string cell_id = *it; ++ ++ char_separator<char> plmn_sep(":"); ++ tokenizer< char_separator<char> > list_tokens(plmn, plmn_sep); ++ ++ mih::l2_3gpp_3g_cell_id lte; ++ uint8 pos = 0; ++ BOOST_FOREACH(mih::octet_string str, list_tokens) { ++ uint8 byte = 0; ++ std::istringstream iss(str); ++ iss >> std::hex >> byte; ++ ++ lte.plmn_id[pos] = byte % 0x100; ++ ++pos; ++ } ++ ++ std::istringstream iss(port); ++ if ((iss >> lte._cell_id).fail()) { ++ throw "invalid cell_id"; ++ } ++ ++ lid.addr = lte; ++ } break; + default: { ++ ODTONE_LOG(0, "Error: Invalid technology! Aborting...") ; + throw "invalid technology"; + } break; + } +@@ -605,7 +634,7 @@ int main(int argc, char **argv) + (kConf_MIHF_Users_List, po::value<std::string>()->default_value(""), "List of local MIH-Users") + (kConf_MIHF_Links_List, po::value<std::string>()->default_value(""), "List of local Links SAPs") + (kConf_MIHF_Transport_List, po::value<std::string>()->default_value("udp"), "List of supported transport protocols") +- (kConf_MIHF_Link_Response_Time, po::value<uint16>()->default_value(3000), "Link SAP response time (milliseconds)") ++ (kConf_MIHF_Link_Response_Time, po::value<uint16>()->default_value(7000), "Link SAP response time (milliseconds)") + (kConf_MIHF_Link_Delete, po::value<uint16>()->default_value(2), "Link SAP response fails to forget") + (kConf_MIHF_Discover, po::value<std::string>()->default_value(""), "MIHF Discovery Mechanisms Order") + (kConf_MIHF_Multicast, "Allows multicast messages") +@@ -716,6 +745,7 @@ int main(int argc, char **argv) + mies_register_callbacks(mies); + mics_register_callbacks(mics); + miis_register_callbacks(miis); ++ std::cout << "Boot complete" << std::endl << std::flush; + + io.run(); + +diff --git a/src/mihf/service_access_controller.cpp b/src/mihf/service_access_controller.cpp +index 7660d50..2528fca 100644 +--- a/src/mihf/service_access_controller.cpp ++++ b/src/mihf/service_access_controller.cpp +@@ -44,6 +44,7 @@ static std::map<uint, handler_t> _callbacks; /**< Callback map of the supported + */ + void sac_register_callback(uint mid, handler_t func) + { ++ ODTONE_LOG(1, "(sac) sac_register_callback(): mid ", mid, " handler ", func); + _callbacks[mid] = func; + } + +@@ -85,8 +86,11 @@ void sac_dispatch::operator()(meta_message_ptr& in) + // send response if it was generated + try { + +- if (process_message(in, out)) ++ if (process_message(in, out)) { + _transmit(out); ++ } else { ++ ODTONE_LOG(1, "sac_dispatch::operator() Message not transmitted out."); ++ } + + } catch(mih::bad_tlv) { + ODTONE_LOG(1, "Discarding malformed message."); +diff --git a/src/mihf/service_management.cpp b/src/mihf/service_management.cpp +index 6a75f82..9c31b2c 100644 +--- a/src/mihf/service_management.cpp ++++ b/src/mihf/service_management.cpp +@@ -638,6 +638,7 @@ bool service_management::link_register_indication(meta_message_ptr &in, + *in >> odtone::mih::indication() + & odtone::mih::tlv_interface_type_addr(link_id); + ++ ODTONE_LOG(4, "((mism) received Link_Register.indication: link_id", link_id); + // Add Link SAP to the list of known Link SAPs + _link_abook.add(in->source().to_string(), in->ip(), in->port(), link_id); + +diff --git a/src/mihf/src_transaction.cpp b/src/mihf/src_transaction.cpp +index af6a468..ef132d8 100644 +--- a/src/mihf/src_transaction.cpp ++++ b/src/mihf/src_transaction.cpp +@@ -16,6 +16,7 @@ + //============================================================================== + + /////////////////////////////////////////////////////////////////////////////// ++#include "log.hpp" + #include "src_transaction.hpp" + #include "utils.hpp" + /////////////////////////////////////////////////////////////////////////////// +@@ -51,6 +52,7 @@ void src_transaction_t::run() + + _init_lbl_: + { ++ ODTONE_LOG(1, "(src_transaction_t) init tid ", out->tid()); + transaction_status = ONGOING; + response_received = false; + transaction_stop_when = 15; // FIXME: read from config +@@ -84,6 +86,7 @@ void src_transaction_t::run() + + _wait_ack_lbl_: + { ++ ODTONE_LOG(1, "(src_transaction_t) wait ack tid ", tid); + state = SRC_WAIT_ACK; + + if (ack_requestor_status == SUCCESS) +@@ -96,6 +99,7 @@ void src_transaction_t::run() + + _wait_response_msg_lbl_: + { ++ ODTONE_LOG(1, "(src_transaction_t) wait response tid ", tid); + state = SRC_WAIT_RESPONSE_MSG; + + if (transaction_stop_when == 0) { +@@ -111,6 +115,7 @@ void src_transaction_t::run() + + _process_msg_lbl_: + { ++ ODTONE_LOG(1, "(src_transaction_t) process msg tid ", tid); + state = SRC_PROCESS_MSG; + + start_ack_responder = in->ackreq(); +@@ -132,6 +137,7 @@ void src_transaction_t::run() + + _failure_lbl_: + { ++ ODTONE_LOG(1, "(src_transaction_t) failure tid ", tid); + transaction_status = FAILURE; + state = SRC_FAILURE; + return; +@@ -139,6 +145,7 @@ void src_transaction_t::run() + + _success_lbl_: + { ++ ODTONE_LOG(1, "(src_transaction_t) success tid ", tid); + transaction_status = SUCCESS; + state = SRC_SUCCESS; + } +diff --git a/src/mihf/transaction_pool.cpp b/src/mihf/transaction_pool.cpp +index f60d125..204e5a1 100644 +--- a/src/mihf/transaction_pool.cpp ++++ b/src/mihf/transaction_pool.cpp +@@ -16,6 +16,7 @@ + //============================================================================== + + /////////////////////////////////////////////////////////////////////////////// ++#include "log.hpp" + #include "transaction_pool.hpp" + /////////////////////////////////////////////////////////////////////////////// + +@@ -58,6 +59,7 @@ void transaction_pool::dec(Set &set, + + for(it = set.begin(); it != set.end(); it++) { + (*it)->transaction_stop_when--; ++ ODTONE_LOG(1, "(transaction_pool) decrementing tid ", (*it)->tid, " counter down to ", (*it)->transaction_stop_when); + if ((*it)->transaction_stop_when == 0) { + (*it)->run(); + +@@ -88,6 +90,7 @@ void transaction_pool::dec(Set &set, + */ + void transaction_pool::tick() + { ++ //ODTONE_LOG(1, "(transaction_pool) tick"); + _timer.expires_from_now(boost::posix_time::seconds(1)); + _timer.async_wait(boost::bind(&transaction_pool::tick, this)); + +@@ -107,6 +110,7 @@ void transaction_pool::tick() + */ + void transaction_pool::add(src_transaction_ptr &t) + { ++ ODTONE_LOG(1, "(transaction_pool) add src tid ", t->tid); + boost::mutex::scoped_lock lock(_src_mutex); + _src.insert(t); + } +@@ -118,6 +122,7 @@ void transaction_pool::add(src_transaction_ptr &t) + */ + void transaction_pool::add(dst_transaction_ptr &t) + { ++ ODTONE_LOG(1, "(transaction_pool) add dst tid ", t->tid); + boost::mutex::scoped_lock lock(_dst_mutex); + _dst.insert(t); + } +@@ -129,6 +134,7 @@ void transaction_pool::add(dst_transaction_ptr &t) + */ + void transaction_pool::del(const src_transaction_ptr &t) + { ++ ODTONE_LOG(1, "(transaction_pool) del src tid ", t->tid); + boost::mutex::scoped_lock lock(_src_mutex); + _src.erase(t); + } +@@ -140,6 +146,7 @@ void transaction_pool::del(const src_transaction_ptr &t) + */ + void transaction_pool::del(const dst_transaction_ptr &t) + { ++ ODTONE_LOG(1, "(transaction_pool) del dst tid ", t->tid); + boost::mutex::scoped_lock lock(_dst_mutex); + _dst.erase(t); + } +-- +1.7.9.5 + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/CMakeLists.txt b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/CMakeLists.txt new file mode 100644 index 0000000000000000000000000000000000000000..8cd9f732aac07a11c12f78b5de0933b224e8569e --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/CMakeLists.txt @@ -0,0 +1,74 @@ +project(libodtone) + +find_package(Boost 1.46 COMPONENTS system program_options thread REQUIRED) + +set(libodtone_MAJOR_VERSION ${odtone_MAJOR_VERSION}) +set(libodtone_MINOR_VERSION ${odtone_MINOR_VERSION}) +set(libodtone_MICRO_VERSION ${odtone_MICRO_VERSION}) +set(libodtone_VERSION "${libodtone_MAJOR_VERSION}.${libodtone_MINOR_VERSION}.${libodtone_MICRO_VERSION}") +set(libodtone_SOVERSION "${libodtone_MAJOR_VERSION}.${libodtone_MINOR_VERSION}") + +set(libodtone_SRC +strutil.cpp +conf.cpp +mih/types/address.cpp +mih/config.cpp +mih/archive.cpp +mih/message.cpp +net/dns/message.cpp +net/dns/resolver.cpp +net/dns/utils.cpp +net/link/address_mac.cpp +net/ip/icmp/icmp_parser.cpp +net/ip/prefix.cpp +debug.cpp +sap/link.cpp +sap/sap.cpp +sap/user.cpp +logger.cpp +win32.cpp +) + +if(MSVC) + LIST(APPEND libodtone_SRC debug_win32.cpp) +elseif(UNIX) + LIST(APPEND libodtone_SRC debug_linux.cpp) +endif() + +set(libodtone_base_FILE_HEADERS +../../inc/odtone/base.hpp +../../inc/odtone/bind_rv.hpp +../../inc/odtone/buffer.hpp +../../inc/odtone/cast.hpp +../../inc/odtone/conf.hpp +../../inc/odtone/debug.hpp +../../inc/odtone/exception.hpp +../../inc/odtone/list_node.hpp +../../inc/odtone/logger.hpp +../../inc/odtone/namespace.hpp +../../inc/odtone/random.hpp +../../inc/odtone/string.hpp +../../inc/odtone/strutil.hpp +) + +set(libodtone_mih_DIR_HEADERS ../../inc/odtone/mih) +set(libodtone_sap_DIR_HEADERS ../../inc/odtone/sap) +set(libodtone_net_DIR_HEADERS ../../inc/odtone/net) + +include_directories(${INCLUDE_DIRECTORIES} ${Boost_INCLUDE_DIRS} "../../inc") +add_library(libodtone SHARED ${libodtone_SRC}) +target_link_libraries(libodtone ${Boost_LIBRARIES}) +set_target_properties(libodtone PROPERTIES OUTPUT_NAME "odtone" + VERSION ${libodtone_VERSION} + SOVERSION ${libodtone_SOVERSION} + DEFINE_SYMBOL LIBODTONE_EXPORTS) + +# install libodtone +install(FILES ${libodtone_base_FILE_HEADERS} DESTINATION include/odtone) +install(DIRECTORY ${libodtone_mih_DIR_HEADERS} DESTINATION include/odtone) +install(DIRECTORY ${libodtone_sap_DIR_HEADERS} DESTINATION include/odtone) +install(DIRECTORY ${libodtone_net_DIR_HEADERS} DESTINATION include/odtone) +install(TARGETS libodtone EXPORT odtone + LIBRARY DESTINATION "${LIB_INSTALL_DIR}" + ARCHIVE DESTINATION "${LIB_INSTALL_DIR}" + RUNTIME DESTINATION bin) diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/Jamfile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/Jamfile new file mode 100644 index 0000000000000000000000000000000000000000..3f37d29a7bb62e78884aa6e52d542a27b81c6596 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/Jamfile @@ -0,0 +1,59 @@ +#============================================================================== +# Brief : ODTONE Base Library Project Build +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------ +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#============================================================================== + +project libodtone + ; + +odtone.runtime-lib pthread ; + +odtone.explicit-alias source-list + : debug_linux.cpp + : <target-os>linux + ; + +odtone.explicit-alias source-list + : debug_win32.cpp + : <target-os>windows + ; + +lib odtone + : conf.cpp + debug.cpp + strutil.cpp + logger.cpp + win32.cpp + source-list + sap/sap.cpp + sap/user.cpp + sap/link.cpp + mih/archive.cpp + mih/config.cpp + mih/message.cpp + mih/types/address.cpp + net/link/address_mac.cpp + net/ip/prefix.cpp + net/ip/icmp/icmp_parser.cpp + net/dns/resolver.cpp + net/dns/message.cpp + net/dns/utils.cpp + /boost//system + /boost//thread + /boost//program_options + : <c++-template-depth>1024 + <define>BOOST_ENABLE_ASSERT_HANDLER + ; diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/address.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/address.hpp new file mode 100644 index 0000000000000000000000000000000000000000..18fac4cbe55047b3b83fdd5527227ac72b4dca8c --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/address.hpp @@ -0,0 +1,561 @@ +//============================================================================= +// Brief : MIH Address Types +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#ifndef ODTONE_MIH_TYPES_ADDRESS__HPP_ +#define ODTONE_MIH_TYPES_ADDRESS__HPP_ + +/////////////////////////////////////////////////////////////////////////////// +#include <odtone/mih/types/base.hpp> +#include <iostream> + +/////////////////////////////////////////////////////////////////////////////// +namespace odtone { namespace mih { + +/////////////////////////////////////////////////////////////////////////////// +typedef uint32 cell_id; /**< CELL_ID data type. */ +typedef uint16 lac; /**< LAC data type. */ +typedef uint16 ci; /**< CI data type. */ + +/////////////////////////////////////////////////////////////////////////////// +/** + * TRANSPORT_ADDR data type. + */ +class transport_addr { +protected: + /** + * Construct a TRANSPORT_ADDR data type. + * + * @param type Transport address type. + */ + transport_addr(uint16 type) : _type(type) + { } + + /** + * Construct a TRANSPORT_ADDR data type. + * + * @param type Transport address type. + * @param raw Raw bytes of the transport address. + * @param len Size of the transport address raw bytes. + */ + transport_addr(uint16 type, const void* raw, size_t len) + : _type(type), _addr(reinterpret_cast<const char*>(raw), len) + { } + +public: + + /** + * Get the TRANSPORT_ADDR type. + * + * @return The transport address type. + */ + uint16 type() const + { + return _type; + } + + /** + * Set the TRANSPORT_ADDR type. + * + * @param type The transport address type. + */ + void type(const uint16 type) + { + _type = type; + } + + /** + * Get the pointer to an array of characters which contains the + * transport address. + * + * @return Pointer to an internal array containing the transport address. + */ + const void* get() const + { + return _addr.data(); + } + + /** + * Get the lenght of the transport address string. + * + * @return The lenght of the transport address string. + */ + size_t length() const + { + return _addr.length(); + } + + /** + * Serialize/deserialize the TRANSPORT_ADDR data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & _type; + ar & _addr; + } + +protected: + uint16 _type; /**< Transport address type. */ + octet_string _addr; /**< Transport address string. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * TRANSPORT_TYPE data type enumeration. + */ +enum transport_type_enum { + l2 = 0, /**< Layer 2 transport address. */ + l3_or_higher = 1, /**< Layer 3 or higher transport address. */ +}; + +/** + * TRANSPORT_TYPE data type. + */ +typedef enumeration<transport_type_enum> transport_type; + +/////////////////////////////////////////////////////////////////////////////// +/** + * MAC_ADDR data type. + */ +class mac_addr : public transport_addr { +public: + /** + * Construct an empty MAC_ADDR data type. + * @note Transport address type = 6 + */ + mac_addr() : transport_addr(6) + { } + + /** + * Construct an empty MAC_ADDR data type. + * @note Transport address type = 6 + * + * @param addr address MAC address string (format: XX:XX:XX:XX:XX:XX). + */ + explicit mac_addr(const octet_string& addr) : transport_addr(6) + { + this->address(addr); + } + + /** + * Construct an empty MAC_ADDR data type. + * @note Transport address type = 6 + * + * @param raw Raw bytes of the MAC address. + * @param len Lenght of the MAC address raw bytes. + */ + mac_addr(const void* raw, size_t len) : transport_addr(6, raw, len) + { } + + /** + * Get the MAC address string (format: XX:XX:XX:XX:XX:XX). + * + * @return The MAC address string. + */ + octet_string address() const; + + /** + * Set the MAC address. + * + * @param addr The MAC address string (format: XX:XX:XX:XX:XX:XX). + */ + void address(const octet_string& addr); + + /** + * MAC_ADDR data type output. + * + * @param out ostream. + * @param tp MAC_ADDR data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const mac_addr& tp) + { + out << "\ntype: " << tp.type(); + out << "\naddress: " << tp.address(); + + return out; + } + + /** + * Check if the MAC_ADDR is equal to another MAC_ADDR. + * + * @param other The MAC_ADDR to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const mac_addr& other) const + { + return ((type() == other.type()) && (address().compare(other.address()) == 0)); + } +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * L2_3GPP_2G_CELL_ID data type. + */ +struct l2_3gpp_2g_cell_id { + uint8 plmn_id[3]; /**< PLMN_ID data type. */ + lac _lac; /**< LAC data type. */ + ci _ci; /**< CI data type. */ + + /** + * Serialize/deserialize the L2_3GPP_2G_CELL_ID data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & plmn_id[0]; + ar & plmn_id[1]; + ar & plmn_id[2]; + ar & _lac; + ar & _ci; + } + + /** + * L2_3GPP_2G_CELL_ID data type output. + * + * @param out ostream. + * @param addr L2_3GPP_2G_CELL_ID data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp_2g_cell_id& addr) + { + out << "l2_3gpp_2g_cell_id: CI:" << addr._ci << " LAC:" << addr._lac << " PLMN id:" << addr.plmn_id[0]<< addr.plmn_id[1]<< addr.plmn_id[2]; + return out; + } + + /** + * Check if the L2_3GPP_2G_CELL_ID is equal to another L2_3GPP_2G_CELL_ID. + * + * @param other The L2_3GPP_2G_CELL_ID to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const l2_3gpp_2g_cell_id& other) const + { + std::cerr << "l2_3gpp_2g_cell_id()==" << std::endl; + std::cerr << "_lac " << _lac << " " << other._lac << std::endl; + std::cerr << "_ci " << _ci << " " << other._ci << std::endl; + std::cerr << "plmn_id " << plmn_id[0] << " " << other.plmn_id[0] << std::endl; + std::cerr << "plmn_id " << plmn_id[1] << " " << other.plmn_id[1] << std::endl; + std::cerr << "plmn_id " << plmn_id[2] << " " << other.plmn_id[2] << std::endl; + return ((_lac == other._lac) + && (_ci == other._ci) + && (plmn_id[0] == other.plmn_id[0]) + && (plmn_id[1] == other.plmn_id[1]) + && (plmn_id[2] == other.plmn_id[2])); + } +}; + +/** + * L2_3GPP_3G_CELL_ID data type. + */ +struct l2_3gpp_3g_cell_id { + uint8 plmn_id[3]; /**< PLMN_ID data type. */ + cell_id _cell_id; /**< CELL_ID data type. */ + + /** + * Serialize/deserialize the L2_3GPP_3G_CELL_ID data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & plmn_id[0]; + ar & plmn_id[1]; + ar & plmn_id[2]; + ar & _cell_id; + } + + /** + * L2_3GPP_3G_CELL_ID data type output. + * + * @param out ostream. + * @param addr L2_3GPP_3G_CELL_ID data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp_3g_cell_id& addr) + { + out << "l2_3gpp_3g_cell_id: Cell id:" << addr._cell_id << " PLMN id:" << addr.plmn_id[0]<< addr.plmn_id[1]<< addr.plmn_id[2]; + return out; + } + + /** + * Check if the L2_3GPP_3G_CELL_ID is equal to another L2_3GPP_3G_CELL_ID. + * + * @param other The L2_3GPP_3G_CELL_ID to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const l2_3gpp_3g_cell_id& other) const + { + std::cerr << "l2_3gpp_3g_cell_id()==" << std::endl; + std::cerr << "_cell_id " << _cell_id << " " << other._cell_id << std::endl; + std::cerr << "plmn_id " << plmn_id[0] << " " << other.plmn_id[0] << std::endl; + std::cerr << "plmn_id " << plmn_id[1] << " " << other.plmn_id[1] << std::endl; + std::cerr << "plmn_id " << plmn_id[2] << " " << other.plmn_id[2] << std::endl; + return ((_cell_id == other._cell_id) + && (plmn_id[0] == other.plmn_id[0]) + && (plmn_id[1] == other.plmn_id[1]) + && (plmn_id[2] == other.plmn_id[2])); + } +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * L2_3GPP_ADDR data type. + */ +struct l2_3gpp_addr { + /** + * Serialize/deserialize the L2_3GPP_ADDR data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & value; + } + + /** + * L2_3GPP_ADDR data type output. + * + * @param out ostream. + * @param addr L2_3GPP_ADDR data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp_addr& addr) + { + out << "l2_3gpp_addr " << addr.value; + return out; + } + + /** + * Check if the L2_3GPP_ADDR is equal to another L2_3GPP_ADDR. + * + * @param other The L2_3GPP_ADDR to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const l2_3gpp_addr& other) const + { + return ((value.compare(other.value) == 0)); + } + + octet_string value; /**< L2_3GPP_ADDR data type. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * Define L2_3GPP2_ADDR data type. + */ +struct l2_3gpp2_addr { + /** + * Serialize/deserialize the L2_3GPP2_ADDR data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & value; + } + + /** + * L2_3GPP2_ADDR data type output. + * + * @param out ostream. + * @param addr L2_3GPP2_ADDR data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const l2_3gpp2_addr& addr) + { + out << "l2_3gpp2_addr:" << addr.value; + return out; + } + + /** + * Check if the L2_3GPP2_ADDR is equal to another L2_3GPP2_ADDR. + * + * @param other The L2_3GPP2_ADDR to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const l2_3gpp2_addr& other) const + { + return ((value == other.value)); + } + + octet_string value; /**< L2_3GPP2_ADDR data type. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * Define OTHER_L2_ADDR data type. + */ +struct other_l2_addr { + /** + * Serialize/deserialize the OTHER_L2_ADDR data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & value; + } + + /** + * OTHER_L2_ADDR data type output. + * + * @param out ostream. + * @param addr OTHER_L2_ADDR data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const other_l2_addr& addr) + { + out << "other_l2_addr"; + return out; + } + + /** + * Check if the OTHER_L2_ADDR is equal to another OTHER_L2_ADDR. + * + * @param other The OTHER_L2_ADDR to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const other_l2_addr& other) const + { + return ((value == other.value)); + } + + octet_string value; /**< OTHER_L2_ADDR data type. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_ADDR data type. + */ +typedef boost::variant<mac_addr, + l2_3gpp_3g_cell_id, + l2_3gpp_2g_cell_id, + l2_3gpp_addr, + l2_3gpp2_addr, + other_l2_addr + > link_addr; + +/** + * LIST(LINK_ADDR) data type. + */ +typedef std::vector<link_addr> link_addr_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * IP_ADDR data type. + */ +class ip_addr : public transport_addr { +public: + /** + * IP_ADDR type enumeration. + */ + enum type_ip_enum { + none = 0, /**< None. */ + ipv4 = 1, /**< IP address version 4. */ + ipv6 = 2, /**< IP address version 6. */ + }; + + /** + * Construct a IP_ADDR data type. + * + * @param tp IP address type. + */ + ip_addr(type_ip_enum tp = none) : transport_addr(tp) + { } + + /** + * Construct a IP_ADDR data type. + * + * @param tp IP address type. + * @param raw Raw bytes of the IP address. + * @param len Size of the IP address raw bytes. + */ + ip_addr(type_ip_enum tp, const void* raw, size_t len) + : transport_addr(tp, raw, len) + { } + + /** + * Construct a IP_ADDR data type. + * + * @param tp IP address type. + * @param raw IP address string format. + */ + explicit ip_addr(type_ip_enum tp, const octet_string& addr) + : transport_addr(tp) + { + this->address(addr); + } + + /** + * Get the IP address string. + * + * @return The IP address string. + */ + octet_string address() const; + + /** + * Set the IP address. + * + * @param addr The IP address string. + */ + void address(const octet_string& addr); + + /** + * IP_ADDR data type output. + * + * @param out ostream. + * @param tp IP_ADDR data type. + * @return ostream reference. + */ + friend std::ostream& operator<<(std::ostream& out, const ip_addr& tp) + { + out << "\ntype: " << tp.type(); + out << "\naddress: " << tp.address(); + + return out; + } + + /** + * Check if the IP_ADDR is equal to another IP_ADDR. + * + * @param other The IP_ADDR to compare with. + * @return True if they are equal or false otherwise. + */ + bool operator==(const ip_addr& other) const + { + return ((type() == other.type()) && (address().compare(other.address()) == 0)); + } +}; + +typedef ip_addr dhcp_serv; /**< DHCP_SERV data type. */ +typedef ip_addr fn_agent; /**< FN_AGENT data type. */ +typedef ip_addr acc_rtr; /**< ACC_RTR data type. */ + +/////////////////////////////////////////////////////////////////////////////// +} /* namespace mih */ } /*namespace odtone */ + +// EOF //////////////////////////////////////////////////////////////////////// +#endif /* ODTONE_MIH_TYPES_ADDRESS__HPP_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/archive.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/archive.hpp new file mode 100644 index 0000000000000000000000000000000000000000..f2225b96f91c201d7a09c4e00529c05a64857202 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/archive.hpp @@ -0,0 +1,1313 @@ +//============================================================================= +// Brief : Archive details +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#ifndef ODTONE_MIH_DETAIL_ARCHIVE__HPP_ +#define ODTONE_MIH_DETAIL_ARCHIVE__HPP_ + +/////////////////////////////////////////////////////////////////////////////// +#include <odtone/cast.hpp> +#include <boost/array.hpp> +#include <boost/variant/variant.hpp> +#include <boost/variant/get.hpp> +#include <vector> +#include <list> + +/////////////////////////////////////////////////////////////////////////////// +namespace odtone { namespace mih { namespace detail { + +/////////////////////////////////////////////////////////////////////////////// +template<class T> +struct serialize { + template<class ArchiveT> + void operator()(ArchiveT& ar, T& val) const + { + val.serialize(ar); + } +}; + +template<class T> +struct serialize<std::vector<T> > { + void operator()(iarchive& ar, std::vector<T>& val) const + { + uint pos = ar.position(); + uint len = ar.list_length(); + + val.resize(len); + try { + typename std::vector<T>::iterator it = val.begin(); + + while (len--) { + ar & *it; + ++it; + } + + } catch (...) { + val.clear(); + ar.position(pos); + throw; + } + } + + void operator()(oarchive& ar, std::vector<T>& val) const + { + typename std::vector<T>::iterator it = val.begin(); + uint len = truncate_cast<uint>(val.size()); + uint pos = ar.position(); + + ar.list_length(len); + try { + while (len--) { + ar & *it; + ++it; + } + + } catch (...) { + ar.position(pos); + throw; + } + } +}; + +template<class T> +struct serialize<std::list<T> > { + void operator()(iarchive& ar, std::list<T>& val) const + { + uint pos = ar.position(); + uint len = ar.list_length(); + + val.resize(len); + try { + typename std::list<T>::iterator it = val.begin(); + + while (len--) { + ar & *it; + ++it; + } + + } catch (...) { + val.clear(); + ar.position(pos); + throw; + } + } + + void operator()(oarchive& ar, std::list<T>& val) const + { + typename std::list<T>::iterator it = val.begin(); + uint len = truncate_cast<uint>(val.size()); + uint pos = ar.position(); + + ar.list_length(len); + try { + while (len--) { + ar & *it; + ++it; + } + + } catch (...) { + ar.position(pos); + throw; + } + } +}; + +template<class T, size_t N> +struct serialize<boost::array<T, N> > { + typedef typename boost::array<T, N> value_type; + + template<class Archive> + void operator()(Archive& ar, value_type& val) const + { + for (size_t i = 0; i < N; ++i) + ar & val.elems[i]; + } +}; + +template<class T1> +struct serialize<boost::variant<T1> > { + void operator()(iarchive& ar, boost::variant<T1>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2> +struct serialize<boost::variant<T1, T2> > { + void operator()(iarchive& ar, boost::variant<T1, T2>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3> +struct serialize<boost::variant<T1, T2, T3> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4> +struct serialize<boost::variant<T1, T2, T3, T4> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5> +struct serialize<boost::variant<T1, T2, T3, T4, T5> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + case 7: + val = T8(); + ar & boost::get<T8>(val); + break; + + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + case 7: + ar & boost::get<T8>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8, class T9> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + case 7: + val = T8(); + ar & boost::get<T8>(val); + break; + + case 8: + val = T9(); + ar & boost::get<T9>(val); + break; + + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + case 7: + ar & boost::get<T8>(val); + break; + + case 8: + ar & boost::get<T9>(val); + break; + + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8, class T9, class T10> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + case 7: + val = T8(); + ar & boost::get<T8>(val); + break; + + case 8: + val = T9(); + ar & boost::get<T9>(val); + break; + + case 9: + val = T10(); + ar & boost::get<T10>(val); + break; + + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + case 7: + ar & boost::get<T8>(val); + break; + + case 8: + ar & boost::get<T9>(val); + break; + + case 9: + ar & boost::get<T10>(val); + break; + + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8, class T9, class T10, class T11> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + case 7: + val = T8(); + ar & boost::get<T8>(val); + break; + + case 8: + val = T9(); + ar & boost::get<T9>(val); + break; + + case 9: + val = T10(); + ar & boost::get<T10>(val); + break; + + case 10: + val = T11(); + ar & boost::get<T11>(val); + break; + + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + case 7: + ar & boost::get<T8>(val); + break; + + case 8: + ar & boost::get<T9>(val); + break; + + case 9: + ar & boost::get<T10>(val); + break; + + case 10: + ar & boost::get<T11>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8, class T9, class T10, class T11, class T12> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + case 7: + val = T8(); + ar & boost::get<T8>(val); + break; + + case 8: + val = T9(); + ar & boost::get<T9>(val); + break; + + case 9: + val = T10(); + ar & boost::get<T10>(val); + break; + + case 10: + val = T11(); + ar & boost::get<T11>(val); + break; + + case 11: + val = T12(); + ar & boost::get<T12>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + case 7: + ar & boost::get<T8>(val); + break; + + case 8: + ar & boost::get<T9>(val); + break; + + case 9: + ar & boost::get<T10>(val); + break; + + case 10: + ar & boost::get<T11>(val); + break; + + case 11: + ar & boost::get<T12>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + + +template<class T1, class T2, class T3, class T4, class T5, class T6, class T7, class T8, class T9, class T10, class T11, class T12, class T13> +struct serialize<boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13> > { + void operator()(iarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13>& val) const + { + octet selector; + + ar & selector; + switch (selector) { + case 0: + val = T1(); + ar & boost::get<T1>(val); + break; + + case 1: + val = T2(); + ar & boost::get<T2>(val); + break; + + case 2: + val = T3(); + ar & boost::get<T3>(val); + break; + + case 3: + val = T4(); + ar & boost::get<T4>(val); + break; + + case 4: + val = T5(); + ar & boost::get<T5>(val); + break; + + case 5: + val = T6(); + ar & boost::get<T6>(val); + break; + + case 6: + val = T7(); + ar & boost::get<T7>(val); + break; + + case 7: + val = T8(); + ar & boost::get<T8>(val); + break; + + case 8: + val = T9(); + ar & boost::get<T9>(val); + break; + + case 9: + val = T10(); + ar & boost::get<T10>(val); + break; + + case 10: + val = T11(); + ar & boost::get<T11>(val); + break; + + case 11: + val = T12(); + ar & boost::get<T12>(val); + break; + + case 12: + val = T13(); + ar & boost::get<T13>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } + + void operator()(oarchive& ar, boost::variant<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13>& val) const + { + octet selector = val.which(); + + ar & selector; + switch (selector) { + case 0: + ar & boost::get<T1>(val); + break; + + case 1: + ar & boost::get<T2>(val); + break; + + case 2: + ar & boost::get<T3>(val); + break; + + case 3: + ar & boost::get<T4>(val); + break; + + case 4: + ar & boost::get<T5>(val); + break; + + case 5: + ar & boost::get<T6>(val); + break; + + case 6: + ar & boost::get<T7>(val); + break; + + case 7: + ar & boost::get<T8>(val); + break; + + case 8: + ar & boost::get<T9>(val); + break; + + case 9: + ar & boost::get<T10>(val); + break; + + case 10: + ar & boost::get<T11>(val); + break; + + case 11: + ar & boost::get<T12>(val); + break; + + case 12: + ar & boost::get<T13>(val); + break; + + // default: + // ODTONE_NEVER_HERE; + } + } +}; + +/////////////////////////////////////////////////////////////////////////////// +} /* namspace detail */ + +/////////////////////////////////////////////////////////////////////////////// +template<class T> +inline iarchive& operator&(iarchive& ar, T& val) +{ + detail::serialize<T>()(ar, val); + return ar; +} + +template<class T> +inline oarchive& operator&(oarchive& ar, T& val) +{ + detail::serialize<T>()(ar, val); + return ar; +} + +/////////////////////////////////////////////////////////////////////////////// +} /* namespace mih */ } /*namespace odtone */ + +// EOF //////////////////////////////////////////////////////////////////////// +#endif /* ODTONE_MIH_DETAIL_ARCHIVE__HPP_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/bin_query.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/bin_query.hpp new file mode 100644 index 0000000000000000000000000000000000000000..06e6c6bcf582fc8df6d4ded72ee84c4aff30094b --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/bin_query.hpp @@ -0,0 +1,164 @@ +//============================================================================= +// Brief : MIH Binary Query Types +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#ifndef ODTONE_MIH_TYPES_BIN_QUERY__HPP_ +#define ODTONE_MIH_TYPES_BIN_QUERY__HPP_ + +/////////////////////////////////////////////////////////////////////////////// +#include <odtone/mih/types/base.hpp> +#include <odtone/mih/types/information.hpp> +#include <odtone/mih/types/location.hpp> + +/////////////////////////////////////////////////////////////////////////////// +namespace odtone { namespace mih { + +/////////////////////////////////////////////////////////////////////////////// +typedef cost_curr curr_pref; /**< CURR_PREG data type value. */ +typedef std::vector<network_id> netwk_inc; /**< NETWK_INC data type value. */ +typedef uint32 nghb_radius;/**< NGHB_RADIUS data type value.*/ +typedef uint32 ie_type; /**< IE_TYPE data type value.*/ +typedef std::vector<ie_type> rpt_templ; /**< RTP_TEMPL data type value.*/ + +/////////////////////////////////////////////////////////////////////////////// +/** + * NET_TYPE_INC data type enumeration. + */ +enum net_type_inc_enum { + net_type_inc_gsm = 0, /**< GSM */ + net_type_inc_gprs = 1, /**< GPRS */ + net_type_inc_edge = 2, /**< EDGE */ + net_type_inc_ethernet = 3, /**< Ethernet */ + net_type_inc_wireless_other = 4, /**< Wireless other */ + net_type_inc_ieee802_11 = 5, /**< IEEE 802.11 */ + net_type_inc_cdma2000 = 6, /**< CDMA 2000 */ + net_type_inc_umts = 7, /**< UMTS */ + net_type_inc_cdma2000_hrpd = 8, /**< CDMA 2000 HRPD */ + net_type_inc_ieee802_16 = 9, /**< IEEE 802.16 */ + net_type_inc_ieee802_20 = 10, /**< IEEE 802.20 */ + net_type_inc_ieee802_22 = 11, /**< IEEE 802.22 */ + net_type_inc_lte = 12, /**< LTE */ +}; + +/** + * NET_TYPE_INC data type. + */ +typedef bitmap<32, net_type_inc_enum> net_type_inc; + +/////////////////////////////////////////////////////////////////////////////// +/** + * QUERIER_LOC data type. + */ +struct querier_loc { + /** + * Construct an empty QUERIER_LOC data type. + */ + querier_loc() : _location(null()), _link_addr(null()), _nghb_radius(null()) + { } + + /** + * Serialize/deserialize the QUERIER_LOC data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & _location; + ar & _link_addr; + ar & _nghb_radius; + } + + boost::variant<null, location> _location; /**< Location information. */ + boost::variant<null, link_addr> _link_addr; /**< Link address. */ + boost::variant<null, nghb_radius> _nghb_radius; /**< Radius from the center point of + querier’s location. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * RPT_LIMIT data type. + */ +struct rpt_limit { + /** + * Construct an empty RPT_LIMIT data type. + */ + rpt_limit() : _max_ies(0), _start_entry(0) + { } + + /** + * Serialize/deserialize the RPT_LIMIT data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & _max_ies; + ar & _start_entry; + } + + uint16 _max_ies; /**< Maximum number of IEs. */ + uint16 _start_entry; /**< Start entry. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * IQ_BIN_DATA data type. + */ +struct iq_bin_data { + /** + * Construct an empty IQ_BIN_DATA data type. + */ + iq_bin_data() + : _querier_loc(null()), _net_type_inc(null()), _netwk_inc(null()), + _rpt_templ(null()), _rpt_limit(null()), _currency(null()) + { } + + /** + * Serialize/deserialize the IQ_BIN_DATA data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & _querier_loc; + ar & _net_type_inc; + ar & _netwk_inc; + ar & _rpt_templ; + ar & _rpt_limit; + ar & _currency; + } + + boost::variant<null, querier_loc> _querier_loc; /**< Querier's location.*/ + boost::variant<null, net_type_inc> _net_type_inc; /**< Set of link types. */ + boost::variant<null, netwk_inc> _netwk_inc; /**< List of network identifiers.*/ + boost::variant<null, rpt_templ> _rpt_templ; /**< List of IEs types. */ + boost::variant<null, rpt_limit> _rpt_limit; /**< Report limitation. */ + boost::variant<null, curr_pref> _currency; /**< Currency preference.*/ +}; + +/** + * LIST(IQ_BIN_DATA) data type. + */ +typedef std::vector<iq_bin_data> iq_bin_data_list; + +/////////////////////////////////////////////////////////////////////////////// +} /* namespace mih */ } /*namespace odtone */ + +// EOF //////////////////////////////////////////////////////////////////////// +#endif /* ODTONE_MIH_TYPES_BIN_QUERY__HPP_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/command_service.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/command_service.cpp new file mode 100644 index 0000000000000000000000000000000000000000..516801b49103d3c3d6e2b76697c42a6139526ee3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/command_service.cpp @@ -0,0 +1,1081 @@ +//============================================================================== +// Brief : Command Service +// Authors : Simao Reis <sreis@av.it.pt> +// Carlos Guimarães <cguimaraes@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "command_service.hpp" + +#include "log.hpp" +#include "utils.hpp" +#include "mihfid.hpp" +#include "transmit.hpp" +#include "link_book.hpp" + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/tlv_types.hpp> + +#include <boost/make_shared.hpp> +#include <boost/foreach.hpp> +/////////////////////////////////////////////////////////////////////////////// + +extern odtone::uint16 kConf_MIHF_Link_Response_Time_Value; +extern odtone::uint16 kConf_MIHF_Link_Delete_Value; + +namespace odtone { namespace mihf { + +/** + * Construct the command service. + * + * @param io The io_service object that command service module will use to + * dispatch handlers for any asynchronous operations performed on + * the socket. + * @param lpool The local transaction pool module. + * @param t The transmit module. + * @param abook The address book module. + * @param link_abook The link book module. + * @param user_abook The user book module. + * @param lrpool The link response pool module. + */ +command_service::command_service(io_service &io, + local_transaction_pool &lpool, + transmit &t, + address_book &abook, + link_book &link_abook, + user_book &user_abook, + link_response_pool &lrpool) + : _io(io), + _lpool(lpool), + _transmit(t), + _abook(abook), + _link_abook(link_abook), + _user_abook(user_abook), + _lrpool(lrpool) +{ +} + +/** + * Handler responsible for processing the received Link Get Parameters + * responses from Link SAPs. + * + * @param ec Error code. + * @param in The input message. + */ +void command_service::link_get_parameters_response_handler(const boost::system::error_code &ec, meta_message_ptr &in) +{ + if(ec) + return; + + { + boost::mutex::scoped_lock lock(_mutex); + _timer.erase(in->tid()); + } + + mih::status st = mih::status_failure; + mih::link_id lid; + mih::link_status_rsp lsr; + mih::status_rsp sr; + mih::status_rsp_list srl; + mih::dev_states_rsp_list dsrl; + meta_message_ptr out(new meta_message()); + + std::vector<mih::octet_string> ids = _link_abook.get_ids(); + std::vector<mih::octet_string>::iterator it_link; + for(it_link = ids.begin(); it_link != ids.end(); it_link++) { + if(_lrpool.check(in->tid(), *it_link)) { + // fill GetStatusResponseList + link_entry a; + mih::link_id lid; + + a = _link_abook.get(*it_link); + + lid.type = a.link_id.type; + lid.addr = a.link_id.addr; + + // fill capabilities + pending_link_response tmp = _lrpool.find(in->tid(), *it_link); + _lrpool.del(in->tid(), *it_link); + + mih::link_status_rsp& link_status = boost::get<mih::link_status_rsp>(tmp.response); + lsr = link_status; + + sr.id = lid; + sr.rsp = lsr; + + srl.push_back(sr); + + // If one or more responses are successful the status + // is set to success + st = mih::status_success; + } + } + + // Send Link_Get_Parameters.confirm to the user + if(st == mih::status_success) { + ODTONE_LOG(1, "(micm) setting response to Link_Get_Parameters.request"); + *out << mih::response(mih::response::link_get_parameters) + & mih::tlv_status(mih::status_success) + // & mih::tlv_dev_states_rsp_list(dsrl) + & mih::tlv_get_status_rsp_list(srl); + } else { + ODTONE_LOG(1, "(micm) setting failure response to Link_Get_Parameters.request"); + *out << mih::response(mih::response::link_get_parameters) + & mih::tlv_status(st); + } + + out->tid(in->tid()); + out->destination(in->source()); + out->source(mihfid); + + out->ip(in->ip()); + out->scope(in->scope()); + out->port(in->port()); + _transmit(out); +} + +/** + * Link Get Parameters Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_get_parameters_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received a Link_Get_Parameters.request from ", + in->source().to_string()); + + + if(utils::this_mihf_is_destination(in)) { + // + // Kick this message to MIH_Link SAP. + // + // local_transactions was made to handle request's + // from users to peer mihf's but in this case we add an + // entry to handle the MIH_Link_Get_Parameters and + // Link_Get_Parameters. + // + boost::optional<mih::dev_states_req> dsr; + mih::link_id_list lil; + mih::link_status_req lsr; + + *in >> mih::request(mih::request::link_get_parameters) + & mih::tlv_dev_states_req(dsr) + & mih::tlv_link_id_list(lil) + & mih::tlv_get_status_req_set(lsr); + + *out << mih::request(mih::request::link_get_parameters) + & mih::tlv_link_parameters_req(lsr._param_type_list) + & mih::tlv_link_states_req(lsr._states_req) + & mih::tlv_link_descriptor_req(lsr._desc_req); + + out->tid(in->tid()); + out->source(mihfid); + + // For each Link_ID in request message + std::vector<mih::link_id>::iterator lid; + for(lid = lil.begin(); lid != lil.end(); lid++) { + out->destination(mih::id(_link_abook.search_interface((*lid).type, (*lid).addr))); + // If the Link SAP it is known send message + if (out->destination().to_string().compare("") != 0) { + // Check if the Link SAP is still active + uint16 fails = _link_abook.fail(out->destination().to_string()); + if(fails > kConf_MIHF_Link_Delete_Value) { + mih::octet_string dst = out->destination().to_string(); + _link_abook.inactive(dst); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } else { + ODTONE_LOG(1, "(mics) forwarding Link_Get_Parameters.request to ", + out->destination().to_string()); + utils::forward_request(out, _lpool, _transmit); + } + } + } + + // Set the timer that will be responsible for aggregate and + // response to this resquest + boost::shared_ptr<boost::asio::deadline_timer> timer = boost::make_shared<boost::asio::deadline_timer>(_io); + timer->expires_from_now(boost::posix_time::milliseconds(kConf_MIHF_Link_Response_Time_Value)); + timer->async_wait(boost::bind(&command_service::link_get_parameters_response_handler, this, _1, in)); + + { + boost::mutex::scoped_lock lock(_mutex); + _timer[in->tid()] = timer; + } + + // Do not respond to the request. The response handler will be + // responsible for that. + return false; + } else { + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Link Get Parameters Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_get_parameters_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received Link_Get_Parameters.response from ", + in->source().to_string()); + + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + ODTONE_LOG(1, "(mics) forwarding Link_Get_Parameters.response to ", + in->destination().to_string()); + + _transmit(in); + + return false; +} + +/** + * Link Get Parameters Confirm message handler. + * + * @param in input message. + * @param out output message. + * @return true if the response is sent immediately or false otherwise. + */ +bool command_service::link_get_parameters_confirm(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received Link_Get_Parameters.confirm from ", + in->source().to_string()); + + _link_abook.reset(in->source().to_string()); + + if(_lpool.set_user_tid(in)) { + mih::status st; + boost::optional<mih::link_param_list> lpl; + boost::optional<mih::link_states_rsp_list> lsrl; + boost::optional<mih::link_desc_rsp_list> ldrl; + + *in >> mih::confirm(mih::confirm::link_get_parameters) + & mih::tlv_status(st) + & mih::tlv_link_parameters_status_list(lpl) + & mih::tlv_link_states_rsp(lsrl) + & mih::tlv_link_descriptor_rsp(ldrl); + + if(st == mih::status_success) { + mih::link_status_rsp link_status; + + link_status.states_rsp_list = lsrl.get(); + link_status.param_list = lpl.get(); + link_status.desc_rsp_list = ldrl.get(); + + _lrpool.add(in->source().to_string(), + in->tid(), + link_status); + } + + return false; + } + + ODTONE_LOG(1, "no pending transaction for this message, discarding"); + return false; +} + +/** + * Handler responsible for processing the received Link Get Parameters + * responses from Link SAPs. + * + * @param ec Error code. + * @param in The input message. + */ +void command_service::link_configure_thresholds_response_timeout(const boost::system::error_code &ec, meta_message_ptr &in) +{ + if(ec) + return; + + { + boost::mutex::scoped_lock lock(_mutex); + _timer.erase(in->tid()); + } + + mih::link_tuple_id link; + meta_message_ptr out(new meta_message()); + + *in >> mih::request(mih::request::link_configure_thresholds) + & mih::tlv_link_identifier(link); + + // Send failure message to the user + ODTONE_LOG(1, "(mics) setting failure response to Link_Configure_Thresholds.request"); + *out << mih::response(mih::response::link_configure_thresholds) + & mih::tlv_status(mih::status_failure) + & mih::tlv_link_identifier(link); + + out->tid(in->tid()); + out->destination(in->source()); + out->source(mihfid); + + _transmit(out); +} + +/** + * Link Configure Thresholds Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_configure_thresholds_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received a Link_Configure_Thresholds.request from ", + in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + // + // Kick this message to MIH_Link SAP. + // + // local_transactions was made to handle request's + // from users to peer mihf's but in this case we add an + // entry to handle the MIH_Link_Get_Parameters and + // Link_Get_Parameters. + // + mih::link_tuple_id lti; + mih::link_cfg_param_list lcpl; + + *in >> mih::request(mih::request::link_configure_thresholds) + & mih::tlv_link_identifier(lti) + & mih::tlv_link_cfg_param_list(lcpl); + + *out << mih::request(mih::request::link_configure_thresholds) + & mih::tlv_link_cfg_param_list(lcpl); + + out->destination(mih::id(_link_abook.search_interface(lti.type, lti.addr))); + out->source(in->source()); + out->tid(in->tid()); + + // If the Link SAP it is known continue + if (out->destination().to_string().compare("") == 0) { + *out << mih::response(mih::response::link_configure_thresholds) + & mih::tlv_status(mih::status_failure) + & mih::tlv_link_identifier(lti); + + out->tid(in->tid()); + out->source(mihfid); + out->destination(in->source()); + + ODTONE_LOG(1, "(mies) forwarding Link_Configure_Thresholds.response to ", + out->destination().to_string()); + + return true; + } + + // Check if the Link SAP is still active + uint16 fails = _link_abook.fail(out->destination().to_string()); + if(fails > kConf_MIHF_Link_Delete_Value) { + mih::octet_string dst = out->destination().to_string(); + _link_abook.inactive(dst); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } else { + ODTONE_LOG(1, "(mics) forwarding Link_Configure_Thresholds.request to ", + out->destination().to_string()); + utils::forward_request(out, _lpool, _transmit); + + // Set the timer that will be responsible for sending a failure + // response if necessary + boost::shared_ptr<boost::asio::deadline_timer> timer = boost::make_shared<boost::asio::deadline_timer>(_io); + timer->expires_from_now(boost::posix_time::milliseconds(kConf_MIHF_Link_Response_Time_Value)); + timer->async_wait(boost::bind(&command_service::link_configure_thresholds_response_timeout, this, _1, in)); + + { + boost::mutex::scoped_lock lock(_mutex); + _timer[in->tid()] = timer; + } + } + + return false; + } else { + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Link Configure Thresholds Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_configure_thresholds_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received Link_Configure_Thresholds.response from ", + in->source().to_string()); + + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + ODTONE_LOG(1, "(mics) forwarding Link_Configure_Thresholds.response to ", in->destination().to_string()); + + _transmit(in); + + return false; +} + +/** + * Link Configure Thresholds Confirm message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_configure_thresholds_confirm(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received Link_Configure_Thresholds.confirm from ", + in->source().to_string()); + + _link_abook.reset(in->source().to_string()); + + out->source(in->source()); + if (!_lpool.set_user_tid(out)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + { + boost::mutex::scoped_lock lock(_mutex); + _timer.erase(in->tid()); + } + + mih::status st; + boost::optional<mih::link_cfg_status_list> lcsl; + + mih::link_tuple_id li; + li.type = _link_abook.get(in->source().to_string()).link_id.type; + li.addr = _link_abook.get(in->source().to_string()).link_id.addr; + + *in >> mih::confirm(mih::confirm::link_configure_thresholds) + & mih::tlv_status(st) + & mih::tlv_link_cfg_status_list(lcsl); + + *out << mih::response(mih::response::link_configure_thresholds) + & mih::tlv_status(st) + & mih::tlv_link_identifier(li) + & mih::tlv_link_cfg_status_list(lcsl); + + out->source(mihfid); + + ODTONE_LOG(1, "(mics) forwarding Link_Configure_Thresholds.confirm to ", out->destination().to_string()); + + _transmit(out); + + return false; +} + +/** + * Handler responsible for setting a failure Link Action + * responses. + * + * @param ec Error code. + * @param in The input message. + */ +void command_service::link_actions_response_handler(const boost::system::error_code &ec, + meta_message_ptr &in) +{ + if(ec) + return; + + { + boost::mutex::scoped_lock lock(_mutex); + _timer.erase(in->tid()); + } + + mih::status st = mih::status_failure; + mih::link_action_rsp_list larl; + mih::link_action_rsp lar; + mih::link_ac_result laresult; + mih::link_scan_rsp_list lsrl; + meta_message_ptr out(new meta_message()); + + std::vector<mih::octet_string> ids = _link_abook.get_ids(); + std::vector<mih::octet_string>::iterator it_link; + for(it_link = ids.begin(); it_link != ids.end(); it_link++) { + if(_lrpool.check(in->tid(), *it_link)) { + // fill LinkActionsResultList + link_entry a; + mih::link_id lid; + + a = _link_abook.get(*it_link); + + lar.id.type = a.link_id.type; + lar.id.addr = a.link_id.addr; + + // fill action result + pending_link_response tmp = _lrpool.find(in->tid(), *it_link); + _lrpool.del(in->tid(), *it_link); + + action& ac = boost::get<action>(tmp.response); + + if(ac.link_scan_rsp_list.is_initialized()) { + lar.scan_list = ac.link_scan_rsp_list.get(); + } + lar.result = ac.link_ac_result.get(); + larl.push_back(lar); + + // If one or more responses are successful the status + // is set to success + st = mih::status_success; + } + } + + // Send Link_Actions.confirm to the user + if(st == mih::status_success) { + ODTONE_LOG(1, "(mics) setting response to Link_Actions.request"); + *out << mih::response(mih::response::link_actions) + & mih::tlv_status(st) + & mih::tlv_link_action_rsp_list(larl); + } else { + ODTONE_LOG(1, "(mics) setting failure response to Link_Actions.request"); + *out << mih::response(mih::response::link_actions) + & mih::tlv_status(st); + } + + out->tid(in->tid()); + out->destination(in->source()); + out->source(mihfid); + + out->ip(in->ip()); + out->scope(in->scope()); + out->port(in->port()); + _transmit(out); +} + +/** + * Link Actions Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_actions_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received a Link_Actions.request from ", + in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + // + // Kick this message to MIH_Link SAP. + // + // The solution found to handle this corner case in the + // 802.21 standard was to send the message, as is, to the + // link sap. + // + mih::link_action_list lal; + + *in >> mih::request(mih::request::link_actions) + & mih::tlv_link_action_list(lal); + + // For each Link_ID in request message + std::vector<mih::link_action_req>::iterator lar; + for(lar = lal.begin(); lar != lal.end(); lar++) { + std::cout<<"ADDR in MIHF "<<(*lar).id.addr<<std::endl; + out->destination(mih::id(_link_abook.search_interface((*lar).id.type, (*lar).id.addr))); + // If the Link SAP it is known send message + if (out->destination().to_string().compare("") != 0) { + // Check if the Link SAP is still active + uint16 fails = _link_abook.fail(out->destination().to_string()); + if(fails > kConf_MIHF_Link_Delete_Value) { + mih::octet_string dst = out->destination().to_string(); + _link_abook.inactive(dst); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } else { + mih::link_addr* a = boost::get<mih::link_addr>(&(*lar).addr); + if (a && ((*lar).action.attr.get(mih::link_ac_attr_data_fwd_req)) ) { + *out << mih::request(mih::request::link_actions) + & mih::tlv_link_action((*lar).action) + & mih::tlv_time_interval((*lar).ex_time) + & mih::tlv_poa(*a); + } else { + *out << mih::request(mih::request::link_actions) + & mih::tlv_link_action((*lar).action) + & mih::tlv_time_interval((*lar).ex_time); + } + + out->tid(in->tid()); + out->source(mihfid); + + ODTONE_LOG(1, "(mics) forwarding Link_Actions.request to ", + out->destination().to_string()); + utils::forward_request(out, _lpool, _transmit); + } + } + } + + // Set the timer that will be responsible for aggregate and + // response to this resquest + boost::shared_ptr<boost::asio::deadline_timer> timer = boost::make_shared<boost::asio::deadline_timer>(_io); + timer->expires_from_now(boost::posix_time::milliseconds(kConf_MIHF_Link_Response_Time_Value)); + timer->async_wait(boost::bind(&command_service::link_actions_response_handler, this, _1, in)); + + { + boost::mutex::scoped_lock lock(_mutex); + _timer[in->tid()] = timer; + } + + // Do not respond to the request. The thread response handler will be + // responsible for that. + return false; + } else { + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Link Actions Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_actions_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received Link_Actions.response from ", + in->source().to_string()); + + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) no local pending transaction for this message, discarding"); + return false; + } + + ODTONE_LOG(1, "(mics) forwarding Link_Actions.response to ", in->destination().to_string()); + + _transmit(in); + + return false; +} + +/** + * Link Actions Confirm message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::link_actions_confirm(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mics) received Link_Actions.confirm from ", + in->source().to_string()); + + _link_abook.reset(in->source().to_string()); + + if(_lpool.set_user_tid(in)) { + mih::status st; + boost::optional<mih::link_scan_rsp_list> lsrl; + boost::optional<mih::link_ac_result> lar; + + *in >> mih::confirm(mih::confirm::link_actions) + & mih::tlv_status(st) + & mih::tlv_link_scan_rsp_list(lsrl) + & mih::tlv_link_ac_result(lar); + + if(st == mih::status_success) { + _lrpool.add(in->source().to_string(), + in->tid(), + lsrl, + lar.get()); + } + + return false; + } + + ODTONE_LOG(1, "no pending transaction for this message, discarding"); + return false; + +} + +/** + * Currently command service handover related messages are handled by + * a single MIH-user. If this MIHF is the destination of the message, + * forward it to the MIH-User with mobility role. + * + * @param recv_msg The receive message output. + * @param send_msg The send message output. + * @param in The input message. + * @param out The output message. + * @param cmd The command that the MIH-Users must support in order to + * receive an indication about the reception opf this message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::generic_command_request(const char *recv_msg, + const char *send_msg, + meta_message_ptr &in, + meta_message_ptr &out, + mih::mih_cmd_list_enum cmd) +{ + ODTONE_LOG(1, recv_msg, in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + // Forward this message to MIH-User for handover as an indication + in->opcode(mih::operation::indication); + std::vector<mih::octet_string> user_list = _user_abook.get_ids(); + BOOST_FOREACH(mih::octet_string id, user_list) { + user_entry user = _user_abook.get(id); + if(user.supp_cmd.is_initialized()) { + if(user.supp_cmd->get(cmd)) { + in->destination(mih::id(id)); + _lpool.add(in); + + if(in->is_local()) + in->source(mihfid); + + ODTONE_LOG(1, send_msg , in->destination().to_string()); + _transmit(in); + } + } + } + + // Restore the original opcode after sending the indication message + in->opcode(mih::operation::request); + return false; + } else { + // try to forward the message, this is to handle the + // special case of the user handling MIH commands + // sending some MIH command request to a peer mihf + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Currently command service handover related messages are handled by + * a single MIH-user. If this MIHF is the destination of the message, + * check for a pending transaction and forwards the message. + * + * @param recv_msg The receive message output. + * @param send_msg The send message output. + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::generic_command_response(const char *recv_msg, + const char *send_msg, + meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, recv_msg, in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + ODTONE_LOG(1, send_msg , in->destination().to_string()); + in->opcode(mih::operation::confirm); + _transmit(in); + } else { + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + ODTONE_LOG(1, send_msg , in->destination().to_string()); + in->source(mihfid); + _transmit(in); + } + + return false; +} + +/** + * Net Handover Candidate Query Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::net_ho_candidate_query_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received Net_HO_Candidate_Query.request from ", + "(mics) sending a Net_HO_Candidate_Query.indication to ", + in, out, mih::mih_cmd_net_ho_candidate_query); +} + +/** + * Net Handover Candidate Query Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::net_ho_candidate_query_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received Net_HO_Candidate_Query.response from ", + "(mics) sending a Net_HO_Candidate_Query.confirm to ", + in, out); +} + +/** + * MN Handover Query Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::mn_ho_candidate_query_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a MN_HO_Candidate_Query.request from ", + "(mics) sending a MN_HO_Candidate_Query.indication to ", + in, out, mih::mih_cmd_mn_ho_candidate_query); +} + +/** + * MN Handover Query Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::mn_ho_candidate_query_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received MN_HO_Candidate_Query.response from ", + "(mics) sending a MN_HO_Candidate_Query.confirm to ", + in, out); +} + +/** + * N2N Handover Query Resources Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::n2n_ho_query_resources_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a MN_N2N_HO_Query_Resources.request from ", + "(mics) sending a MN_N2N_HO_Query_Resources.indication to ", + in, out, mih::mih_cmd_n2n_ho_query_resources); +} + +/** + * N2N Handover Query Resources Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::n2n_ho_query_resources_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received MN_N2N_HO_Query_Resources.response from ", + "(mics) sending a MN_N2N_HO_Query_Resources.confirm to ", + in, out); +} + + +/** + * MN Handover Commit Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::mn_ho_commit_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a MN_HO_Commit.request from ", + "(mics) sending a MN_HO_Commit.indication to ", + in, out, mih::mih_cmd_mn_ho_commit); +} + +/** + * MN Handover Commit Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::mn_ho_commit_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received MN_HO_Commit.response from ", + "(mics) sending a MN_HO_Commit.confirm to ", + in, out); +} + +/** + * Net Handover Commit Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::net_ho_commit_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a Net_HO_Commit.request from ", + "(mics) sending a Net_HO_Commit.indication to ", + in, out, mih::mih_cmd_net_ho_commit); +} + +/** + * Net Handover Commit Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::net_ho_commit_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received Net_HO_Commit.response from ", + "(mics) sending a Net_HO_Commit.confirm to ", + in, out); +} + +/** + * MN Handover Complete Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::mn_ho_complete_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a MN_HO_Complete.request from ", + "(mics) sending a MN_HO_Complete.indication to ", + in, out, mih::mih_cmd_mn_ho_complete); +} + +/** + * MN Handover Complete Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::mn_ho_complete_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received MN_HO_Complete.response from ", + "(mics) sending a MN_HO_Complete.confirm to ", + in, out); +} + +/** + * N2N Handover Commit Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::n2n_ho_commit_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a N2N_HO_Commit.request from ", + "(mics) sending a N2N_HO_Commit.indication to ", + in, out, mih::mih_cmd_n2n_ho_commit); +} + +/** + * N2N Handover Commit Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::n2n_ho_commit_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received N2N_HO_Commit.response from ", + "(mics) sending a N2N_HO_Commit.confirm to ", + in, out); +} + + +/** + * N2N Handover Complete Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::n2n_ho_complete_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_request("(mics) received a N2N_HO_Complete.request from ", + "(mics) sending a N2N_HO_Complete.indication to ", + in, out, mih::mih_cmd_n2n_ho_complete); +} + +/** + * N2N Handover Complete Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool command_service::n2n_ho_complete_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + return generic_command_response("(mics) received N2N_HO_Complete.response from ", + "(mics) sending a N2N_HO_Complete.confirm to ", + in, out); +} + + +} /* namespace mihf */ } /* namespace odtone */ + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/conf.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/conf.cpp new file mode 100644 index 0000000000000000000000000000000000000000..e99fd91498c087987b5c3007a9addbee71cb6d03 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/conf.cpp @@ -0,0 +1,43 @@ +//======================================================================================================= +// Brief : Configuration DSL +// Authors : Bruno Santos <bsantos@av.it.pt> +// ------------------------------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//======================================================================================================= + +#include <odtone/conf.hpp> +#include <boost/spirit/include/qi_parse_attr.hpp> + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +namespace odtone { namespace conf { + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +bool exec(std::string::const_iterator& begin, std::string::const_iterator end, functions const& cm) +{ + skipper_grammar<std::string::const_iterator> skipper; + parser_grammar<std::string::const_iterator> parser(cm); + std::string::const_iterator pos = begin; + + while (begin != end) { + if (!qi::phrase_parse(begin, end, parser, skipper) || pos == begin) + return false; + pos = begin; + } + + return true; +} + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +} /* namespace conf */ } /* namespace odtone */ + +// EOF ////////////////////////////////////////////////////////////////////////////////////////////////// diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/conf.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/conf.hpp new file mode 100644 index 0000000000000000000000000000000000000000..61b773c936c430b4fab68a86812d92a1aa5639ed --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/conf.hpp @@ -0,0 +1,670 @@ +//======================================================================================================= +// Brief : Configuration DSL +// Authors : Bruno Santos <bsantos@av.it.pt> +// ------------------------------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//======================================================================================================= + +#ifndef ODTONE_PMIP_CONF__HPP_ +#define ODTONE_PMIP_CONF__HPP_ + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +#include <odtone/base.hpp> +#include <odtone/net/ip/address.hpp> +#include <boost/variant.hpp> +#include <boost/function.hpp> +#include <boost/mpl/for_each.hpp> +#include <boost/mpl/vector.hpp> +#include <boost/mpl/at.hpp> +#include <boost/mpl/back.hpp> +#include <boost/mpl/deref.hpp> +#include <boost/mpl/prior.hpp> +#include <boost/function_types/result_type.hpp> +#include <boost/function_types/parameter_types.hpp> +#include <boost/function_types/function_arity.hpp> +#include <boost/spirit/include/qi.hpp> +#include <boost/spirit/include/phoenix_core.hpp> +#include <boost/spirit/include/phoenix_operator.hpp> +#include <boost/spirit/include/phoenix_bind.hpp> +#include <boost/spirit/include/phoenix_fusion.hpp> +#include <boost/spirit/include/phoenix_stl.hpp> +#include <boost/fusion/include/std_pair.hpp> +#include <boost/mpl/find.hpp> +#include <algorithm> +#include <cstring> +#include <string> +#include <map> + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +namespace odtone { namespace conf { + +namespace qi = boost::spirit::qi; +namespace ph = boost::phoenix; + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +enum error_reason { + invalid_syntax, + invalid_command, + invalid_arg_type, + to_many_args, + invalid_prop, + invalid_prop_arg_type, + to_many_prop_args, + + error_reason_size +}; + +static char const* const error_reason_string[error_reason_size] = { + "invalid syntax", + "invalid command", + "invalid argument type", + "to many arguments", + "invalid property", + "invalid property argument type", + "to many property arguments" +}; + +typedef boost::variant<uint, + sint, + double, + std::string, + net::ip::address_v4, + net::ip::address_v6 + > arg_type; + +typedef std::vector<arg_type> args_type; +typedef std::pair<std::string, args_type> prop_type; +typedef std::map<std::string, args_type> pset_type; + +template<class T> +class type_id_ { + typedef typename boost::remove_cv<typename boost::remove_reference<T>::type>::type tt; + typedef typename boost::mpl::find<arg_type::types, tt>::type ft; + typedef typename boost::mpl::distance<boost::mpl::begin<arg_type::types>::type, ft>::type tp; + + ODTONE_STATIC_ASSERT( + (!boost::is_same<typename boost::mpl::end<arg_type::types>::type, ft>::value), + "Type not supported" + ); + +public: + static const uint value = tp::value; +}; + +struct property_class { + const char* name; //property name + uint type_id; //property argument type + uint count; //maximum number of arguments it can take +}; + +class function { + template<class F, size_t N> + class at_ { + typedef typename boost::function_types::parameter_types<F>::type ps; + typedef typename boost::mpl::at_c<ps, N>::type tp; + + public: + typedef typename boost::remove_cv<typename boost::remove_reference<tp>::type>::type type; + }; + + template<class F, size_t N> + static const typename at_<F, N>::type& get_(arg_type const& arg) + { + typedef typename at_<F, N>::type type; + + return boost::get<type>(arg); + } + + template<class F, class WithPropSet, size_t N> + struct adaptor_; + + template<class F> + struct adaptor_<F, boost::mpl::true_, 1> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), pset); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::false_, 1> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0])); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::true_, 2> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), pset); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::false_, 2> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1])); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::true_, 3> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), pset); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::false_, 3> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2])); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::true_, 4> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), + get_<F, 3>(args[3]), pset); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::false_, 4> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), + get_<F, 3>(args[3])); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::true_, 5> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), + get_<F, 3>(args[3]), get_<F, 4>(args[4]), pset); + } + }; + + template<class F> + struct adaptor_<F, boost::mpl::false_, 5> { + F f; + + adaptor_(F f_) : f(f_) + { } + + void operator()(args_type const& args, pset_type const& pset) const + { + f(get_<F, 0>(args[0]), get_<F, 1>(args[1]), get_<F, 2>(args[2]), + get_<F, 3>(args[3]), get_<F, 4>(args[4])); + } + }; + + template<class F, class WithPropSet> + class traits_; + + template<class F> + class traits_<F, boost::mpl::true_> { + typedef typename boost::function_types::parameter_types<F>::type tmp; + typedef typename boost::mpl::prior<typename boost::mpl::end<tmp>::type>::type last; + + ODTONE_STATIC_ASSERT( + (boost::is_same<void, + typename boost::function_types::result_type<F>::type>::value), + "Result type must be void" + ); + + ODTONE_STATIC_ASSERT( + (boost::is_same<pset_type const&, + typename boost::mpl::deref<last>::type>::value), + "Last argument must a const reference to a pset_type" + ); + + public: + typedef typename boost::mpl::erase<tmp, last>::type arg_types; + + static const size_t arg_count = boost::mpl::size<arg_types>::value; + }; + + template<class F> + class traits_<F, boost::mpl::false_> { + ODTONE_STATIC_ASSERT( + (boost::is_same<void, + typename boost::function_types::result_type<F>::type>::value), + "Result type must be void" + ); + + public: + typedef typename boost::function_types::parameter_types<F>::type arg_types; + + static const size_t arg_count = boost::mpl::size<arg_types>::value; + }; + + struct push_arg_id_ { + push_arg_id_(std::vector<uint>& ais) + : args_id(ais) + { } + + template<class T> + void operator()(T) + { + const uint id = type_id_<T>::value; + + args_id.push_back(id); + } + + std::vector<uint>& args_id; + }; + +public: + function() + : _pclass(0), _pcsize(0) + {} + + template<class F> + function(F f) + : _pclass(0), _pcsize(0) + { + typedef typename traits_<F, boost::mpl::false_>::arg_types types; + + boost::mpl::for_each< + types, + boost::remove_cv< boost::remove_reference<boost::mpl::_1> > + >(push_arg_id_(_args)); + + _func = adaptor_<F, boost::mpl::false_, traits_<F, boost::mpl::false_>::arg_count>(f); + } + + template<class F, size_t N> + function(F f, property_class const (&pc)[N]) + : _pclass(pc), _pcsize(N) + { + typedef typename traits_<F, boost::mpl::true_>::arg_types types; + + boost::mpl::for_each< + types, + boost::remove_cv< boost::remove_reference<boost::mpl::_1> > + >(push_arg_id_(_args)); + + _func = adaptor_<F, boost::mpl::true_, traits_<F, boost::mpl::true_>::arg_count>(f); + } + + void operator()(args_type const& args, pset_type const& pset) const + { + _func(args, pset); + } + + size_t max_args() const + { + return _args.size(); + } + + uint arg_id(uint n) const + { + return _args[n]; + } + + property_class const* get_prop_class(std::string const& name) const + { + return std::find_if(_pclass, _pclass + _pcsize, + !ph::bind(&std::strcmp, name.c_str(), + ph::bind(&property_class::name, ph::arg_names::arg1))); + } + +private: + std::vector<uint> _args; + property_class const* _pclass; + size_t _pcsize; + + boost::function<void(args_type const& args, pset_type const& pset)> _func; +}; + +typedef std::map<std::string, function> functions; + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +template<class Iterator> +struct skipper_grammar : qi::grammar<Iterator> { + skipper_grammar() + : skipper_grammar::base_type(sk, "skipper") + { + sk = + qi::blank + | ('#' >> *(qi::standard::char_ - qi::eol) > &qi::eol) + | ('\\' >> qi::eol) + ; + } + + qi::rule<Iterator> sk; +}; + +template<class Iterator> +struct string_grammar : qi::grammar<Iterator, std::string()> { + string_grammar() + : string_grammar::base_type(str, "string") + { + esc.add("\\\"", '\"') + ("\\\\", '\\') + ("\\t", '\t') + ("\\n", '\n') + ("\\r", '\r') + ; + + str = + '"' + >> qi::as_string[*((esc | qi::standard::char_) - '"')][qi::_val = qi::_1] + > '"' + ; + } + + qi::symbols<const char, const char> esc; + qi::rule<Iterator, std::string()> str; +}; + +template<class Iterator> +struct ip4_grammar : qi::grammar<Iterator, net::ip::address_v4()> { + + static net::ip::address_v4 conv(std::string const& str) + { + return net::ip::address_v4::from_string(str); + } + + ip4_grammar() + : ip4_grammar::base_type(ip4, "ip4") + { + ip4 = qi::as_string + [ + qi::raw + [ + qi::repeat(3)[u8 >> '.'] >> u8 + ] + ] + [qi::_val = ph::bind(&ip4_grammar::conv, qi::_1)]; + } + + qi::rule<Iterator, net::ip::address_v4()> ip4; + qi::uint_parser<uint8, 10, 1, 3> u8; +}; + +template<class Iterator> +struct ip6_grammar : qi::grammar<Iterator, qi::locals<uint>, net::ip::address_v6()> { + typedef qi::locals<uint> locals; + + static net::ip::address_v6 conv(std::string const& str) + { + return net::ip::address_v6::from_string(str); + } + + ip6_grammar() + : ip6_grammar::base_type(ip6, "ip6") + { + ip6 = qi::as_string + [ + qi::raw + [ + ( qi::repeat(1, 6)[h16 >> ':'] >> ((h16 >> (':' >> (h16 | ':'))) | ':' | ip4) ) + | ( "::" >> -(qi::repeat(0, 5)[h16 >> ':'] >> ((h16 >> -(':' >> (h16 ))) | ip4)) ) + | ( + qi::eps[qi::_a = 0] + >> +(h16[qi::_pass = ++qi::_a < 7u] >> ':') + >> +(':' >> h16[qi::_pass = ++qi::_a < 8u]) + ) + ] + ] + [qi::_val = ph::bind(&ip6_grammar::conv, qi::_1)]; + } + + qi::rule<Iterator, locals, net::ip::address_v6()> ip6; + qi::uint_parser<uint16, 16, 1, 4> h16; + ip4_grammar<Iterator> ip4; +}; + +template<class Iterator> +struct parser_grammar : qi::grammar<Iterator, + skipper_grammar<Iterator> + > { + typedef skipper_grammar<Iterator> skipper; + + struct act_on_name_ { + template<class, class, class, class> + struct result { typedef bool type; }; + + bool operator()(function const*& f, functions const& cm, std::string const& name, error_reason& er) const + { + functions::const_iterator it = cm.find(name); + + if (it == cm.end()) { + er = invalid_command; + return false; + } + + f = &it->second; + return true; + } + }; + + struct act_on_arg_ { + template<class, class, class, class> + struct result { typedef bool type; }; + + bool operator()(function const& f, args_type& args, arg_type& arg, error_reason& er) const + { + const uint n = args.size(); + + if (n >= f.max_args()) { + er = to_many_args; + return false; + } + + if (static_cast<uint>(arg.which()) != f.arg_id(n)) { + er = invalid_arg_type; + return false; + } + + args.push_back(arg); + return true; + } + }; + + struct act_on_prop_ { + template<class, class, class, class> + struct result { typedef bool type; }; + + bool operator()(property_class const*& pc, function const& f, std::string const& name, error_reason& er) const + { + pc = f.get_prop_class(name); + + if (!pc) { + er = invalid_prop; + return false; + } + return true; + } + + bool operator()(property_class const* pc, arg_type const& arg, uint& count, error_reason& er) const + { + if (++count > pc->count) { + er = to_many_prop_args; + return false; + } + if (pc->type_id != uint(arg.which())) { + er = invalid_prop_arg_type; + return false; + } + return true; + } + }; + + struct run_cmd_ { + template<class, class, class> + struct result { typedef void type; }; + + void operator()(function const& f, args_type const& args, pset_type const& pset) const + { + f(args, pset); + } + }; + + struct error_handler_ { + template<class, class, class, class, class> + struct result { typedef void type; }; + + void operator()(Iterator begin, Iterator end, Iterator pos, qi::info const& what, error_reason er) const + { + Iterator eol = std::find(begin, end, '\n'); + + std::cout << "error: " + << error_reason_string[er] + << ", expecting \'" + << what + << "\':\n" + << std::string(begin, eol) + << std::endl + << std::string(std::distance(begin, pos), '~') + << '^' + << std::string(std::distance(pos, eol), '~') + << std::endl; + } + }; + + parser_grammar(functions const& cm) + : parser_grammar::base_type(start), cmds(cm), ereason(invalid_syntax) + { + start = qi::eol | cmd; + + cmd = ( + name [qi::_pass = act_on_name(qi::_a, ph::cref(cmds), qi::_1, ph::ref(ereason))] + > *arg [qi::_pass = act_on_arg(*qi::_a, qi::_b, qi::_1, ph::ref(ereason))] + > -pset(*qi::_a) [qi::_c = qi::_1] + > qi::eol + ) + [run_cmd(*qi::_a, qi::_b, qi::_c)]; + + name %= qi::lexeme[qi::alpha >> *(qi::alnum | '-')]; + + arg %= + ip4 + | ip6 + | qi::uint_ + | qi::int_ + | qi::double_ + | str + ; + + pset = + qi::lit('{') + > qi::eol + > +prop(qi::_r1) + > qi::lit('}') + ; + + prop = + name[qi::_val = qi::_1, qi::_a = 0, + qi::_pass = act_on_prop(qi::_a, qi::_r1, qi::_1, ph::ref(ereason))] + > (*arg[qi::_val = qi::_1, + qi::_pass = act_on_prop(qi::_a, qi::_1, qi::_b, ph::ref(ereason))]) + > qi::eol + ; + + cmd.name("command"); + name.name("identifier"); + arg.name("argument"); + pset.name("property-set"); + prop.name("property"); + + qi::on_error<qi::fail>(start, error_handler(qi::_1, qi::_2, qi::_3, qi::_4, ph::ref(ereason))); + } + + typedef qi::locals<function const*, args_type, pset_type> cmd_locals; + typedef qi::locals<property_class const*, uint> prop_locals; + + qi::rule<Iterator, skipper> start; + qi::rule<Iterator, cmd_locals, skipper> cmd; + qi::rule<Iterator, std::string(), skipper> name; + qi::rule<Iterator, arg_type(), skipper> arg; + qi::rule<Iterator, pset_type(function const&), skipper> pset; + qi::rule<Iterator, prop_type(function const&), prop_locals, skipper> prop; + + string_grammar<Iterator> str; + ip4_grammar<Iterator> ip4; + ip6_grammar<Iterator> ip6; + + functions const& cmds; + error_reason ereason; + + ph::function<act_on_name_> act_on_name; + ph::function<act_on_arg_> act_on_arg; + ph::function<act_on_prop_> act_on_prop; + ph::function<run_cmd_> run_cmd; + ph::function<error_handler_> error_handler; +}; + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +bool exec(std::string::const_iterator& begin, std::string::const_iterator end, functions const& cm); + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +} /* namespace conf */ } /* namespace odtone */ + +// EOF ////////////////////////////////////////////////////////////////////////////////////////////////// +#endif /* ODTONE_PMIP_CONF__HPP_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/dst_transaction.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/dst_transaction.cpp new file mode 100644 index 0000000000000000000000000000000000000000..f6d7d86b61e20fb55a89cbc33ce8043be0bdf9f5 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/dst_transaction.cpp @@ -0,0 +1,136 @@ +//============================================================================== +// Brief : Destination Transaction +// Authors : Simao Reis <sreis@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "log.hpp" +#include "dst_transaction.hpp" +#include "utils.hpp" +/////////////////////////////////////////////////////////////////////////////// + +namespace odtone { namespace mihf { + +/** + * Constructor a Destination State Machine transaction. + * + * @param f The transaction handler. + * @param netsap The netsap module. + */ +dst_transaction_t::dst_transaction_t(handler_t &f, net_sap &netsap) + : transaction_t(f, netsap) +{ + state = DST_INIT; +} + +/** + * Run Destination State Machine transaction. + */ +void dst_transaction_t::run() +{ + switch (state) + { + case DST_INIT: goto _init_lbl_; + case DST_WAIT_RESPONSE_PRM: goto _wait_response_prm_lbl_; + case DST_SEND_RESPONSE: goto _send_response_lbl_; + case DST_FAILURE: goto _failure_lbl_; + case DST_SUCCESS: goto _success_lbl_; + } + + _init_lbl_: + { + ODTONE_LOG(1, "(dst_transaction_t) init tid ", in->tid()); + transaction_status = ONGOING; + opcode = in->opcode(); + tid = in->tid(); + transaction_stop_when = 15; // TODO: read from config + is_multicast = utils::is_multicast(in); + peer_mihf_id = in->source(); + my_mihf_id = in->destination(); + start_ack_responder = (in->ackreq() && !is_multicast); + msg_in_avail = false; + + out.reset(new meta_message); + msg_out_avail = process_message(in, out); + + if (start_ack_responder) + ack_responder(); + + if (opcode == mih::operation::request) + goto _wait_response_prm_lbl_; + else if ((opcode == mih::operation::indication) + || (opcode == mih::operation::response)) + goto _success_lbl_; + } + + _wait_response_prm_lbl_: + { + ODTONE_LOG(1, "(dst_transaction_t) wait response tid ", tid); + state = DST_WAIT_RESPONSE_PRM; + + if (transaction_stop_when == 0) + goto _failure_lbl_; + + if (msg_out_avail && (!start_ack_responder || out->ackrsp())) + goto _send_response_begin_lbl_; + + return; + } + + _send_response_begin_lbl_: + { + ODTONE_LOG(1, "(dst_transaction_t) send response begin tid ", tid); + state = DST_SEND_RESPONSE; + + start_ack_requestor = out->ackreq(); + if(start_ack_requestor) + ack_requestor(); + ack_requestor_status = ONGOING; + + _netsap.send(out); + } + + _send_response_lbl_: + { + ODTONE_LOG(1, "(dst_transaction_t) send response tid ", tid); + if (!start_ack_requestor || ack_requestor_status == SUCCESS) + goto _success_lbl_; + else if (ack_requestor_status == FAILURE) + goto _failure_lbl_; + + return; + } + + _failure_lbl_: + { + ODTONE_LOG(1, "(dst_transaction_t) failure tid ", tid); + state = DST_FAILURE; + transaction_status = FAILURE; + + return; + } + + _success_lbl_: + { + ODTONE_LOG(1, "(dst_transaction_t) success tid ", tid); + state = DST_SUCCESS; + transaction_status = SUCCESS; + } + + return; +} + + +} /* namespace mihf */ } /* namespace odtone */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/event_service.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/event_service.cpp new file mode 100644 index 0000000000000000000000000000000000000000..9c021eac74034e4c4fe9ed5a8ab94173fa90645f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/event_service.cpp @@ -0,0 +1,956 @@ +//============================================================================== +// Brief : Event Service +// Authors : Simao Reis <sreis@av.it.pt> +// Carlos Guimarães <cguimaraes@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include "event_service.hpp" + +#include "log.hpp" +#include "link_book.hpp" +#include "mihfid.hpp" +#include "transmit.hpp" +#include "utils.hpp" + +#include <odtone/debug.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/tlv_types.hpp> + +#include <boost/make_shared.hpp> +#include <boost/foreach.hpp> +/////////////////////////////////////////////////////////////////////////////// + +extern odtone::uint16 kConf_MIHF_Link_Response_Time_Value; +extern odtone::uint16 kConf_MIHF_Link_Delete_Value; + +namespace odtone { namespace mihf { + + + + +/** + * Construct the event service. + * + * @param io The io_service object that event service module will use to + * dispatch handlers for any asynchronous operations performed on + * the socket. + * @param lpool The local transaction pool module. + * @param t The transmit module. + * @param abook The address book module. + * @param lbook The link book module. + * @param ubook The user book module. + */ +event_service::event_service(io_service &io, local_transaction_pool &lpool, + transmit &t, address_book &abook, link_book &lbook, + user_book &ubook) + : _io(io), + _lpool(lpool), + _transmit(t), + _abook(abook), + _link_abook(lbook), + _user_abook(ubook) +{ +} + +/** + * Handler responsible for setting a failure Link Event Subscribe + * response. + * + * @param ec Error code. + * @param in The input message. + */ +void event_service::link_event_subscribe_response_timeout(const boost::system::error_code &ec, meta_message_ptr &in) +{ + if(ec) + return; + + { + boost::mutex::scoped_lock lock(_mutex); + _timer.erase(in->tid()); + } + + mih::link_tuple_id link; + meta_message_ptr out(new meta_message()); + + *in >> mih::request(mih::request::event_subscribe) + & mih::tlv_link_identifier(link); + + // Send failure message to the user + ODTONE_LOG(1, "(mism) setting failure response to Link_Event_Subscribe.request"); + *out << mih::response(mih::response::event_subscribe) + & mih::tlv_status(mih::status_failure) + & mih::tlv_link_identifier(link); + + out->tid(in->tid()); + out->destination(in->source()); + out->source(mihfid); + + _transmit(out); +} + +/** + * Make a subscription for a given user. + * + * @param user The MIH-User/MIHF that request the subscription. + * @param link The link to make the subscription. + * @param events The events to subscribe. + * @return The status of the operation. + */ +mih::status event_service::subscribe(const mih::id &user, + mih::link_tuple_id &link, + mih::mih_evt_list &events) +{ + event_registration_t reg; + reg.user.assign(user.to_string()); + reg.link = link; + + boost::mutex::scoped_lock lock(_event_mutex); + + for(int i = 0; i < 32; i++) { + if (events.get((mih::mih_evt_list_enum) i)) { + reg.event = (mih::mih_evt_list_enum) i; + std::list<event_registration_t>::iterator tmp; + tmp = std::find(_event_subscriptions.begin(), _event_subscriptions.end(), reg); + if (tmp == _event_subscriptions.end()) { + _event_subscriptions.push_back(reg); + ODTONE_LOG(3, "(mies) added subscription ", reg.user, + ":", reg.link.addr, ":", reg.event); + } + } + } + + return mih::status_success; +} + +/** + * Deserialize the message, subscribe the user and send a response immediatly + * if the events are already subscribed with the Link SAP. Otherwise, the MIHF + * sends a request to the Link SAP to subscribe the desired events. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::local_event_subscribe_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + mih::mih_evt_list events; + mih::link_tuple_id link; + + // TODO: optional is not take in cosideration yet + *in >> mih::request(mih::request::event_subscribe) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); + + mih::octet_string link_id = _link_abook.search_interface(link.type, link.addr); + + // Check if the Link SAP exists + // If not replies with a failure status + if(link_id.compare("") == 0) { + *out << mih::response(mih::response::event_subscribe) + & mih::tlv_status(mih::status_failure) + & mih::tlv_link_identifier(link); + + out->tid(in->tid()); + out->source(mihfid); + out->destination(in->source()); + + ODTONE_LOG(1, "(mies) forwarding Event_Subscribe.response to ", + out->destination().to_string()); + + return true; + } + + // Check if requested events have been already subscribed + mih::mih_evt_list event_tmp; + { + boost::mutex::scoped_lock lock(_event_mutex); + + BOOST_FOREACH(event_registration_t item, _event_subscriptions) + { + if(item.link == link) + event_tmp.set(item.event); + } + } + + if(events == event_tmp) { + mih::status st = subscribe(in->source(), link, events); + + if(st == mih::status_success) { + *out << mih::response(mih::response::event_subscribe) + & mih::tlv_status(st) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); + } else { + *out << mih::response(mih::response::event_subscribe) + & mih::tlv_status(st) + & mih::tlv_link_identifier(link); + } + + out->tid(in->tid()); + out->source(mihfid); + out->destination(in->source()); + + ODTONE_LOG(1, "(mies) forwarding Event_Subscribe.response to ", + out->destination().to_string()); + + return true; + } else { // Subscribe requested events with Link SAP + mih::link_evt_list evt; + // Since the two bitmaps have the same values + // we can assign them directly + for (size_t i = 0; i < 32; ++i) { + if(events.get((mih::mih_evt_list_enum)i)) { + evt.set((mih::link_evt_list_enum)i); + } + } + // + + *out << mih::request(mih::request::event_subscribe) + & mih::tlv_link_evt_list(evt); + + out->destination(mih::id(link_id)); + out->source(in->source()); + out->tid(in->tid()); + + // Check if the Link SAP is still active + uint16 fails = _link_abook.fail(out->destination().to_string()); + if(fails > kConf_MIHF_Link_Delete_Value) { + mih::octet_string dst = out->destination().to_string(); + _link_abook.inactive(dst); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } else { + ODTONE_LOG(1, "(mies) forwarding Event_Subscribe.request to ", + out->destination().to_string()); + utils::forward_request(out, _lpool, _transmit); + + // Set the timer that will be responsible for sending a failure + // response if necessary + boost::shared_ptr<boost::asio::deadline_timer> timer = boost::make_shared<boost::asio::deadline_timer>(_io); + timer->expires_from_now(boost::posix_time::milliseconds(kConf_MIHF_Link_Response_Time_Value)); + timer->async_wait(boost::bind(&event_service::link_event_subscribe_response_timeout, this, _1, in)); + + { + boost::mutex::scoped_lock lock(_mutex); + _timer[in->tid()] = timer; + } + } + + return false; + } +} + +/** + * Event Subscribe Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::event_subscribe_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Event_Subscribe.request from ", + in->source().to_string()); + + if (utils::this_mihf_is_destination(in)) { + return local_event_subscribe_request(in, out); + } else { + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Event Subscribe Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::event_subscribe_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Event_Subscribe.response from ", + in->source().to_string()); + + // do we have a request from a user? + if (!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mies) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + mih::status st; + mih::link_tuple_id link; + boost::optional<mih::mih_evt_list> events; + + // parse incoming message to (event_registration_t) reg + *in >> mih::response() + & mih::tlv_status(st) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); + + + std::cout<< "(mies) The link received is "<<link<<std::endl; + // add a subscription + if (st == mih::status_success) { + st = subscribe(mih::id(in->destination().to_string()), link, events.get()); + } + + ODTONE_LOG(1, "(mies) forwarding Event_Subscribe.response to ", + in->destination().to_string()); + + // forward to user + _transmit(in); + + return false; +} + +/** + * Event Subscribe Confirm message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::event_subscribe_confirm(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Event_Subscribe.confirm from ", + in->source().to_string()); + + _link_abook.reset(in->source().to_string()); + + // do we have a request from a user? + out->source(in->source()); + if (!_lpool.set_user_tid(out)) { + ODTONE_LOG(1, "(mies) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + { + boost::mutex::scoped_lock lock(_mutex); + _timer.erase(in->tid()); + } + + mih::status st; + boost::optional<mih::link_evt_list> events; + + *in >> mih::confirm(mih::confirm::event_subscribe) + & mih::tlv_status(st) + & mih::tlv_link_evt_list(events); + + mih::link_tuple_id link; + link.type = _link_abook.get(in->source().to_string()).link_id.type; + link.addr = _link_abook.get(in->source().to_string()).link_id.addr; + + if(st == mih::status_success) { + mih::mih_evt_list evt; + + // Since the two bitmaps have the same values + // we can assign them directly + for (size_t i = 0; i < 32; ++i) { + if(events.get().get((mih::link_evt_list_enum)i)) { + evt.set((mih::mih_evt_list_enum)i); + } + } + // + + *out << mih::response(mih::response::event_subscribe) + & mih::tlv_status(st) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(evt); + + st = subscribe(mih::id(out->destination().to_string()), link, evt); + } else { + *out << mih::response(mih::response::event_subscribe) + & mih::tlv_status(st) + & mih::tlv_link_identifier(link); + } + + ODTONE_LOG(1, "(mies) forwarding Event_Subscribe.confirm to ", + out->destination().to_string()); + + // forward to user + out->source(mihfid); + _transmit(out); + + return false; +} + +/** + * Check if there is events subscribed to a given Link SAP, which + * are not required anymore. + * + * @param in The input message. + * @param link The link to make the unsubscription. + * @param events The events to unsubscribed. + */ +void event_service::link_unsubscribe(meta_message_ptr &in, + mih::link_tuple_id &link, + mih::mih_evt_list &events) +{ + boost::mutex::scoped_lock lock(_event_mutex); + + // Check if requested events have been already subscribed + mih::mih_evt_list event_unsubscribe = events; + BOOST_FOREACH(event_registration_t item, _event_subscriptions) + { + if(item.link == link) { + if(item.user.compare(in->source().to_string()) != 0) { + event_unsubscribe.clear(item.event); + } + } + } + + // Only send message to Link SAP if there is any event to unsubscribed + // with it + mih::mih_evt_list empty; + if(!(empty == event_unsubscribe)) + { + mih::link_evt_list evt; + // Since the two bitmaps have the same values + // we can assign them directly + for (size_t i = 0; i < 32; ++i) { + if(event_unsubscribe.get((mih::mih_evt_list_enum)i)) { + evt.set((mih::link_evt_list_enum)i); + } + } + // + + *in << mih::request(mih::request::event_unsubscribe) + & mih::tlv_link_evt_list(evt); + + mih::octet_string link_id = _link_abook.search_interface(link.type, link.addr); + in->destination(mih::id(link_id)); + in->source(mihfid); + + // Check if the Link SAP is still active + uint16 fails = _link_abook.fail(in->destination().to_string()); + if(fails > kConf_MIHF_Link_Delete_Value) { + mih::octet_string dst = in->destination().to_string(); + _link_abook.inactive(dst); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } + else { + ODTONE_LOG(1, "(mies) forwarding Event_Subscribe.request to ", + in->destination().to_string()); + utils::forward_request(in, _lpool, _transmit); + } + } +} + +/** + * Unsubscribe the events related to a given user. + * + * @param user The MIH-User/MIHF that request the unsubscription. + * @param link The link to make the unsubscription. + * @param events The events to unsubscribe. + * @return The status of the operation. + */ +mih::status event_service::unsubscribe(const mih::id &user, + mih::link_tuple_id &link, + mih::mih_evt_list &events) +{ + boost::mutex::scoped_lock lock(_event_mutex); + + std::list<event_registration_t>::iterator it; + + it = _event_subscriptions.begin(); + while (it != _event_subscriptions.end()) + { + if (it->link == link && + (it->user.compare(user.to_string()) == 0) && + events.get((mih::mih_evt_list_enum) it->event)) { + ODTONE_LOG(3, "(mies) removed subscription ", it->user, + ":", it->link.addr ,":", it->event); + _event_subscriptions.erase(it++); + } + else { + it++; + } + } + + return mih::status_success; +} + +/** + * Deserialize message, unsubscribe user and send a response to the + * requestor. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::local_event_unsubscribe_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + mih::status st; + mih::link_tuple_id link; + mih::mih_evt_list events; + + *in >> mih::request(mih::request::event_unsubscribe) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); + + st = unsubscribe(in->source(), link, events); + + *out << mih::response(mih::response::event_unsubscribe) + & mih::tlv_status(st) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); + + out->tid(in->tid()); + out->source(mihfid); + out->destination(in->source()); + + // Check if there is any request for the events + link_unsubscribe(in, link, events); + + ODTONE_LOG(1, "(mies) forwarding Event_Unsubscribe.response to ", + out->destination().to_string()); + + return true; +} + +/** + * Event Unsubscribe Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::event_unsubscribe_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Event_Unsubscribe.request from ", + in->source().to_string()); + + if (utils::this_mihf_is_destination(in)) { + return local_event_unsubscribe_request(in, out); + } else { + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Event Unsubscribe Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::event_unsubscribe_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Event_Unsubscribe.response from ", + in->source().to_string()); + + // do we have a request from a user? + if (!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + + return false; + } + + mih::status st; + mih::link_tuple_id link; + boost::optional<mih::mih_evt_list> events; + + // parse incoming message to (event_registration_t) reg + *in >> mih::response(mih::response::event_unsubscribe) + & mih::tlv_status(st) + & mih::tlv_link_identifier(link) + & mih::tlv_event_list(events); + + // remove subscription + if (st == mih::status_success) { + st = unsubscribe(mih::id(in->destination().to_string()), link, events.get()); + } + + ODTONE_LOG(1, "(mies) forwarding Event_Unsubscribe.response to ", + in->destination().to_string()); + + // forward to user + in->opcode(mih::operation::response); + _transmit(in); + + return false; +} + +/** + * Event Unsubscribe Confirm message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::event_unsubscribe_confirm(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Event_Unsubscribe.confirm from ", + in->source().to_string()); + + _link_abook.reset(in->source().to_string()); + + // do we have a request from a user? + if (!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mies) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + mih::status st; + boost::optional<mih::link_evt_list> events; + + // parse incoming message to (event_registration_t) reg + *in >> mih::confirm(mih::confirm::event_unsubscribe) + & mih::tlv_status(st) + & mih::tlv_link_evt_list(events); + + if (st == mih::status_success) { + ODTONE_LOG(1, "(mies) Events successfully unsubscribed in Link SAP ", + in->source().to_string()); + } + + return false; +} + +/** + * Forward the message for all users subscribed to event from the + * Link SAP. + * + * @param msg The MIH Message. + * @param li The link identifier. + * @param event The related event. + */ +void event_service::msg_forward(meta_message_ptr &msg, + mih::link_tuple_id &li, + mih::mih_evt_list_enum event) +{ + std::list<event_registration_t>::iterator it; + int i = 0; // for logging purposes + + if(msg->is_local()) + msg->source(mihfid); + + for(it = _event_subscriptions.begin(); + it != _event_subscriptions.end(); + it++, i++) { + ODTONE_LOG(3, "(mies) msg_forward() comparing event ", event, " with event_subscription it->event ", it->event); + ODTONE_LOG(3, "(mies) msg_forward() comparing link_tuple_id.type ", li.type, " with event_subscription it->link.type ", it->link.type); + ODTONE_LOG(3, "(mies) msg_forward() comparing link_tuple_id.addr ", li.addr, " with event_subscription it->link.addr ", it->link.addr); + if ((it->event == event) && (it->link == li)) { + ODTONE_LOG(3, "(mies) found registration of user: ", + it->user, " for event type ", event); + msg->destination(mih::id(it->user)); + _transmit(msg); + } + } + ODTONE_LOG(3, "(mies) msg_forward() end"); +} + + +/** + * Parse the link identifier from incoming message and forwards the + * message to subscribed users + * + * @param msg The MIH Message. + * @param event The related event. + */ +void event_service::link_event_forward(meta_message_ptr &msg, + mih::mih_evt_list_enum event) +{ + mih::link_tuple_id li; + ODTONE_LOG(1, "(mies) link_event_forward()"); + *msg >> mih::indication() + & mih::tlv_link_identifier(li); + + msg_forward(msg, li, event); +} + + +/** + * Link Up Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_up_indication(meta_message_ptr &in, meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Up.indication from ", + in->source().to_string()); + + if(in->is_local()) { + ODTONE_LOG(1, "(mies) link_up_indication() reset ", in->source().to_string() ," in link_abook"); + _link_abook.reset(in->source().to_string()); + } + + link_event_forward(in, mih::mih_evt_link_up); + + return false; +} + + +/** + * Link Down Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_down_indication(meta_message_ptr &in, meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Down.indication from ", + in->source().to_string()); + + if(in->is_local()) + _link_abook.reset(in->source().to_string()); + + link_event_forward(in, mih::mih_evt_link_down); + + return false; +} + + +/** + * Link Detected Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_detected_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Detected.indication from ", + in->source().to_string()); + + if(in->is_local()) { + _link_abook.reset(in->source().to_string()); + // link detected info from incoming message + mih::link_det_info link_info; + // link detected info on outgoing message + mih::link_det_info_list list_rsp; + + *in >> mih::indication() + & mih::tlv_link_det_info(link_info); + + list_rsp.push_back(link_info); + + *in << mih::indication(mih::indication::link_detected) + & mih::tlv_link_det_info_list(list_rsp); + } + + std::list<event_registration_t>::iterator it; + int i = 0; // for logging purposes + + if(in->is_local()) + in->source(mihfid); + + for(it = _event_subscriptions.begin(); + it != _event_subscriptions.end(); + it++, i++) { + if (it->event == mih::mih_evt_link_detected) { + ODTONE_LOG(3, i, " (mies) found registration of user: ", + it->user, " for event type ", mih::mih_evt_link_detected); + in->destination(mih::id(it->user)); + _transmit(in); + } + } + + return false; +} + +/** + * Link Going Down Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_going_down_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Going_Down.indication from ", + in->source().to_string()); + + if(in->is_local()) + _link_abook.reset(in->source().to_string()); + + link_event_forward(in, mih::mih_evt_link_going_down); + + return false; +} + +/** + * Link Parameters Report Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_parameters_report_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Parameters_Report.indication from ", + in->source().to_string()); + + if(in->is_local()) + _link_abook.reset(in->source().to_string()); + + link_event_forward(in, mih::mih_evt_link_parameters_report); + + return false; +} + +/** + * Link Handover Imminent Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_handover_imminent_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Handover_Imminent.indication from ", + in->source().to_string()); + + if(in->is_local()) + _link_abook.reset(in->source().to_string()); + + std::list<event_registration_t>::iterator it; + int i = 0; // for logging purposes + + if(in->is_local()) + in->source(mihfid); + + for(it = _event_subscriptions.begin(); + it != _event_subscriptions.end(); + it++, i++) { + if (it->event == mih::mih_evt_link_handover_imminent) { + ODTONE_LOG(3, i, " (mies) found registration of user: ", + it->user, " for event type ", mih::mih_evt_link_handover_imminent); + in->destination(mih::id(it->user)); + _transmit(in); + } + } + + return false; +} + +/** + * Link Handover Complete Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_handover_complete_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_Handover_Complete.indication from ", + in->source().to_string()); + + if(in->is_local()) + _link_abook.reset(in->source().to_string()); + + mih::link_tuple_id oli; + mih::link_tuple_id nli; + boost::optional<mih::link_addr> oar; + boost::optional<mih::link_addr> nar; + + if(in->is_local()) { + mih::status st; + *in >> mih::indication(mih::indication::link_handover_complete) + & mih::tlv_link_identifier(oli) + & mih::tlv_new_link_identifier(nli) + & mih::tlv_old_access_router(oar) + & mih::tlv_new_access_router(nar) + & mih::tlv_status(st); + } else { + *in >> mih::indication(mih::indication::link_handover_complete) + & mih::tlv_link_identifier(oli) + & mih::tlv_new_link_identifier(nli) + & mih::tlv_old_access_router(oar) + & mih::tlv_new_access_router(nar); + } + + *in << mih::indication(mih::indication::link_handover_complete) + & mih::tlv_link_identifier(oli) + & mih::tlv_new_link_identifier(nli) + & mih::tlv_old_access_router(oar) + & mih::tlv_new_access_router(nar); + + std::list<event_registration_t>::iterator it; + int i = 0; // for logging purposes + + if(in->is_local()) + in->source(mihfid); + + for(it = _event_subscriptions.begin(); + it != _event_subscriptions.end(); + it++, i++) { + if (it->event == mih::mih_evt_link_handover_complete) { + ODTONE_LOG(3, i, " (mies) found registration of user: ", + it->user, " for event type ", mih::mih_evt_link_handover_complete); + in->destination(mih::id(it->user)); + _transmit(in); + } + } + + return false; +} + +/** + * Link PDU Transmit Status Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool event_service::link_pdu_transmit_status_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mies) received Link_PDU_Transmit_Status.indication from ", + in->source().to_string()); + + if(in->is_local()) + _link_abook.reset(in->source().to_string()); + + link_event_forward(in, mih::mih_evt_link_pdu_transmit_status); + + return false; +} + +} /* namespace mihf */ } /* namespace odtone */ +// EOF //////////////////////////////////////////////////////////////////////// diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/link.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/link.hpp new file mode 100644 index 0000000000000000000000000000000000000000..f9c8718250cd49ed0b4fba2ff54b71a8c6bf9718 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/link.hpp @@ -0,0 +1,1353 @@ +//============================================================================= +// Brief : MIH Link Types +// Authors : Bruno Santos <bsantos@av.it.pt> +// Simao Reis <sreis@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#ifndef ODTONE_MIH_TYPES_LINK__HPP_ +#define ODTONE_MIH_TYPES_LINK__HPP_ + +/////////////////////////////////////////////////////////////////////////////// +#include <odtone/mih/types/base.hpp> +#include <odtone/mih/types/address.hpp> +#include <odtone/mih/types/qos.hpp> + +/////////////////////////////////////////////////////////////////////////////// +namespace odtone { namespace mih { + +/////////////////////////////////////////////////////////////////////////////// +/** + * BATT_LEVEL data type. + */ +typedef uint8 batt_level; + +/** + * NUM_COS data type. + */ +typedef uint8 num_cos; + +/** + * NUM_QUEUE data type. + */ +typedef uint8 num_queue; + +/** + * CHANNEL_ID data type. + */ +typedef uint16 channel_id; + +/** + * CONFIG_STATUS data type. + */ +typedef bool config_status; + +/** + * DEVICE_INFO data type. + */ +typedef octet_string device_info; + +/** + * LINK_AC_EX_TIME data type. + */ +typedef uint16 link_ac_ex_time; + +/** + * SIG_STRENGTH data type. + */ +typedef boost::variant<sint8, percentage> sig_strength; + +/** + * LINK_RES_STATUS data type. + */ +typedef bool link_res_status; + +/** + * MAX_DELAY data type. + */ +typedef uint16 max_delay; + +/** + * BITRATE data type. + */ +typedef uint32 bitrate; + +/** + * JITTER data type. + */ +typedef uint16 jitter; + +/** + * PKT_LOSS_RATE data type. + */ +typedef uint16 pkt_loss_rate; + +/** + * COS data type. + */ +typedef uint16 cos; + +/** + * DROP_ELIGIBILITY data type. + */ +typedef bool drop_eligibility; + +/** + * MULTICAST_ENABLE data type. + */ +typedef bool multicast_enable; + +/** + * JUMBO_ENABLE data type. + */ +typedef bool jumbo_enable; + +/** + * PORT data type. + */ +typedef uint16 port; + +/** + * MARK data type. + */ +typedef uint8 mark; + +/** + * FLOW_ID data type. + */ +typedef uint32 flow_id; + +/////////////////////////////////////////////////////////////////////////////// +/** + * PROTO data type enumeration. + */ +enum proto_enum { + proto_tcp = 0, /**< TCP. */ + proto_udp = 1 /**< UDP. */ +}; + +/** + * PROTO data type. + */ +typedef enumeration<proto_enum> proto; +/////////////////////////////////////////////////////////////////////////////// +/** + * OP_MODE data type enumeration. + */ +enum op_mode_enum { + op_mode_normal = 0, /**< Normal mode. */ + op_mode_power_saving = 1, /**< Power saving mode. */ + op_mode_powered_down = 2, /**< Powered Down. */ +}; + +/** + * OP_MODE data type. + */ +typedef enumeration<op_mode_enum> op_mode; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_STATES_RSP data type. + */ +typedef boost::variant<op_mode, channel_id> link_states_rsp; + +/** + * LIST(LINK_STATES_RSP) data type. + */ +typedef std::vector<link_states_rsp> link_states_rsp_list; + +/** + * LINK_DESC_RSP data type. + */ +typedef boost::variant<num_cos, num_queue> link_desc_rsp; + +/** + * LIST(LINK_DESC_RSP) data type. + */ +typedef std::vector<link_desc_rsp> link_desc_rsp_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * DEV_STATES_REQ data type enumeration. + */ +enum dev_states_req_enum { + dev_state_device_info = 0, /**< Device information. */ + dev_state_batt_level = 1, /**< Battery level. */ +}; + +/** + * DEV_STATES_REQ data type. + */ +typedef bitmap<16, dev_states_req_enum> dev_states_req; + +/////////////////////////////////////////////////////////////////////////////// +/** + * DEV_STATES_RSP data type. + */ +typedef boost::variant<device_info, batt_level> dev_states_rsp; + +/** + * LIST(DEV_STATES_RSP) data type. + */ +typedef std::vector<dev_states_rsp> dev_states_rsp_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_STATES_REQ data type enumeration. + */ +enum link_states_req_enum { + link_states_req_op_mode = 0, /**< Operation mode. */ + link_states_req_channel_id = 1, /**< Channel identifier. */ +}; + +/** + * LINK_STATES_REQ data type. + */ +typedef bitmap<16, link_states_req_enum> link_states_req; + +/** + * LIST(LINK_STATES_REQ) data type. + */ +typedef std::vector<link_states_req> link_states_req_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * THRESHOLD data type. + */ +struct threshold { + /** + * Threshold direction enumeration. + */ + enum type_ip_enum { + above_threshold = 0, /**< Above threshold. */ + below_threshold = 1, /**< Below threshold. */ + }; + + /** + * Serialize/deserialize the THRESHOLD data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & threshold_val; + ar & threshold_x_dir; + } + + uint16 threshold_val; /**< Threshold value. */ + enumeration<type_ip_enum> threshold_x_dir; /**< Threshold Direction. */ +}; +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_TYPE data type enumeration. + */ +enum link_type_enum { + link_type_gsm = 1, /**< Wireless - GSM. */ + link_type_gprs = 2, /**< Wireless - GPRS. */ + link_type_edge = 3, /**< Wireless - EDGE. */ + + link_type_ethernet = 15, /**< Ethernet. */ + + link_type_wireless_other = 18, /**< Wireless - Other. */ + link_type_802_11 = 19, /**< Wireless - IEEE 802.11. */ + + link_type_cdma2000 = 22, /**< Wireless - CDMA-2000. */ + link_type_umts = 23, /**< Wireless - UMTS. */ + link_type_cdma2000_hrpd = 24, /**< Wireless - CDMA-2000-HRPD. */ + link_type_lte = 25, /**< Wireless - LTE. */ + + link_type_802_16 = 27, /**< Wireless - IEEE 802.16. */ + link_type_802_20 = 28, /**< Wireless - IEEE 802.20. */ + link_type_802_22 = 29 /**< Wireless - IEEE 802.22. */ +}; + +/** + * LINK_TYPE data type. + */ +typedef enumeration<link_type_enum> link_type; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_ID data type. + */ +struct link_id { + /** + * Construct an empty LINK_ID data type. + */ + link_id() : type(link_type_enum(0)) + { } + + /** + * Serialize/deserialize the LINK_ID data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & type; + ar & addr; + } + + /** + * Check if the LINK_ID data type is equal to another LINK_ID. + * + * @param other The LINK_ID to compare to. + * @return True if they are equal or false otherwise. + */ + bool operator==(const link_id& other) const + { + return ((type == other.type) && (addr == other.addr)); + } + + /** + * LINK_ID data type output. + * + * @param out ostream. + * @param addr LINK_ID data type. + * @return ostream reference. + */ friend std::ostream& operator<<(std::ostream& out, const link_id& lid) + { + out << "link id " << lid.type << " " << lid.addr; + return out; + } + + + link_type type; /**< Link address type. */ + link_addr addr; /**< Link address. */ +}; + +/** + * LIST(LINK_ID) data type. + */ +typedef std::vector<link_id> link_id_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * IP_TUPLE data type. + */ +struct ip_tuple { + /** + * Serialize/deserialize the IP_TUPLE data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & ip; + ar & port_val; + } + + ip_addr ip; /**< IP address.*/ + port port_val; /**< Port. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * Auxiliar QOS data type. + */ +struct qos_sequence { + /** + * Serialize/deserialize the auxiliar QOS data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & delay_val; + ar & bitrate_val; + ar & jitter_val; + ar & pkt_loss_val; + } + + max_delay delay_val; /**< Maximum delay. */ + bitrate bitrate_val; /**< Bitrate. */ + jitter jitter_val; /**< Jitter. */ + pkt_loss_rate pkt_loss_val; /**< Packet loss. */ +}; + +/** + * QOS data type. + */ +struct qos { + /** + * Serialize/deserialize the QOS data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & value; + } + + boost::variant<qos_sequence, cos> value; /**< QoS value. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * Auxiliar MARK_QoS data type. + */ +struct mark_qos_sequence { + /** + * Serialize/deserialize the auxiliar FLOW_ATTRIBUTE data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & mark_val; + ar & qos_val; + } + + mark mark_val; /**< Mark value. */ + qos qos_val; /**< QoS value. */ +}; + +/** + * Auxiliar MARK_DROP_ELIG data type. + */ +struct mark_drop_elig_sequence { + /** + * Serialize/deserialize the auxiliar FLOW_ATTRIBUTE data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & mark_val; + ar & drop_val; + } + + mark mark_val; /**< Mark value. */ + drop_eligibility drop_val; /**< Drop eligibility value. */ +}; + +/** + * FLOW_ATTRIBUTE data type. + */ +struct flow_attribute { + /** + * Serialize/deserialize the FLOW_ATTRIBUTE data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & multicast; + ar & qos_val; + ar & drop_elig_val; + } + + flow_id id; /**< Flow ID. */ + boost::variant<null, multicast_enable> multicast; /**< Multicast enable. */ + boost::variant<null, mark_qos_sequence> qos_val; /**< Mask/QoS value. */ + boost::variant<null, mark_drop_elig_sequence> drop_elig_val; /**< Mask/drop eligibility value. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * RESOURCE_DESC data type. + */ +struct resource_desc { + /** + * Serialize/deserialize the RESOURCE_DESC data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & lid; + ar & fid; + ar & data_rate; + ar & qos_val; + ar & jumbo; + ar & multicast; + } + + link_id lid; /**< Link ID. */ + flow_id fid; /**< Flow ID. */ + boost::variant<null, uint32> data_rate; /**< Link data rate. */ + boost::variant<null, qos> qos_val; /**< QoS value. */ + boost::variant<null, jumbo_enable> jumbo; /**< Jumbo enable. */ + boost::variant<null, multicast_enable> multicast; /**< Multicast enable. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_AC_RESULT data type enumeration. + */ +enum link_ac_result_enum { + link_ac_success = 0, /**< Success. */ + link_ac_failure = 1, /**< Failure. */ + link_ac_refused = 2, /**< Refused. */ + link_ac_incapable = 3, /**< Incapable. */ +}; + +/** + * LINK_AC_RESULT data type. + */ +typedef enumeration<link_ac_result_enum> link_ac_result; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_AC_TYPE data type enumeration. + */ +enum link_ac_type_enum { + link_ac_type_none = 0, /**< None. */ + link_ac_type_disconnect = 1, /**< Link disconnect. */ + link_ac_type_low_power = 2, /**< Link low power. */ + link_ac_type_power_down = 3, /**< Link power down. */ + link_ac_type_power_up = 4, /**< Link power up. */ + link_ac_type_flow_attr = 5, /**< Flow Attribute. */ + link_ac_type_link_activate_resources = 6, /**< Link Activate Resources. */ + link_ac_type_link_deactivate_resources = 7, /**< Link Deactivate Resources. */ +}; + +/** + * LINK_AC_TYPE data type. + */ +typedef enumeration<link_ac_type_enum> link_ac_type; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_AC_ATTR data type enumeration. + */ +enum link_ac_attr_enum { + link_ac_attr_scan = 0, /**< Link scan. */ + link_ac_attr_res_retain = 1, /**< Link resource retain. */ + link_ac_attr_data_fwd_req = 2, /**< Link data forward. */ +}; + +/** + * LINK_AC_ATTR data type. + */ +typedef bitmap<8, link_ac_attr_enum> link_ac_attr; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_AC_PARAM data type. + */ +struct link_ac_param { + /** + * Serialize/deserialize the LINK_AC_PARAM data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & param; + } + + boost::variant<null, flow_attribute, resource_desc> param; /**< Link action parameter.*/ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_ACTION data type. + */ +struct link_action { + link_ac_type type; /**< Link action type. */ + link_ac_attr attr; /**< Link action attribute. */ + link_ac_param param;/**< Link action parameter. */ + + /** + * Serialize/deserialize the LINK_ACTION data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + {; + ar & type; + ar & attr; + ar & param; + } +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * TH_ACTION data type enumeration. + */ +enum th_action_enum { + th_action_normal = 0, /**< Set normal threshold. */ + th_action_one_shot = 1, /**< Set one-shot threshold. */ + th_action_cancel = 2 /**< Cancel threshold. */ +}; + +/** + * TH_ACTION data type. + */ +typedef enumeration<th_action_enum> th_action; + + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_DN_REASON data type enumeration. + */ +enum link_dn_reason_enum { + link_dn_reason_explicit_disconnect = 0, /**< Explicit disconnect. */ + link_dn_reason_packet_timeout = 1, /**< Packet timeout. */ + link_dn_reason_no_resource = 2, /**< No resource. */ + link_dn_reason_no_broadcast = 3, /**< No broadcast. */ + link_dn_reason_authentication_failure = 4, /**< Authentication failure. */ + link_dn_reason_billing_failure = 5, /**< Billing failure. */ +}; + +/** + * LINK_DN_REASON data type. + */ +typedef enumeration<link_dn_reason_enum> link_dn_reason; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_GD_REASON data type enumeration. + */ +enum link_gd_reason_enum { + link_gd_reason_explicit_disconnect = 0, /**< Explicit disconnect. */ + link_gd_reason_link_parameter_degrading = 1, /**< Link parameter degrading. */ + link_gd_reason_low_power = 2, /**< Low power. */ + link_gd_reason_no_resource = 3, /**< No resource. */ +}; + +/** + * LINK_GD_REASON data type. + */ +typedef enumeration<link_gd_reason_enum> link_gd_reason; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_MIHCAP_FLAG data type enumeration. + */ +enum link_mihcap_flag_enum { + link_mihcap_event_service = 1, /**< Event service. */ + link_mihcap_command_service = 2, /**< Command service. */ + link_mihcap_information_service = 3, /**< Information service. */ +}; + +/** + * LINK_MIHCAP_FLAG data type. + */ +typedef bitmap<8, link_mihcap_flag_enum> link_mihcap_flag; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_GEN data type enumeration. + */ +enum link_param_gen_enum { + link_param_gen_data_rate = 0, /**< Data rate. */ + link_param_gen_signal_strength = 1, /**< Signal strength. */ + link_param_gen_sinr = 2, /**< SINR. */ + link_param_gen_throughput = 3, /**< Throughput. */ + link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ +}; + +/** + * LINK_PARAM_GEM data type. + */ +typedef enumeration<link_param_gen_enum> link_param_gen; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_QOS data type. + */ +typedef uint8 link_param_qos; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_GG data type enumeration. + */ +enum link_param_gg_enum { + link_param_gg_rx_qual = 0, /**< RxQual. */ + link_param_gg_rs_lev = 1, /**< RsLev. */ + link_param_gg_mean_bep = 2, /**< Mean BEP. */ + link_param_gg_st_dev_bep = 3, /**< StDev BEP. */ +}; + +/** + * LINK_PARAM_GG data type. + */ +typedef enumeration<link_param_gg_enum> link_param_gg; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_EDGE data type enumeration. + */ +enum link_param_edge_enum { +}; + +/** + * LINK_PARAM_EDGE data type. + */ +typedef enumeration<link_param_edge_enum> link_param_edge; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_ETH data type enumeration. + */ +enum link_param_eth_enum { +}; + +/** + * LINK_PARAM_ETH data type. + */ +typedef enumeration<link_param_eth_enum> link_param_eth; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_802_11 data type enumeration. + */ +enum link_param_802_11_enum { + link_param_802_11_rssi = 0, /**< RSSI. */ + link_param_802_11_no_qos = 1, /**< No QoS resource available. */ + link_param_802_11_multicast_packet_loss_rate = 2, /**< Multicast packet loss rate.*/ +}; + +/** + * LINK_PARAM_802_11 data type. + */ +typedef enumeration<link_param_802_11_enum> link_param_802_11; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_C2K data type enumeration. + */ +enum link_param_c2k_enum { + link_param_c2k_pilot_strength = 0, /**< Pilot strength. */ +}; + +/** + * LINK_PARAM_C2K data type. + */ +typedef enumeration<link_param_c2k_enum> link_param_c2k; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_FFD data type enumeration. + */ +enum link_param_ffd_enum { + link_param_ffd_cpich_rscp = 0, /**< CPICH RSCP. */ + link_param_ffd_pccpch_rscp = 1, /**< PCCPCH RSCP. */ + link_param_ffd_ultra_carrie_rssi = 2, /**< UTRA carrier RSSI. */ + link_param_ffd_gsm_carrie_rssi = 3, /**< GSM carrier RSSI. */ + link_param_ffd_cpich_ec_no = 4, /**< CPICH Ec/No. */ + link_param_ffd_transport_channel_bler = 5, /**< Transport channel BLER.*/ + link_param_ffd_ue = 6, /**< UE transmitted power. */ +}; + +/** + * LINK_PARAM_FFD data type. + */ +typedef enumeration<link_param_ffd_enum> link_param_ffd; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_HRPD data type enumeration. + */ +enum link_param_hrpd_enum { + link_param_hrpd_pilot_strength = 0, /**< HRPD pilot strength. */ +}; + +/** + * LINK_PARAM_HRPD data type. + */ +typedef enumeration<link_param_hrpd_enum> link_param_hrpd; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_802_16 data type enumeration. + */ +enum link_param_802_16_enum { +}; + +/** + * LINK_PARAM_802_16 data type. + */ +typedef enumeration<link_param_802_16_enum> link_param_802_16; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_802_20 data type enumeration. + */ +enum link_param_802_20_enum { +}; + +/** + * LINK_PARAM_802_20 data type. + */ +typedef enumeration<link_param_802_20_enum> link_param_802_20; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_802_22 data type enumeration. + */ +enum link_param_802_22_enum { +}; + +/** + * LINK_PARAM_802_22 data type. + */ +typedef enumeration<link_param_802_22_enum> link_param_802_22; + + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_lte data type enumeration. + */ +enum link_param_lte_enum { + link_param_lte_rsrp = 0, /**< RSSI. */ + link_param_lte_rsrq = 1, /**< No QoS resource available. */ + link_param_lte_cqi = 2, /**< Multicast packet loss rate.*/ + link_param_lte_bandwidth = 3, /**< System Load. */ + link_param_lte_pkt_delay = 4, /**< Number of registered users. */ + link_param_lte_pkt_loss = 5, /**< Number of active users. */ + link_param_lte_l2_buffer = 6, /**< Congestion windows of users. */ + link_param_lte_MN_cap = 7, /**< Congestion windows of users. */ + link_param_lte_embms = 8, /**< Congestion windows of users. */ + link_param_lte_jumbo_feasibility = 9, /**< Congestion windows of users. */ + link_param_lte_jumbo_setup = 10, /**< Congestion windows of users. */ + link_param_lte_active_embms = 11, /**< Transmission rate of users. */ + link_param_lte_link_congestion = 12, /**< Link congestion. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_LTE data type. + */ +typedef enumeration<link_param_lte_enum> link_param_lte; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_TUPLE_ID data type. + */ +class link_tuple_id : public link_id { +public: + /** + * Construct an empty LINK_TUPLE_ID data type. + */ + link_tuple_id() : poa_addr(null()) + { } + + /** + * Serialize/deserialize the LINK_TUPLE_ID data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + link_id::serialize(ar); + ar & poa_addr; + } + + /** + * Check if the LINK_TUPLE_ID data type is equal to another LINK_TUPLE_ID. + * + * @param other The LINK_TUPLE_ID to compare to. + * @return True if they are equal or false otherwise. + */ + bool operator==(const link_tuple_id& other) const + { + return ((static_cast<const link_id&>(*this) == static_cast<const link_id&>(other)) + && (addr == other.addr)); + } + + boost::variant<null, link_addr> poa_addr; /**< PoA link address. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_SCAN_RSP data type. + */ +struct link_scan_rsp { + link_addr id; /**< Link address. */ + octet_string net_id; /**< Network identifier. */ + sig_strength signal; /**< Signal strength. */ + + /** + * Serialize/deserialize the LINK_SCAN_RSP data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & net_id; + ar & signal; + } +}; + +/** + * LIST(LINK_SCAN_RSP) data type. + */ +typedef std::vector<link_scan_rsp> link_scan_rsp_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_ACTION_REQ data type. + */ +struct link_action_req { + link_id id; /**< Link identifier. */ + boost::variant<null, link_addr> addr; /**< PoA link address. */ + link_action action; /**< Link action. */ + link_ac_ex_time ex_time; /**< Link action execution time.*/ + + /** + * Serialize/deserialize the LINK_ACTION_REQ data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & addr; + ar & action; + ar & ex_time; + } +}; + +/** + * LIST(LINK_ACTION_REQ) data type. + */ +typedef std::vector<link_action_req> link_action_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_ACTION_RSP data type. + */ +struct link_action_rsp { + link_id id; /**< Link identifier. */ + link_ac_result result; /**< Link action result.*/ + boost::variant<null, link_scan_rsp_list> scan_list; /**< Link action response list.*/ + + /** + * Serialize/deserialize the LINK_ACTION_RSP data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & result; + ar & scan_list; + } +}; + +/** + * LIST(LINK_ACTION_RSP) data type. + */ +typedef std::vector<link_action_rsp> link_action_rsp_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_EVT_LIST data type enumeration. + */ +enum link_evt_list_enum { + evt_link_detected = 0, /**< Link detected */ + evt_link_up = 1, /**< Link up */ + evt_link_down = 2, /**< Link down */ + evt_link_parameters_report = 3, /**< Link parameters report */ + evt_link_going_down = 4, /**< Link doing down */ + evt_link_handover_imminent = 5, /**< Link handover imminent */ + evt_link_handover_complete = 6, /**< Link handover complete */ + evt_link_pdu_transmit_status = 7, /**< Link PDU transmit status */ +}; + +/** + * LINK_EVT_LIST data type. + */ +typedef bitmap<32, link_evt_list_enum> link_evt_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_CMD_LIST data type enumeration. + */ +enum link_cmd_list_enum { + cmd_link_event_subscribe = 1, /**< Event subscribe. */ + cmd_link_event_unsubscribe = 2, /**< Event unsubscribe. */ + cmd_link_get_parameters = 3, /**< Get parameters. */ + cmd_link_configure_thresholds = 4, /**< Configure thresholds. */ + cmd_link_action = 5, /**< Action. */ +}; + +/** + * LINK_CMD_LIST data type. + */ +typedef bitmap<32, link_cmd_list_enum> link_cmd_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_DET_CFG data type. + */ +struct link_det_cfg { + /** + * Serialize/deserialize the LINK_DET_CFG data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & network_id; + ar & signal; + ar & link_data_rate; + } + + boost::variant<null, octet_string> network_id; /**< Network identifier.*/ + boost::variant<null, sig_strength> signal; /**< Signal strength. */ + boost::variant<null, uint32> link_data_rate; /**< Link data rate. */ +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_TYPE data type. + */ +typedef boost::variant<link_param_gen, + link_param_qos, + link_param_gg, + link_param_edge, + link_param_eth, + link_param_802_11, + link_param_c2k, + link_param_ffd, + link_param_hrpd, + link_param_802_16, + link_param_802_20, + link_param_802_22, + link_param_lte> link_param_type; + +/** + * LIST(LINK_PARAM_TYPE) data type. + */ +typedef std::vector<link_param_type> link_param_type_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_DESC_REQ_LIST data type enumeration. + */ +enum link_desc_req_enum { + link_desc_req_classes_of_service_supported = 0, /**< Number of classes of services supported.*/ + link_desc_req_queues_supported = 1, /**< Number of queues supported.*/ +}; + +/** + * LINK_DESC_REQ data type. + */ +typedef bitmap<16, link_desc_req_enum> link_desc_req; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_STATUS_REQ data type. + */ +struct link_status_req { + link_states_req _states_req; /**< Link states to be requested.*/ + link_param_type_list _param_type_list; /**< Link parameter type list. */ + link_desc_req _desc_req; /**< Link dsecriptors. */ + + /** + * Serialize/deserialize the LINK_STATUS_REQ data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & _states_req; + ar & _param_type_list; + ar & _desc_req; + } +}; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_VAL data type. + */ +typedef uint16 link_param_val; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM data type. + */ +struct link_param { + /** + * Serialize/deserialize the LINK_PARAM data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & type; + ar & value; + } + + link_param_type type; /**< Link parameter type. */ + boost::variant<link_param_val, + qos_param_val> value; /**< Link parameter value. */ +}; + +/** + * LINK_PARAM_LIST data type. + */ +typedef std::vector<link_param> link_param_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_PARAM_RPT data type. + */ +struct link_param_report { + /** + * Construct a LINK_PARAM_RPT data type. + */ + link_param_report() : thold(null()) + { } + + /** + * Serialize/deserialize the LINK_PARAM_RPT data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & param; + ar & thold; + } + + + link_param param; /**< Link parameter. */ + boost::variant<null, threshold> thold; /**< Threshold. */ +}; + +/** + * LIST(LINK_PARAM_RPT) data type. + */ +typedef std::vector<link_param_report> link_param_rpt_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_CFG_PARAM data type. + */ +struct link_cfg_param { + /** + * Serialize/deserialize the LINK_CFG_PARAM data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & type; + ar & timer_interval; + ar & action; + ar & threshold_list; + } + + link_param_type type; /**< Link parameter type.*/ + boost::variant<null, uint16> timer_interval; /**< Timer interval. */ + th_action action; /**< Action to apply to the listed thresholds.*/ + std::vector<threshold> threshold_list; /**< Thresholds. */ +}; + +/** + * LIST(LINK_CFG_PARAM) data type. + */ +typedef std::vector<link_cfg_param> link_cfg_param_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_CFG_STATUS data type. + */ +struct link_cfg_status { + link_param_type type; /**< Link parameter type. */ + threshold thold; /**< Threshold. */ + config_status status; /**< Link parameter configuration status. */ + + /** + * Serialize/deserialize the LINK_CFG_STATUS data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & type; + ar & thold; + ar & status; + } +}; + +/** + * LIST(LINK_CFG_STATUS) data type. + */ +typedef std::vector<link_cfg_status> link_cfg_status_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * NET_CAPS data type enumeration. + */ +enum net_caps_enum { + net_caps_security = 0, /**< Security. */ + net_caps_qos_0 = 1, /**< QoS Class 0. */ + net_caps_qos_1 = 2, /**< QoS Class 1. */ + net_caps_qos_2 = 3, /**< QoS Class 2. */ + net_caps_qos_3 = 4, /**< QoS Class 3. */ + net_caps_qos_4 = 5, /**< QoS Class 4. */ + net_caps_qos_5 = 6, /**< QoS Class 5. */ + net_caps_internet = 7, /**< Internet access. */ + net_caps_emergency_services = 8, /**< Emergency services. */ + net_caps_mih = 9, /**< MIH capability. */ +}; + +/** + * NET_CAPS data type. + */ +typedef bitmap<32, net_caps_enum> net_caps; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_DET_INFO data type. + */ +struct link_det_info { + /** + * Serialize/deserialize the LINK_DET_INFO data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & network_id; + ar & net_aux_id; + ar & signal; + ar & sinr; + ar & data_rate; + ar & mih_capabilities; + ar & net_capabilities; + } + + link_tuple_id id; /**< Link tuple identifier. */ + octet_string network_id; /**< Network identifier. */ + octet_string net_aux_id; /**< Auxiliar network identifier. */ + sig_strength signal; /**< Signal strength. */ + uint16 sinr; /**< SINR. */ + uint32 data_rate; /**< Link data rate. */ + link_mihcap_flag mih_capabilities; /**< MIH capabilities. */ + net_caps net_capabilities; /**< Network capabilities. */ +}; + +/** + * LIST(LINK_DET_INFO) data type. + */ +typedef std::vector<link_det_info> link_det_info_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_POA_LIST data type. + */ +struct link_poa_list { + /** + * Serialize/deserialize the LINK_POA_LIST data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & addr_list; + } + + link_id id; /**< Link identifier. */ + link_addr_list addr_list; /**< Link address list. */ +}; + +/** + * LIST(LINK_POA_LIST) data type. + */ +typedef std::vector<link_poa_list> list_of_link_poa_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * LINK_STATUS_RSP data type. + */ +struct link_status_rsp { + /** + * Serialize/deserialize the LINK_STATUS_RSP data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & states_rsp_list; + ar & param_list; + ar & desc_rsp_list; + } + + + link_states_rsp_list states_rsp_list; /**< Link states response list. */ + link_param_list param_list; /**< Link parameter list. */ + link_desc_rsp_list desc_rsp_list; /**< Link descriptors response list.*/ +}; + +/** + * LIST(LINK_STATUS_RSP) data type. + */ +typedef std::vector<link_status_rsp> link_status_rsp_list; + +/////////////////////////////////////////////////////////////////////////////// +/** + * SEQUENCE(LINK_ID,LINK_STATUS_RSP) data type. + */ +struct status_rsp { + /** + * Serialize/deserialize the SEQUENCE(LINK_ID,LINK_STATUS_RSP) data type. + * + * @param ar The archive to/from where serialize/deserialize the data type. + */ + template<class ArchiveT> + void serialize(ArchiveT& ar) + { + ar & id; + ar & rsp; + } + + link_id id; /**< Link identifier. */ + link_status_rsp rsp; /**< Link status response. */ +}; + +/** + * LIST(SEQUENCE(LINK_ID,LINK_STATUS_RSP)) data type. + */ +typedef std::vector<status_rsp> status_rsp_list; + +/////////////////////////////////////////////////////////////////////////////// +} /* namespace mih */ } /*namespace odtone */ + +// EOF //////////////////////////////////////////////////////////////////////// +#endif /* ODTONE_MIH_TYPES_LINK__HPP_ */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/link_book.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/link_book.cpp new file mode 100644 index 0000000000000000000000000000000000000000..271112ece2630ea47eecdcc14d468ee33cbe89ea --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/link_book.cpp @@ -0,0 +1,290 @@ +//============================================================================== +// Brief : Link Book +// Authors : Carlos Guimarães <cguimaraes@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "link_book.hpp" +#include "log.hpp" + +/////////////////////////////////////////////////////////////////////////////// + +namespace odtone { namespace mihf { + +//----------------------------------------------------------------------------- +static std::string evt2string(mih::link_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(mih::evt_link_detected)) s += "DETECTED "; + if(evtP.get(mih::evt_link_up)) s += "UP "; + if(evtP.get(mih::evt_link_down)) s += "DOWN "; + if(evtP.get(mih::evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(mih::evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(mih::evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(mih::evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(mih::evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +enum link_cmd_list_enum { + cmd_link_event_subscribe = 1, /**< Event subscribe. */ + cmd_link_event_unsubscribe = 2, /**< Event unsubscribe. */ + cmd_link_get_parameters = 3, /**< Get parameters. */ + cmd_link_configure_thresholds = 4, /**< Configure thresholds. */ + cmd_link_action = 5, /**< Action. */ +}; +//----------------------------------------------------------------------------- +static std::string cmd2string(mih::link_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(mih::cmd_link_event_subscribe)) s += "EVENT_SUBSCRIBE "; + if(cmdP.get(mih::cmd_link_event_unsubscribe)) s += "EVENT_UNSUBSCRIBE"; + if(cmdP.get(mih::cmd_link_get_parameters)) s += "GET_PARAMETERS "; + if(cmdP.get(mih::cmd_link_configure_thresholds)) s += "CONFIGURE_THRESHOLDS "; + if(cmdP.get(mih::cmd_link_action)) s += "ACTION "; + return s; +} +/** + * Add a new Link SAP entry in the link book. + * + * @param id Link SAP MIH Identifier. + * @param ip Link SAP IP address. + * @param port Link SAP listening port. + * @param link_id interfaces that Link SAP manages. + */ +void link_book::add(const mih::octet_string &id, + mih::octet_string& ip, + uint16 port, + mih::link_id link_id) +{ + boost::mutex::scoped_lock lock(_mutex); + // TODO: add thread safety + link_entry a; + + a.ip.assign(ip); + a.port = port; + a.link_id = link_id; + a.fail = 0; + a.status = true; + + _lbook[id] = a; + ODTONE_LOG(4, "(link_book) added: ", id, " ", ip, " ", port, " ", link_id); +} + +/** + * Set the IP address of an existing Link SAP entry. + * + * @param id Link SAP MIH Identifier. + * @param ip The IP address to set. + */ +void link_book::set_ip(const mih::octet_string &id, std::string ip) +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + + if (it != _lbook.end()) + it->second.ip = ip; +} + +/** + * Set the port of an existing Link SAP entry. + * + * @param id Link SAP MIH Identifier. + * @param port The port to set. + */ +void link_book::set_port(const mih::octet_string &id, uint16 port) +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + + if (it != _lbook.end()) + it->second.port = port; +} + +/** + * Update the events and commands supported by a Link SAP. + * + * @param id Link SAP MIH Identifier. + * @param event_list Supported event list. + * @param cmd_list Supported command list. + */ +void link_book::update_capabilities(const mih::octet_string &id, + mih::link_evt_list event_list, + mih::link_cmd_list cmd_list) + +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + + if (it != _lbook.end()) { + ODTONE_LOG(4, "(link_book) update_capabilities() FOUND: ", id, " ", evt2string(event_list), " ", cmd2string(cmd_list)); + it->second.event_list = event_list; + it->second.cmd_list = cmd_list; + } else { + ODTONE_LOG(4, "(link_book) update_capabilities() NOT FOUND: ", id, " ", evt2string(event_list), " ", cmd2string(cmd_list)); + } +} + +/** + * Remove an existing Link SAP entry. + * + * @param id Link SAP MIH Identifier. + */ +void link_book::del(mih::octet_string &id) +{ + boost::mutex::scoped_lock lock(_mutex); + + _lbook.erase(id); +} + +/** + * Inactive an existing Link SAP entry. + * + * @param id Link SAP MIH Identifier. + */ +void link_book::inactive(mih::octet_string &id) +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + + if (it != _lbook.end()) { + ODTONE_LOG(4, "(link_book) inactive() : ", id); + it->second.status = false; + } +} + +/** + * Get the record for a given Link SAP. + * + * @param id Link SAP MIH Identifier. + * @return The record for a given Link SAP. + */ +const link_entry& link_book::get(const mih::octet_string &id) +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::const_iterator it; + it = _lbook.find(id); + + if (it == _lbook.end()) + boost::throw_exception(unknown_link_sap()); + + return it->second; +} + +/** + * Get the list of all known Link SAPs. + * + * @return The list of all known Link SAPs. + */ +const std::vector<mih::octet_string> link_book::get_ids() +{ + boost::mutex::scoped_lock lock(_mutex); + + std::vector<mih::octet_string> ids; + for(std::map<mih::octet_string, link_entry>::iterator it = _lbook.begin(); it != _lbook.end(); it++) { + ids.push_back(it->first); + } + + return ids; +} + +/** + * Search for the Link SAP MIH Identifier of a given interface. + * + * @param lt The link type of the Link SAP to search for. + * @param la The link address of the Link SAP to search for. + * @return The Link SAP MIH Identifier. + */ +const mih::octet_string link_book::search_interface(mih::link_type lt, mih::link_addr la) +{ + boost::mutex::scoped_lock lock(_mutex); + + mih::octet_string id; + std::map<mih::octet_string, link_entry>::iterator it; + for(it = _lbook.begin(); it != _lbook.end(); it++) { + + ODTONE_LOG(4, "(link_book) ID LINK TYPE : ", it->first, " ADDRESS ", it->second.link_id.addr, " STATUS", it->second.status); + } + + for(/*std::map<mih::octet_string, link_entry>::iterator*/ it = _lbook.begin(); it != _lbook.end(); it++) { + + ODTONE_LOG(4, "(link_book) search 1st step ID LINK TYPE comparison type:",it->second.link_id.type, "LT", lt); + if(it->second.link_id.type == lt) { + ODTONE_LOG(4, "(link_book) search 2nd step ID LINK TYPE comparison addr:",it->second.link_id.addr, "LA", la); + if(it->second.link_id.addr == la) { + ODTONE_LOG(4, "(link_book) search 3rd step ID LINK TYPE check status:",it->second.status); + if(it->second.status) { + ODTONE_LOG(4, "(link_book) search 3rd step ID LINK TYPE : ", it->first, " ADDRESS ", it->second.link_id.addr, " STATUS", it->second.status); + id = it->first; + break; + } + } + } + } + + ODTONE_LOG(4, "(link_book) search_interface() : ", lt, " ", la, " Found:", id); + return id; +} + +/** + * Update and return the number of fail responses of a given Link SAP. + * + * @param id Link SAP MIH Identifier. + * @return The number of fails responses. + */ +uint16 link_book::fail(const mih::octet_string &id) +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + + if (it == _lbook.end()) + boost::throw_exception(unknown_link_sap()); + + (it->second.fail)++; + ODTONE_LOG(4, "(link_book) fail() : ", id ," num fails:",it->second.fail); + return it->second.fail; +} + +/** + * Reset the number of fail responses of a given Link SAP. + * + * @param id Link SAP MIH Identifier. + */ +void link_book::reset(const mih::octet_string &id) +{ + boost::mutex::scoped_lock lock(_mutex); + + std::map<mih::octet_string, link_entry>::iterator it; + it = _lbook.find(id); + + if (it == _lbook.end()) + boost::throw_exception(unknown_link_sap()); + + ODTONE_LOG(4, "(link_book) reset() : ", id); + it->second.fail = 0; + it->second.status = true; +} + +} /* namespace mihf */ } /* namespace odtone */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/main.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/main.cpp new file mode 100644 index 0000000000000000000000000000000000000000..46a9db9dc56b6862a127d4bb75cd4201a889347d --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/main.cpp @@ -0,0 +1,753 @@ +//============================================================================== +// Brief : ODTONE MIHF +// Authors : Simao Reis <sreis@av.it.pt> +// Carlos Guimarães <cguimaraes@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// + +#include "mihfid.hpp" +#include "log.hpp" +// #include "transaction_ack_service.hpp" +// #include "transaction_manager.hpp" +#include "address_book.hpp" +#include "link_book.hpp" +#include "user_book.hpp" +#include "local_transaction_pool.hpp" +#include "transaction_pool.hpp" + +#include "net_sap.hpp" +#include "message_out.hpp" +#include "transmit.hpp" +#include "service_management.hpp" +#include "event_service.hpp" +#include "command_service.hpp" +#include "information_service.hpp" +#include "service_access_controller.hpp" + +#include "message_in.hpp" +#include "udp_listener.hpp" +#include "tcp_listener.hpp" + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/mih/config.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/types/capabilities.hpp> + +#include <list> +#include <iostream> + +#include <boost/algorithm/string.hpp> +#include <boost/program_options.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/asio.hpp> +/////////////////////////////////////////////////////////////////////////////// +// using namespace odtone::mihf; +using namespace odtone; +using namespace odtone::mihf; + + +namespace po = boost::program_options; + +// available config options +static const char* const kConf_File = "conf.file"; +static const char* const kConf_Receive_Buffer_Len = "conf.recv_buff_len"; +static const char* const kConf_MIHF_Id = "mihf.id"; +static const char* const kConf_MIHF_Ip = "mihf.ip"; +static const char* const kConf_MIHF_Peer_List = "mihf.peers"; +static const char* const kConf_MIHF_Users_List = "mihf.users"; +static const char* const kConf_MIHF_Links_List = "mihf.links"; +static const char* const kConf_MIHF_Transport_List = "mihf.transport"; +static const char* const kConf_MIHF_Remote_Port = "mihf.remote_port"; +static const char* const kConf_MIHF_Local_Port = "mihf.local_port"; +static const char* const kConf_MIHF_Link_Response_Time = "mihf.link_response_time"; +static const char* const kConf_MIHF_Link_Delete = "mihf.link_delete"; +static const char* const kConf_MIHF_Discover = "mihf.discover"; +static const char* const kConf_MIHF_Multicast = "enable_multicast"; +static const char* const kConf_MIHF_Unsolicited = "enable_unsolicited"; +static const char* const kConf_MIHF_Verbosity = "log"; + +uint16 kConf_MIHF_Link_Response_Time_Value; +uint16 kConf_MIHF_Link_Delete_Value; + +/** + * Remove a character from the string. + */ +void __trim(mih::octet_string &str, const char chr) +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} + +// +// list is a comma separated list of mihf id ip and port +// +// example: mihf2 192.168.0.1 4551 +// +void set_list_peer_mihfs(mih::octet_string &list, address_book &abook) +{ + using namespace boost; + + char_separator<char> sep1(","); + char_separator<char> sep2(" "); + tokenizer< char_separator<char> > list_tokens(list, sep1); + + BOOST_FOREACH(mih::octet_string str, list_tokens) + { + tokenizer< char_separator<char> > tokens(str, sep2); + tokenizer< char_separator<char> >::iterator it = tokens.begin(); + + mih::octet_string id = *it; + ++it; + mih::octet_string ip = *it; + ++it; + mih::octet_string port = *it; + + uint16 port_; + std::istringstream iss(port); + if ((iss >> port_).fail()) + throw "invalid port"; + + mih::transport_list trans; + + std::map<std::string, odtone::mih::transport_list_enum> enum_map; + enum_map["udp"] = odtone::mih::transport_udp; + enum_map["tcp"] = odtone::mih::transport_tcp; + + while(it != tokens.end()) { + if(enum_map.find(*it) != enum_map.end()) + trans.set(odtone::mih::transport_list_enum(enum_map[*it])); + ++it; + } + + address_entry entry; + entry.ip = ip; + entry.port = port_; + entry.capabilities_trans_list = trans; + + abook.add(id, entry); + } +} + +void set_users(mih::octet_string &list, user_book &ubook) +{ + using namespace boost; + + char_separator<char> sep1(","); + char_separator<char> sep2(" "); + tokenizer< char_separator<char> > list_tokens(list, sep1); + + BOOST_FOREACH(mih::octet_string str, list_tokens) { + tokenizer< char_separator<char> > tokens(str, sep2); + tokenizer< char_separator<char> >::iterator it = tokens.begin(); + + mih::octet_string id = *it; + ++it; + mih::octet_string ip("127.0.0.1"); + mih::octet_string port = *it; + ++it; + + uint16 port_; + std::istringstream iss_port(port); + if ((iss_port >> port_).fail()) + throw "invalid port"; + + mih::mih_cmd_list supp_cmd_tmp; + mih::iq_type_list supp_iq_tmp; + bool has_cmd = false; + bool has_iq = false; + if(it != tokens.end()) { + std::map<std::string, mih::mih_cmd_list_enum> enum_map_cmd; + enum_map_cmd["mih_link_get_parameters"] = mih::mih_cmd_link_get_parameters; + enum_map_cmd["mih_link_configure_thresholds"] = mih::mih_cmd_link_configure_thresholds; + enum_map_cmd["mih_link_actions"] = mih::mih_cmd_link_actions; + enum_map_cmd["mih_net_ho_candidate_query"] = mih::mih_cmd_net_ho_candidate_query; + enum_map_cmd["mih_net_ho_commit"] = mih::mih_cmd_net_ho_commit; + enum_map_cmd["mih_n2n_ho_query_resources"] = mih::mih_cmd_n2n_ho_query_resources; + enum_map_cmd["mih_n2n_ho_commit"] = mih::mih_cmd_n2n_ho_commit; + enum_map_cmd["mih_n2n_ho_complete"] = mih::mih_cmd_n2n_ho_complete; + enum_map_cmd["mih_mn_ho_candidate_query"] = mih::mih_cmd_mn_ho_candidate_query; + enum_map_cmd["mih_mn_ho_commit"] = mih::mih_cmd_mn_ho_commit; + enum_map_cmd["mih_mn_ho_complete"] = mih::mih_cmd_mn_ho_complete; + + std::map<std::string, mih::iq_type_list_enum> enum_map_iq; + enum_map_iq["iq_type_binary_data"] = mih::iq_type_binary_data; + enum_map_iq["iq_type_rdf_data"] = mih::iq_type_rdf_data; + enum_map_iq["iq_type_rdf_schema_url"] = mih::iq_type_rdf_schema_url; + enum_map_iq["iq_type_rdf_schema"] = mih::iq_type_rdf_schema; + enum_map_iq["iq_type_ie_network_type"] = mih::iq_type_ie_network_type; + enum_map_iq["iq_type_ie_operator_id"] = mih::iq_type_ie_operator_id; + enum_map_iq["iq_type_ie_service_provider_id"] = mih::iq_type_ie_service_provider_id; + enum_map_iq["iq_type_ie_country_code"] = mih::iq_type_ie_country_code; + enum_map_iq["iq_type_ie_network_id"] = mih::iq_type_ie_network_id; + enum_map_iq["iq_type_ie_network_aux_id"] = mih::iq_type_ie_network_aux_id; + enum_map_iq["iq_type_ie_roaming_parteners"] = mih::iq_type_ie_roaming_parteners; + enum_map_iq["iq_type_ie_cost"] = mih::iq_type_ie_cost; + enum_map_iq["iq_type_ie_network_qos"] = mih::iq_type_ie_network_qos; + enum_map_iq["iq_type_ie_network_data_rate"] = mih::iq_type_ie_network_data_rate; + enum_map_iq["iq_type_ie_net_regult_domain"] = mih::iq_type_ie_net_regult_domain; + enum_map_iq["iq_type_ie_net_frequency_bands"] = mih::iq_type_ie_net_frequency_bands; + enum_map_iq["iq_type_ie_net_ip_cfg_methods"] = mih::iq_type_ie_net_ip_cfg_methods; + enum_map_iq["iq_type_ie_net_capabilities"] = mih::iq_type_ie_net_capabilities; + enum_map_iq["iq_type_ie_net_supported_lcp"] = mih::iq_type_ie_net_supported_lcp; + enum_map_iq["iq_type_ie_net_mob_mgmt_prot"] = mih::iq_type_ie_net_mob_mgmt_prot; + enum_map_iq["iq_type_ie_net_emserv_proxy"] = mih::iq_type_ie_net_emserv_proxy; + enum_map_iq["iq_type_ie_net_ims_proxy_cscf"] = mih::iq_type_ie_net_ims_proxy_cscf; + enum_map_iq["iq_type_ie_net_mobile_network"] = mih::iq_type_ie_net_mobile_network; + enum_map_iq["iq_type_ie_poa_link_addr"] = mih::iq_type_ie_poa_link_addr; + enum_map_iq["iq_type_ie_poa_location"] = mih::iq_type_ie_poa_location; + enum_map_iq["iq_type_ie_poa_channel_range"] = mih::iq_type_ie_poa_channel_range; + enum_map_iq["iq_type_ie_poa_system_info"] = mih::iq_type_ie_poa_system_info; + enum_map_iq["iq_type_ie_poa_subnet_info"] = mih::iq_type_ie_poa_subnet_info; + enum_map_iq["iq_type_ie_poa_ip_addr"] = mih::iq_type_ie_poa_ip_addr; + + while(it != tokens.end()) { + if(enum_map_cmd.find(*it) != enum_map_cmd.end()) { + supp_cmd_tmp.set((mih::mih_cmd_list_enum) enum_map_cmd[*it]); + has_cmd = true; + } else if(enum_map_iq.find(*it) != enum_map_iq.end()) { + supp_iq_tmp.set((mih::iq_type_list_enum) enum_map_iq[*it]); + has_iq = true; + } + + ++it; + } + } + + boost::optional<mih::mih_cmd_list> supp_cmd; + boost::optional<mih::iq_type_list> supp_iq; + + if(has_cmd) { + supp_cmd = supp_cmd_tmp; + } else { + supp_cmd_tmp.full(); + supp_cmd = supp_cmd_tmp; + } + + if(has_iq) + supp_iq = supp_iq_tmp; + + ubook.add(id, ip, port_, supp_cmd, supp_iq); + } +} + +void set_links(mih::octet_string &list, link_book &lbook) +{ + using namespace boost; + + char_separator<char> sep1(","); + char_separator<char> sep2(" "); + tokenizer< char_separator<char> > list_tokens(list, sep1); + + BOOST_FOREACH(mih::octet_string str, list_tokens) { + tokenizer< char_separator<char> > tokens(str, sep2); + tokenizer< char_separator<char> >::iterator it = tokens.begin(); + + mih::octet_string id = *it; + ++it; + mih::octet_string port = *it; + ++it; + mih::octet_string tec = *it; + ++it; + mih::octet_string address = *it; + mih::octet_string ip("127.0.0.1"); + + uint16 port_; + std::istringstream iss(port); + if ((iss >> port_).fail()) { + throw "invalid port"; + } + + if(tec == "" || address == "") { + throw "invalid technology and/or address"; + } + + // Extract technology + std::map<std::string, odtone::mih::link_type_enum> enum_map; + enum_map["GSM"] = odtone::mih::link_type_gsm; + enum_map["GPRS"] = odtone::mih::link_type_gprs; + enum_map["EDGE"] = odtone::mih::link_type_edge; + enum_map["802_3"] = odtone::mih::link_type_ethernet; + enum_map["Other"] = odtone::mih::link_type_wireless_other; + enum_map["802_11"] = odtone::mih::link_type_802_11; + enum_map["CDMA2000"] = odtone::mih::link_type_cdma2000; + enum_map["UMTS"] = odtone::mih::link_type_umts; + enum_map["CDMA2000-HRPD"] = odtone::mih::link_type_cdma2000_hrpd; + enum_map["LTE"] = odtone::mih::link_type_lte; + enum_map["802_16"] = odtone::mih::link_type_802_16; + enum_map["802_20"] = odtone::mih::link_type_802_20; + enum_map["802_22"] = odtone::mih::link_type_802_22; + + mih::link_id lid; + if(enum_map.find(tec) != enum_map.end()) + lid.type = odtone::mih::link_type(enum_map[tec]); + + // TODO: Parse the link address for all link types. + switch(lid.type.get()) { + case 1: + case 2: + case 3: { + throw "not supported yet"; + } + break; + + case 15: + case 19: + case 27: + case 28: + case 29: { + mih::mac_addr mac; + mac.address(address); + lid.addr = mac; + } break; + + case 18: + case 22: + case 23: + case 24: + throw "not supported yet"; + break; + + case 25: { + ++it; + mih::octet_string plmn = *it; + ++it; + mih::octet_string cell_id = *it; + + char_separator<char> plmn_sep(":"); + tokenizer< char_separator<char> > list_tokens(plmn, plmn_sep); + + mih::l2_3gpp_3g_cell_id lte; + uint8 pos = 0; + BOOST_FOREACH(mih::octet_string str, list_tokens) { + uint8 byte = 0; + std::istringstream iss(str); + iss >> std::hex >> byte; + + lte.plmn_id[pos] = byte % 0x100; + ++pos; + } + + std::istringstream iss(port); + if ((iss >> lte._cell_id).fail()) { + throw "invalid cell_id"; + } + + lid.addr = lte; + } break; + default: { + ODTONE_LOG(0, "Error: Invalid technology! Aborting...") ; + throw "invalid technology"; + } break; + } + + lbook.add(id, ip, port_, lid); + } +} + +void parse_mihf_information(mih::config &cfg, address_book &abook) +{ + using namespace boost; + + // Insert the MIHF itself in the address book + mih::octet_string id = cfg.get<mih::octet_string>(kConf_MIHF_Id); + mih::octet_string ip = cfg.get<mih::octet_string>(kConf_MIHF_Ip); + uint16 port = cfg.get<uint16>(kConf_MIHF_Remote_Port); + mih::transport_list trans; + + mih::octet_string tmp = cfg.get<mih::octet_string>(kConf_MIHF_Transport_List); + boost::algorithm::to_lower(tmp); + + char_separator<char> sep(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep); + + std::map<std::string, odtone::mih::transport_list_enum> enum_map; + enum_map["udp"] = odtone::mih::transport_udp; + enum_map["tcp"] = odtone::mih::transport_tcp; + + BOOST_FOREACH(mih::octet_string str, list_tokens) { + __trim(str, ' '); + if(enum_map.find(str) != enum_map.end()) + trans.set(odtone::mih::transport_list_enum(enum_map[str])); + } + + address_entry entry; + entry.ip = ip; + entry.port = port; + entry.capabilities_trans_list = trans; + + abook.add(id, entry); +} + +void parse_peer_registrations(mih::config &cfg, address_book &abook) +{ + mih::octet_string mihfs = cfg.get<mih::octet_string>(kConf_MIHF_Peer_List); + + set_list_peer_mihfs(mihfs, abook); +} + +void parse_sap_registrations(mih::config &cfg, user_book &ubook, link_book &lbook) +{ + mih::octet_string users = cfg.get<mih::octet_string>(kConf_MIHF_Users_List); + mih::octet_string lsaps = cfg.get<mih::octet_string>(kConf_MIHF_Links_List); + + set_users(users, ubook); + set_links(lsaps, lbook); +} + +std::vector<std::string> parse_discover_order(mih::config &cfg) +{ + using namespace boost; + + mih::octet_string order = cfg.get<mih::octet_string>(kConf_MIHF_Discover); + std::vector<std::string> list; + + char_separator<char> sep(","); + tokenizer< char_separator<char> > list_tokens(order, sep); + + BOOST_FOREACH(mih::octet_string str, list_tokens) { + list.push_back(str); + } + + return list; +} + +void sm_register_callbacks(service_management &sm) +{ + sac_register_callback(mih::request::capability_discover, + boost::bind(&service_management::capability_discover_request, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::response::capability_discover, + boost::bind(&service_management::capability_discover_response, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::confirm::capability_discover, + boost::bind(&service_management::capability_discover_confirm, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::request::mih_register, + boost::bind(&service_management::register_request, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::response::mih_register, + boost::bind(&service_management::register_response, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::request::mih_deregister, + boost::bind(&service_management::deregister_request, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::response::mih_deregister, + boost::bind(&service_management::deregister_response, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::indication::link_register, + boost::bind(&service_management::link_register_indication, + boost::ref(sm), _1, _2)); + + sac_register_callback(mih::indication::user_register, + boost::bind(&service_management::user_register_indication, + boost::ref(sm), _1, _2)); + +} + +void mies_register_callbacks(event_service &mies) +{ + sac_register_callback(mih::request::event_subscribe, + boost::bind(&event_service::event_subscribe_request, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::response::event_subscribe, + boost::bind(&event_service::event_subscribe_response, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::confirm::event_subscribe, + boost::bind(&event_service::event_subscribe_confirm, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_up, + boost::bind(&event_service::link_up_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_down, + boost::bind(&event_service::link_down_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_detected, + boost::bind(&event_service::link_detected_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_going_down, + boost::bind(&event_service::link_going_down_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_parameters_report, + boost::bind(&event_service::link_parameters_report_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_handover_imminent, + boost::bind(&event_service::link_handover_imminent_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::indication::link_handover_complete, + boost::bind(&event_service::link_handover_complete_indication, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::request::event_unsubscribe, + boost::bind(&event_service::event_unsubscribe_request, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::response::event_unsubscribe, + boost::bind(&event_service::event_unsubscribe_response, + boost::ref(mies), _1, _2)); + sac_register_callback(mih::confirm::event_unsubscribe, + boost::bind(&event_service::event_unsubscribe_confirm, + boost::ref(mies), _1, _2)); +} +// REGISTER(event_service::link_pdu_transmit_status_indication) + +void mics_register_callbacks(command_service &mics) +{ + sac_register_callback(mih::request::link_get_parameters, + boost::bind(&command_service::link_get_parameters_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::link_get_parameters, + boost::bind(&command_service::link_get_parameters_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::confirm::link_get_parameters, + boost::bind(&command_service::link_get_parameters_confirm, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::link_configure_thresholds, + boost::bind(&command_service::link_configure_thresholds_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::link_configure_thresholds, + boost::bind(&command_service::link_configure_thresholds_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::confirm::link_configure_thresholds, + boost::bind(&command_service::link_configure_thresholds_confirm, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::link_actions, + boost::bind(&command_service::link_actions_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::link_actions, + boost::bind(&command_service::link_actions_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::confirm::link_actions, + boost::bind(&command_service::link_actions_confirm, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::net_ho_candidate_query, + boost::bind(&command_service::net_ho_candidate_query_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::net_ho_candidate_query, + boost::bind(&command_service::net_ho_candidate_query_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::mn_ho_candidate_query, + boost::bind(&command_service::mn_ho_candidate_query_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::mn_ho_candidate_query, + boost::bind(&command_service::mn_ho_candidate_query_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::n2n_ho_query_resources, + boost::bind(&command_service::n2n_ho_query_resources_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::n2n_ho_query_resources, + boost::bind(&command_service::n2n_ho_query_resources_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::mn_ho_commit, + boost::bind(&command_service::mn_ho_commit_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::mn_ho_commit, + boost::bind(&command_service::mn_ho_commit_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::net_ho_commit, + boost::bind(&command_service::net_ho_commit_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::net_ho_commit, + boost::bind(&command_service::net_ho_commit_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::n2n_ho_commit, + boost::bind(&command_service::n2n_ho_commit_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::n2n_ho_commit, + boost::bind(&command_service::n2n_ho_commit_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::n2n_ho_commit, + boost::bind(&command_service::n2n_ho_commit_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::n2n_ho_commit, + boost::bind(&command_service::n2n_ho_commit_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::mn_ho_complete, + boost::bind(&command_service::mn_ho_complete_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::mn_ho_complete, + boost::bind(&command_service::mn_ho_complete_response, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::request::n2n_ho_complete, + boost::bind(&command_service::n2n_ho_complete_request, + boost::ref(mics), _1, _2)); + sac_register_callback(mih::response::n2n_ho_complete, + boost::bind(&command_service::n2n_ho_complete_response, + boost::ref(mics), _1, _2)); +} + +void miis_register_callbacks(information_service &miis) +{ + sac_register_callback(mih::request::get_information, + boost::bind(&information_service::get_information_request, + boost::ref(miis), _1, _2)); + sac_register_callback(mih::response::get_information, + boost::bind(&information_service::get_information_response, + boost::ref(miis), _1, _2)); + sac_register_callback(mih::request::push_information, + boost::bind(&information_service::push_information_request, + boost::ref(miis), _1, _2)); + sac_register_callback(mih::indication::push_information, + boost::bind(&information_service::push_information_indication, + boost::ref(miis), _1, _2)); +} + +int main(int argc, char **argv) +{ + odtone::setup_crash_handler(); + + boost::asio::io_service io; + + // declare MIHF supported options + po::options_description desc(mih::octet_string("MIHF Configuration Options")); + + desc.add_options() + ("help", "Display configuration options") + (kConf_File, po::value<std::string>()->default_value("odtone.conf"), "Configuration file") + (kConf_Receive_Buffer_Len, po::value<uint16>()->default_value(4096), "Receive buffer length") + (kConf_MIHF_Id, po::value<std::string>()->default_value("mihf"), "MIHF ID") + (kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "MIHF IP") + (kConf_MIHF_Remote_Port, po::value<uint16>()->default_value(4551), "Remote MIHF communication port") + (kConf_MIHF_Local_Port, po::value<uint16>()->default_value(1025), "Local SAPs communications port") + (kConf_MIHF_Peer_List, po::value<std::string>()->default_value(""), "List of peer MIHFs") + (kConf_MIHF_Users_List, po::value<std::string>()->default_value(""), "List of local MIH-Users") + (kConf_MIHF_Links_List, po::value<std::string>()->default_value(""), "List of local Links SAPs") + (kConf_MIHF_Transport_List, po::value<std::string>()->default_value("udp"), "List of supported transport protocols") + (kConf_MIHF_Link_Response_Time, po::value<uint16>()->default_value(7000), "Link SAP response time (milliseconds)") + (kConf_MIHF_Link_Delete, po::value<uint16>()->default_value(2), "Link SAP response fails to forget") + (kConf_MIHF_Discover, po::value<std::string>()->default_value(""), "MIHF Discovery Mechanisms Order") + (kConf_MIHF_Multicast, "Allows multicast messages") + (kConf_MIHF_Unsolicited, "Allows unsolicited discovery") + (kConf_MIHF_Verbosity, po::value<uint16>()->default_value(1), "Log level [0-4]") + ; + + odtone::mih::config cfg(desc); + if (!cfg.parse(argc, argv, kConf_File) && !cfg.help()) { + std::cerr << "Error: Couldn't open config file: " << cfg.get<std::string>(kConf_File) << std::endl; + return EXIT_FAILURE; + } + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + // get command line parameters + bool enable_multicast = (cfg.count(kConf_MIHF_Multicast) == 1); + bool enable_unsolicited = (cfg.count(kConf_MIHF_Unsolicited) == 1); + + uint16 buff_size = cfg.get<uint16>(kConf_Receive_Buffer_Len); + uint16 lport = cfg.get<uint16>(kConf_MIHF_Local_Port); + uint16 rport = cfg.get<uint16>(kConf_MIHF_Remote_Port); + mih::octet_string id = cfg.get<mih::octet_string>(kConf_MIHF_Id); + uint16 loglevel = cfg.get<uint16>(kConf_MIHF_Verbosity); + kConf_MIHF_Link_Response_Time_Value = cfg.get<uint16>(kConf_MIHF_Link_Response_Time); + kConf_MIHF_Link_Delete_Value = cfg.get<uint16>(kConf_MIHF_Link_Delete); + std::vector<std::string> dscv_order = parse_discover_order(cfg); + // + + // set this mihf id + mihfid_t::instance()->assign(id.c_str()); + // set log level + ODTONE_LOG.level(loglevel); + + // create address books that stores info on how to contact mih + // saps and peer mihfs + address_book mihf_abook(io); + user_book user_abook; + link_book link_abook; + parse_mihf_information(cfg, mihf_abook); + parse_sap_registrations(cfg, user_abook, link_abook); + parse_peer_registrations(cfg, mihf_abook); + // + + // pool of pending transactions with peer mihfs + transaction_pool tpool(io); + + // pool of pending transactions with local mih saps (user and links) + local_transaction_pool lpool; + + // pool of pending capability discover requests + link_response_pool lrpool; + + // handler for remote messages + handler_t process_message = boost::bind(&sac_process_message, _1, _2); + + // wrapper for sending messages + net_sap netsap(io, mihf_abook, rport); + + // transaction manager for outgoing messages + message_out msgout(tpool, lpool, process_message, netsap); + transmit trnsmt(io, user_abook, link_abook, msgout, lport); + + // transaction manager for incoming messages + message_in msgin(tpool, process_message, netsap); + + // sac dispatch is for handling messages from local mih saps + // (users and links) + sac_dispatch sacd(trnsmt); + + // handler of messages received on local port + dispatch_t ldispatch = boost::bind(&sac_dispatch::operator(), sacd, _1); + // handler of messages received from peer mihfs + dispatch_t rdispatch = boost::bind(&message_in::operator(), msgin, _1); + + // create and bind to port 'lport' on loopback interface and + // call ldispatch when a message is received + udp_listener commhandv4(io, buff_size, ip::udp::v4(), "127.0.0.1", lport, ldispatch, true); + udp_listener commhandv6(io, buff_size, ip::udp::v6(), "::1", lport, ldispatch, true); + + // create and bind to port rport and call rdispatch when a + // message is received + udp_listener remotelistener_udp(io, buff_size, ip::udp::v6(), "::", rport, rdispatch, enable_multicast); + + // create and bind to port rport and call rdispatch when a + // message is received + tcp_listener remotelistener_tcp(io, buff_size, ip::tcp::v6(), "::", rport, rdispatch, enable_multicast); + + // start listening on local and remote ports + commhandv4.start(); + commhandv6.start(); + remotelistener_udp.start(); + + if(mihf_abook.get(mihfid_t::instance()->to_string()).capabilities_trans_list->get(mih::transport_tcp) == 1) + remotelistener_tcp.start(); + + // instantiate mihf services + event_service mies(io, lpool, trnsmt, mihf_abook, link_abook, user_abook); + command_service mics(io, lpool, trnsmt, mihf_abook, link_abook, user_abook, lrpool); + information_service miis(lpool, trnsmt, user_abook); + service_management sm(io, lpool, link_abook, user_abook, mihf_abook, trnsmt, lrpool, dscv_order, enable_unsolicited); + + // register callbacks with service access controller + sm_register_callbacks(sm); + mies_register_callbacks(mies); + mics_register_callbacks(mics); + miis_register_callbacks(miis); + std::cout << "Boot complete" << std::endl << std::flush; + + io.run(); + + return EXIT_SUCCESS; +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/patch.txt b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/patch.txt new file mode 100644 index 0000000000000000000000000000000000000000..bc19890ec1846bd8eaef5a7106b9c03fab6431a2 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/patch.txt @@ -0,0 +1,13 @@ +ODTONE/inc/odtone/mih/types/address.hpp +ODTONE/inc/odtone/mih/types/link.hpp +ODTONE/lib/odtone/CMakeLists.txt +ODTONE/lib/odtone/Jamfile +ODTONE/src/mihf/command_service.cpp +ODTONE/src/mihf/dst_transaction.cpp +ODTONE/src/mihf/event_service.cpp +ODTONE/src/mihf/link_book.cpp +ODTONE/src/mihf/main.cpp +ODTONE/src/mihf/service_access_controller.cpp +ODTONE/src/mihf/service_management.cpp +ODTONE/src/mihf/src_transaction.cpp +ODTONE/src/mihf/transaction_pool.cpp diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/service_access_controller.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/service_access_controller.cpp new file mode 100644 index 0000000000000000000000000000000000000000..2528fca8b655581412d4b9c929dc0ea31948c17f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/service_access_controller.cpp @@ -0,0 +1,174 @@ +//============================================================================== +// Brief : Service Access Controller +// Authors : Simao Reis <sreis@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "service_access_controller.hpp" +#include "meta_message.hpp" +#include "log.hpp" +#include "mihfid.hpp" + +#include <odtone/debug.hpp> +#include <odtone/mih/config.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> + +#include <map> +/////////////////////////////////////////////////////////////////////////////// + + +namespace odtone { namespace mihf { + +static std::map<uint, handler_t> _callbacks; /**< Callback map of the supported messages.*/ + +/** + * Registering a callback handler for a MIH message identifier. + * This should only be used on MIHF initialization because it's not thread safe. + * + * @param mid The MIH Message identifier. + * @param func The handler function. + */ +void sac_register_callback(uint mid, handler_t func) +{ + ODTONE_LOG(1, "(sac) sac_register_callback(): mid ", mid, " handler ", func); + _callbacks[mid] = func; +} + +/** + * Construct a Service Access Controller. + * + * @param t The transmit module. + */ +sac_dispatch::sac_dispatch(transmit &t) + : _transmit(t) +{ +} + +/** + * Check if there is a handler for this message and call it, else + * discard message. + * + * @param in The input message. + */ +void sac_dispatch::operator()(meta_message_ptr& in) +{ + /** __no__ authentication at this point */ + + uint mid = in->mid(); + + ODTONE_LOG(1, "(sac) dispatching message with mid: ", mid); + // + // no thread safety because insertion should __only__ be made + // on MIHF initialization + // + std::map<uint, handler_t>::iterator it; + it = _callbacks.find(mid); + + if(it != _callbacks.end()) { + handler_t process_message = it->second; + meta_message_ptr out(new meta_message); + + out->tid(in->tid()); + // send response if it was generated + try { + + if (process_message(in, out)) { + _transmit(out); + } else { + ODTONE_LOG(1, "sac_dispatch::operator() Message not transmitted out."); + } + + } catch(mih::bad_tlv) { + ODTONE_LOG(1, "Discarding malformed message."); + } catch(unknown_link_sap) { + ODTONE_LOG(1, "Received message from an unknown Link SAP. Discarding message."); + } catch(unknown_mih_user) { + ODTONE_LOG(1, "Received message from an unknown MIH-User. Discarding message."); + } + } else { + ODTONE_LOG(1, "(sac) (warning) message with mid: ", mid, + " unknown, discarding."); + } +} + +/** + * Check if there's a handler for this message and call it, else + * discard message. + * + * @param in The input message. + * @param out The output message. + */ +bool sac_process_message(meta_message_ptr& in, meta_message_ptr& out) +{ + // discard messages that this MIHF broadcasted to itself + // discard messages that are not destined to this MIHF or if + // multicast messages are not supported + if(in->source() == mihfid) { + ODTONE_LOG(1, "(sac) Discarding message! Reason: ", + "message was broadcasted to itself"); + return false; + } + + if(!utils::this_mihf_is_destination(in) && !utils::is_multicast(in)) { + ODTONE_LOG(1, "(sac) Discarding message! Reason: ", + "this is not the message destination"); + return false; + } + + /** __no__ authentication at this point */ + + uint mid = in->mid(); + + // + // no thread safety because insertion should __only__ be made + // on MIHF initialization + // + std::map<uint, handler_t>::iterator it; + it = _callbacks.find(mid); + + if(it != _callbacks.end()) { + handler_t process_message = it->second; + + bool rsp; + + try { + + rsp = process_message(in, out); + + } catch(mih::bad_tlv) { + ODTONE_LOG(1, "Discarding malformed message."); + return false; + } + + // set ip and port of response message + out->ip(in->ip()); + out->scope(in->scope()); + out->port(in->port()); + + // response message must have the same tid + out->tid(in->tid()); + + return rsp; + } else { + ODTONE_LOG(1, "(sac) (warning) message with mid: ", mid, + " unknown, discarding."); + } + + return false; +} + +} /* namespace mihf */ } /* namespace odtone */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/service_management.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/service_management.cpp new file mode 100644 index 0000000000000000000000000000000000000000..9c31b2c9e8fc43c987669b1031b81a989de1fd90 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/service_management.cpp @@ -0,0 +1,699 @@ +//============================================================================== +// Brief : Service Management +// Authors : Simao Reis <sreis@av.it.pt> +// Carlos Guimarães <cguimaraes@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "service_management.hpp" +#include "log.hpp" +#include "utils.hpp" +#include "mihfid.hpp" +#include "transmit.hpp" + +#include <odtone/mih/types/capabilities.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/tlv_types.hpp> + +#include <boost/foreach.hpp> +/////////////////////////////////////////////////////////////////////////////// + +extern odtone::uint16 kConf_MIHF_Link_Response_Time_Value; +extern odtone::uint16 kConf_MIHF_Link_Delete_Value; + +namespace odtone { namespace mihf { + +/** + * Construct the service management. + * + * @param io The io_service object that service management module will + * use to dispatch handlers for any asynchronous operations performed on + * the socket. + * @param lpool The local transaction pool module. + * @param link_abook The link book module. + * @param user_abook The user book module. + * @param address_abook The address book module. + * @param t The transmit module. + * @param lrpool The link response pool module. + * @param dscv_order Ordered list of entities that will manage the + * discovery of new PoS. + * @param enable_unsolicited Allows unsolicited discovery. + */ +service_management::service_management(io_service &io, + local_transaction_pool &lpool, + link_book &link_abook, + user_book &user_abook, + address_book &address_book, + transmit &t, + link_response_pool &lrpool, + std::vector<mih::octet_string> &dscv_order, + bool enable_unsolicited) + : _lpool(lpool), + _link_abook(link_abook), + _user_abook(user_abook), + _abook(address_book), + _transmit(t), + _lrpool(lrpool), + _discover(io, lpool, address_book, user_abook, t, dscv_order, enable_unsolicited) +{ + _dscv_order = dscv_order; + _enable_unsolicited = enable_unsolicited; + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + + // Get capabilities from statically configured Link SAPs + const std::vector<mih::octet_string> link_sap_list = _link_abook.get_ids(); + + BOOST_FOREACH(mih::octet_string id, link_sap_list) { + meta_message_ptr out(new meta_message()); + + *out << mih::request(mih::request::capability_discover); + out->tid(0); + out->destination(mih::id(id)); + + ODTONE_LOG(1, "(mics) forwarding Link_Capability_Discover.request to ", + out->destination().to_string()); + utils::forward_request(out, _lpool, _transmit); + } +} + +/** + * Asks for the capabilities of all local Link SAPs. + * + * @param in The input message. + * @param out The output message. + * @return Always false, because it does not send any response directly. + */ +bool service_management::link_capability_discover_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + // Asks for local Link SAPs capabilities + ODTONE_LOG(1, "(mism) gathering information about local Link SAPs capabilities"); + + *out << mih::request(mih::request::capability_discover); + out->tid(in->tid()); + out->destination(in->source()); + + // Check if the Link SAP is still active + uint16 fails = _link_abook.fail(out->destination().to_string()); + if(fails > kConf_MIHF_Link_Delete_Value) { + mih::octet_string dst = out->destination().to_string(); + _link_abook.inactive(dst); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } + else { + ODTONE_LOG(1, "(mics) forwarding Link_Capability_Discover.request to ", + out->destination().to_string()); + utils::forward_request(out, _lpool, _transmit); + } + + return false; +} + +/** + * Piggyback local MIHF Capabilities in request message. + * + * @param in input message. + * @param out output message. + */ +void service_management::piggyback_capabilities(meta_message_ptr& in, + meta_message_ptr& out) +{ + // Get local capabilities + address_entry mihf_cap = _abook.get(mihfid_t::instance()->to_string()); + + *out << mih::request(mih::request::capability_discover) + & mih::tlv_net_type_addr_list(mihf_cap.capabilities_list_net_type_addr) + & mih::tlv_event_list(mihf_cap.capabilities_event_list) + & mih::tlv_command_list(mihf_cap.capabilities_cmd_list) + & mih::tlv_query_type_list(mihf_cap.capabilities_query_type) + & mih::tlv_transport_option_list(mihf_cap.capabilities_trans_list) + & mih::tlv_mbb_ho_supp_list(mihf_cap.capabilities_mbb_ho_supp); + + out->tid(in->tid()); + out->source(in->source()); + out->destination(in->destination()); +} + +/** + * Parse all capabilities from MIH Capability Discover message and stores + * them. + * + * @param in input message. + * @param out output message. + */ +void service_management::get_capabilities(meta_message_ptr& in, + meta_message_ptr& out) +{ + address_entry mihf_info; + mihf_info.ip = in->ip(); + mihf_info.port = in->port(); + + if(in->opcode() == mih::operation::request) { + *in >> mih::request(mih::request::capability_discover) + & mih::tlv_net_type_addr_list(mihf_info.capabilities_list_net_type_addr) + & mih::tlv_event_list(mihf_info.capabilities_event_list) + & mih::tlv_command_list(mihf_info.capabilities_cmd_list) + & mih::tlv_query_type_list(mihf_info.capabilities_query_type) + & mih::tlv_transport_option_list(mihf_info.capabilities_trans_list) + & mih::tlv_mbb_ho_supp_list(mihf_info.capabilities_mbb_ho_supp); + } else if(in->opcode() == mih::operation::response) { + mih::status st; + *in >> mih::response(mih::response::capability_discover) + & mih::tlv_status(st) + & mih::tlv_net_type_addr_list(mihf_info.capabilities_list_net_type_addr) + & mih::tlv_event_list(mihf_info.capabilities_event_list) + & mih::tlv_command_list(mihf_info.capabilities_cmd_list) + & mih::tlv_query_type_list(mihf_info.capabilities_query_type) + & mih::tlv_transport_option_list(mihf_info.capabilities_trans_list) + & mih::tlv_mbb_ho_supp_list(mihf_info.capabilities_mbb_ho_supp); + } + + _abook.add(in->source().to_string(), mihf_info); +} + +/** + * Set response to MIH Capability Discover message. + * + * @param in input message. + * @param out output message. + */ +void service_management::set_capability_discover_response(meta_message_ptr& in, + meta_message_ptr& out) +{ + // Create and piggyback local capabilities in response message + address_entry mihf_cap = _abook.get(mihfid_t::instance()->to_string()); + + *out << mih::response(mih::response::capability_discover) + & mih::tlv_status(mih::status_success) + & mih::tlv_net_type_addr_list(mihf_cap.capabilities_list_net_type_addr) + & mih::tlv_event_list(mihf_cap.capabilities_event_list) + & mih::tlv_command_list(mihf_cap.capabilities_cmd_list) + & mih::tlv_query_type_list(mihf_cap.capabilities_query_type) + & mih::tlv_transport_option_list(mihf_cap.capabilities_trans_list) + & mih::tlv_mbb_ho_supp_list(mihf_cap.capabilities_mbb_ho_supp); + + out->tid(in->tid()); + out->source(mihfid); + out->destination(in->source()); +} + +/** + * Send Capability Discover Indication message to all MIH Users. + * + * @param in input message. + * @param out output message. + */ +void service_management::send_indication(meta_message_ptr& in, + meta_message_ptr& out) +{ + std::vector<mih::octet_string> ids = _user_abook.get_ids(); + in->opcode(mih::operation::indication); + for (std::vector<mih::octet_string>::iterator it = ids.begin(); it < ids.end(); ++it) { + if(std::find(_dscv_order.begin(), _dscv_order.end(), *it) == _dscv_order.end()) { + in->destination(mih::id(*it)); + _transmit(in); + ODTONE_LOG(3, "(mism) Capability_Discover.indication sent to ", + in->destination().to_string()); + } + } +} + +/** + * Capability Discover Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::capability_discover_request(meta_message_ptr& in, + meta_message_ptr& out) +{ + ODTONE_LOG(1, "(mism) received Capability_Discover.request from ", + in->source().to_string(), " with destination ", + in->destination().to_string()); + + // User requests the capabilities of a remote MIHF + if (in->is_local() && !utils::this_mihf_is_destination(in)) { + // Multicast && Discover Module ON + if(utils::is_multicast(in) && _dscv_order.size() != 0) { + _discover.request(in, out); + return false; + } + // Multicast && Discover Module OFF + piggyback_capabilities(in, out); + utils::forward_request(out, _lpool, _transmit); + return false; + // User requets the capabilitties of the local MIHF + } else if (in->is_local() && utils::this_mihf_is_destination(in)) { + set_capability_discover_response(in, out); + return true; + // Remote requets received + } else if (utils::this_mihf_is_destination(in)) { + get_capabilities(in, out); + send_indication(in, out); + set_capability_discover_response(in, out); + return true; + // Multicast request received + } else if (utils::is_multicast(in)) { + get_capabilities(in, out); + send_indication(in, out); + set_capability_discover_response(in, out); + return true; + } else { + ODTONE_LOG(3, "(mism) response to broadcast Capability_Discover.request disabled"); + return false; + } + + return false; +} + +/** + * Capability Discover Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::capability_discover_response(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received Capability_Discover.response from ", + in->source().to_string()); + + // Check if it is a discovery message + if(in->is_local() && std::find(_dscv_order.begin(), + _dscv_order.end(), + in->source().to_string()) != _dscv_order.end()) { + _discover.response(in, out); + return false; + } + + // Store remote MIHF capabilities + get_capabilities(in, out); + + // do we have a request from a user? + if (_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "forwarding Capability_Discover.response to ", + in->destination().to_string()); + in->opcode(mih::operation::confirm); + _transmit(in); + return false; + } + + // set source id to broadcast id and check if there's a + // broadcast request from a user + mih::id tmp = in->source(); + in->source(mih::id("")); + if (_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "forwarding Capability_Discover.response to ", + in->destination().to_string()); + in->opcode(mih::operation::confirm); + in->source(tmp); + _transmit(in); + return false; + } + + if(_enable_unsolicited) { + ODTONE_LOG(1, "forwarding Capability_Discover.response to all", + "MIH-Users"); + + std::vector<mih::octet_string> user_id_list = _user_abook.get_ids(); + BOOST_FOREACH(mih::octet_string user, user_id_list) { + ODTONE_LOG(3, "forwarding Capability_Discover.response to ", + user); + in->opcode(mih::operation::confirm); + in->source(mihfid); + in->destination(mih::id(user)); + _transmit(in); + } + + return false; + } else { + ODTONE_LOG(1, "no pending transaction for this message, discarding"); + return false; + } +} + +/** + * Capability Discover Confirm message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::capability_discover_confirm(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received Capability_Discover.confirm from ", + in->source().to_string()); + + _link_abook.reset(in->source().to_string()); + + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "no pending transaction for this message, discarding"); + return false; + } + + mih::status st; + boost::optional<mih::link_evt_list> event; + boost::optional<mih::link_cmd_list> command; + + *in >> mih::confirm(mih::confirm::capability_discover) + & mih::tlv_status(st) + & mih::tlv_link_evt_list(event) + & mih::tlv_link_cmd_list(command); + + if(st == mih::status_success) { + // Update Link SAP capabilities in the Link Book + _link_abook.update_capabilities(in->source().to_string(), event.get(), command.get()); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + } + + return false; +} + +/** + * Register Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::register_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received MIH_Register.request from ", in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + if(in->is_local()) { + in->source(mihfid); + } else { + // Set registration + mih::link_id_list lil; + *in >> mih::request(mih::request::mih_register) + & mih::tlv_link_id_list(lil); + _abook.set_registration(in->source().to_string(), lil); + // + } + + // Forward this message to MIH-User for handover as an indication + in->opcode(mih::operation::indication); + std::vector<mih::octet_string> user_list = _user_abook.get_ids(); + BOOST_FOREACH(mih::octet_string id, user_list) { + in->destination(mih::id(id)); + _lpool.add(in); + _transmit(in); + + ODTONE_LOG(1, "(mism) forwarding MIH_Register.indication to " , in->destination().to_string()); + } + + // Restore the original opcode after sending the indication message + in->opcode(mih::operation::request); + return false; + } else { + // Set registration + mih::link_id_list lil; + *in >> mih::request(mih::request::mih_register) + & mih::tlv_link_id_list(lil); + _abook.set_registration(in->destination().to_string(), lil); + // + + // try to forward the message, this is to handle the + // special case of the user handling MIH commands + // sending some MIH command request to a peer mihf + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * Register Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::register_response(meta_message_ptr&in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received MIH_Register.response from ", in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + // Set registration time interval + mih::status st; + uint16 interval; + *in >> mih::confirm(mih::confirm::mih_register) + & mih::tlv_status(st) + & mih::tlv_time_interval(interval); + + if(st == mih::status_success) { + _abook.set_registration(in->source().to_string(), interval); + } + // + + ODTONE_LOG(1, "(mism) forwarding MIH_Register.confirm to " , in->destination().to_string()); + in->opcode(mih::operation::confirm); + _transmit(in); + } else { + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + // Set registration time interval + mih::status st; + uint16 interval; + *in >> mih::confirm(mih::confirm::mih_register) + & mih::tlv_status(st) + & mih::tlv_time_interval(interval); + + if(st == mih::status_success) { + _abook.set_registration(in->destination().to_string(), interval); + } + // + + in->source(mihfid); + _transmit(in); + } + + return false; +} + +/** + * DeRegister Request message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::deregister_request(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received MIH_DeRegister.request from ", + in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + if(in->is_local()) + in->source(mihfid); + + // Forward this message to MIH-User for handover as an indication + in->opcode(mih::operation::indication); + std::vector<mih::octet_string> user_list = _user_abook.get_ids(); + BOOST_FOREACH(mih::octet_string id, user_list) { + in->destination(mih::id(id)); + _lpool.add(in); + _transmit(in); + + ODTONE_LOG(1, "(mism) forwarding MIH_DeRegister.indication to ", + in->destination().to_string()); + } + + // Restore the original opcode after sending the indication message + in->opcode(mih::operation::request); + return false; + } else { + // try to forward the message, this is to handle the + // special case of the user handling MIH commands + // sending some MIH command request to a peer mihf + utils::forward_request(in, _lpool, _transmit); + return false; + } + + return false; +} + +/** + * DeRegister Response message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::deregister_response(meta_message_ptr&in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received MIH_DeRegister.response from ", in->source().to_string()); + + if(utils::this_mihf_is_destination(in)) { + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + // Set registration + mih::status st; + *in >> mih::confirm(mih::confirm::mih_deregister) + & mih::tlv_status(st); + + if(st == mih::status_success) { + // Set empty registered link id list + boost::optional<mih::link_id_list> lil; + boost::optional<uint16> interval; + _abook.set_registration(in->source().to_string(), lil); + _abook.set_registration(in->source().to_string(), interval); + } + // + + ODTONE_LOG(1, "(mism) forwarding MIH_DeRegister.confirm to " , in->destination().to_string()); + in->opcode(mih::operation::confirm); + _transmit(in); + } else { + if(!_lpool.set_user_tid(in)) { + ODTONE_LOG(1, "(mics) warning: no local transaction for this msg ", + "discarding it"); + return false; + } + + // Set registration + mih::status st; + *in >> mih::confirm(mih::confirm::mih_deregister) + & mih::tlv_status(st); + + if(st == mih::status_success) { + // Set empty registered link id list + boost::optional<mih::link_id_list> lil; + boost::optional<uint16> interval; + _abook.set_registration(in->destination().to_string(), lil); + _abook.set_registration(in->destination().to_string(), interval); + } + // + + in->source(mihfid); + _transmit(in); + } + + return false; +} + +/** + * Link Register Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::link_register_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received Link_Register.indication from ", + in->source().to_string()); + + // Add Link SAP to the list of known Link SAPs + mih::status st; + mih::link_id link_id; + + mih::octet_string ip(in->ip()); + + *in >> odtone::mih::indication() + & odtone::mih::tlv_interface_type_addr(link_id); + + ODTONE_LOG(4, "((mism) received Link_Register.indication: link_id", link_id); + // Add Link SAP to the list of known Link SAPs + _link_abook.add(in->source().to_string(), in->ip(), in->port(), link_id); + + // Update MIHF network type address + address_entry mihf_cap = _abook.get(mihfid_t::instance()->to_string()); + + // Set Network Type Address + mih::net_type_addr nta; + nta.nettype.link = link_id.type; + nta.addr = link_id.addr; + + if(mihf_cap.capabilities_list_net_type_addr.is_initialized()) { + mihf_cap.capabilities_list_net_type_addr.get().push_back(nta); + } else { + mih::net_type_addr_list ntal; + ntal.push_back(nta); + mihf_cap.capabilities_list_net_type_addr = ntal; + } + _abook.add(mihfid_t::instance()->to_string(), mihf_cap); + + // Request the Link SAP capabilities + link_capability_discover_request(in, out); + + return false; +} + +/** + * User Register Indication message handler. + * + * @param in The input message. + * @param out The output message. + * @return True if the response is sent immediately or false otherwise. + */ +bool service_management::user_register_indication(meta_message_ptr &in, + meta_message_ptr &out) +{ + ODTONE_LOG(1, "(mism) received User_Register.indication from ", + in->source().to_string()); + + // Add MIH User to the list of known MIH Users + boost::optional<mih::mih_cmd_list> supp_cmd; + boost::optional<mih::iq_type_list> supp_iq; + + mih::octet_string ip(in->ip()); + + *in >> odtone::mih::indication() + & odtone::mih::tlv_command_list(supp_cmd) + & odtone::mih::tlv_query_type_list(supp_iq); + + _user_abook.add(in->source().to_string(), ip, in->port(), supp_cmd, supp_iq); + + // Update MIHF capabilities + utils::update_local_capabilities(_abook, _link_abook, _user_abook); + + return false; +} + +} /* namespace mihf */ } /* namespace odtone */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/src_transaction.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/src_transaction.cpp new file mode 100644 index 0000000000000000000000000000000000000000..ef132d87b0e4e8ee59f3f0a3fb57c4a1f935aa38 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/src_transaction.cpp @@ -0,0 +1,157 @@ +//============================================================================== +// Brief : Source Transaction +// Authors : Simao Reis <sreis@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "log.hpp" +#include "src_transaction.hpp" +#include "utils.hpp" +/////////////////////////////////////////////////////////////////////////////// + +namespace odtone { namespace mihf { + +/** + * Constructor a Source State Machine transaction. + * + * @param f The transaction handler. + * @param netsap The netsap module. + */ +src_transaction_t::src_transaction_t(handler_t &f, net_sap &netsap) + : transaction_t(f, netsap) +{ + state = SRC_INIT; +} + +/** + * Run Source State Machine transaction. + */ +void src_transaction_t::run() +{ + switch (state) + { + case SRC_INIT: goto _init_lbl_; + case SRC_WAIT_RESPONSE_MSG: goto _wait_response_msg_lbl_; + case SRC_WAIT_ACK: goto _wait_ack_lbl_; + case SRC_PROCESS_MSG: goto _process_msg_lbl_; + case SRC_FAILURE: goto _failure_lbl_; + case SRC_SUCCESS: goto _success_lbl_; + } + + _init_lbl_: + { + ODTONE_LOG(1, "(src_transaction_t) init tid ", out->tid()); + transaction_status = ONGOING; + response_received = false; + transaction_stop_when = 15; // FIXME: read from config + opcode = out->opcode(); + is_multicast = utils::is_multicast(out); + start_ack_requestor = (out->ackreq() && !is_multicast); + tid = out->tid(); + my_mihf_id = out->source(); + peer_mihf_id = out->destination(); + + _netsap.send(out); + + if(start_ack_requestor) + ack_requestor(); + + if (opcode == mih::operation::indication) { + if (start_ack_requestor) { + ack_requestor_status = ONGOING; + goto _wait_ack_lbl_; + } + else + goto _success_lbl_; + } + else if (opcode == mih::operation::response) + goto _success_lbl_; + else if (opcode == mih::operation::request) + goto _wait_response_msg_lbl_; + + assert(0); // failsafe + } + + _wait_ack_lbl_: + { + ODTONE_LOG(1, "(src_transaction_t) wait ack tid ", tid); + state = SRC_WAIT_ACK; + + if (ack_requestor_status == SUCCESS) + goto _success_lbl_; + else if (ack_requestor_status == FAILURE) + goto _failure_lbl_; + + return; + } + + _wait_response_msg_lbl_: + { + ODTONE_LOG(1, "(src_transaction_t) wait response tid ", tid); + state = SRC_WAIT_RESPONSE_MSG; + + if (transaction_stop_when == 0) { + if (response_received) + goto _success_lbl_; + else + goto _failure_lbl_; + } else if (msg_in_avail) + goto _process_msg_lbl_; + + return; + } + + _process_msg_lbl_: + { + ODTONE_LOG(1, "(src_transaction_t) process msg tid ", tid); + state = SRC_PROCESS_MSG; + + start_ack_responder = in->ackreq(); + + msg_out_avail = false; + + if (start_ack_responder) + ack_responder(); + + process_message(in, out); + msg_in_avail = false; + response_received = true; + + if (is_multicast) + goto _wait_response_msg_lbl_; + else + goto _success_lbl_; + } + + _failure_lbl_: + { + ODTONE_LOG(1, "(src_transaction_t) failure tid ", tid); + transaction_status = FAILURE; + state = SRC_FAILURE; + return; + } + + _success_lbl_: + { + ODTONE_LOG(1, "(src_transaction_t) success tid ", tid); + transaction_status = SUCCESS; + state = SRC_SUCCESS; + } + + return; +} + + +} /* namespace mihf */ } /* namespace odtone */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/transaction_pool.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/transaction_pool.cpp new file mode 100644 index 0000000000000000000000000000000000000000..204e5a1ee162f7474334cfaeba4548e455e573d6 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/mihf_src/transaction_pool.cpp @@ -0,0 +1,195 @@ +//============================================================================== +// Brief : Transaction Pool +// Authors : Simao Reis <sreis@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2013 Universidade Aveiro +// Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +/////////////////////////////////////////////////////////////////////////////// +#include "log.hpp" +#include "transaction_pool.hpp" +/////////////////////////////////////////////////////////////////////////////// + +namespace odtone { namespace mihf { + +/** + * Construct a transaction pool. + * + * @param io The io_service object that transaction pool module will + * use to dispatch handlers for any asynchronous operations performed on + * the socket. + */ +transaction_pool::transaction_pool(io_service &io) + : _timer(io, boost::posix_time::seconds(1)), + _dst_mutex(), + _src_mutex() +{ + // start timer + _timer.expires_from_now(boost::posix_time::seconds(1)); + _timer.async_wait(boost::bind(&transaction_pool::tick, this)); +} + +/** + * Decrements the timer of each transaction only if its value is + * greater than 0. + * + * @param set The transaction type. + * @param it The transaction. + * @param mutex The mutex. + */ +template <class Set, class SetIterator> +void transaction_pool::dec(Set &set, + SetIterator &it, + boost::mutex &mutex) +{ + Set del_these; + + { + boost::mutex::scoped_lock lock(mutex); + + for(it = set.begin(); it != set.end(); it++) { + (*it)->transaction_stop_when--; + ODTONE_LOG(1, "(transaction_pool) decrementing tid ", (*it)->tid, " counter down to ", (*it)->transaction_stop_when); + if ((*it)->transaction_stop_when == 0) { + (*it)->run(); + + if ((*it)->transaction_status == ONGOING) + del_these.insert(*it); + } + + if ((*it)->ack_requestor_status == ONGOING) { + (*it)->retransmission_when--; + if ((*it)->retransmission_when == 0) { + (*it)->ack_requestor(); + (*it)->run(); + } + } + + if ((*it)->transaction_status != ONGOING) + del_these.insert(*it); + } + } + + // delete finished transactions + for (it = del_these.begin(); it != del_these.end(); it++) + del(*it); +} + +/** + * Decrements each transaction timer existente in the transaction pool. + */ +void transaction_pool::tick() +{ + //ODTONE_LOG(1, "(transaction_pool) tick"); + _timer.expires_from_now(boost::posix_time::seconds(1)); + _timer.async_wait(boost::bind(&transaction_pool::tick, this)); + + src_transaction_set::iterator src_it; + if (_src.size() > 0) + dec(_src, src_it, _src_mutex); + + dst_transaction_set::iterator dst_it; + if (_dst.size() > 0) + dec(_dst, dst_it, _dst_mutex); +} + +/** + * Add a new source transaction entry in the transaction pool. + * + * @param t The source transaction pointer. + */ +void transaction_pool::add(src_transaction_ptr &t) +{ + ODTONE_LOG(1, "(transaction_pool) add src tid ", t->tid); + boost::mutex::scoped_lock lock(_src_mutex); + _src.insert(t); +} + +/** + * Add a new destination transaction entry in the transaction pool. + * + * @param t The destination transaction pointer. + */ +void transaction_pool::add(dst_transaction_ptr &t) +{ + ODTONE_LOG(1, "(transaction_pool) add dst tid ", t->tid); + boost::mutex::scoped_lock lock(_dst_mutex); + _dst.insert(t); +} + +/** + * Remove an existing source transaction entry from the transaction pool. + * + * @param t The source transaction pointer to be removed. + */ +void transaction_pool::del(const src_transaction_ptr &t) +{ + ODTONE_LOG(1, "(transaction_pool) del src tid ", t->tid); + boost::mutex::scoped_lock lock(_src_mutex); + _src.erase(t); +} + +/** + * Remove a existing destination transaction entry from the transaction pool. + * + * @param t destination transaction pointer to be removed. + */ +void transaction_pool::del(const dst_transaction_ptr &t) +{ + ODTONE_LOG(1, "(transaction_pool) del dst tid ", t->tid); + boost::mutex::scoped_lock lock(_dst_mutex); + _dst.erase(t); +} + +/** + * Searchs for a source transaction in the transaction pool. + * + * @param id The MIH destination identifier to search for. + * @param tid The transaction identifier to search for. + * @param t The source transaction pointer. + */ +void transaction_pool::find(const mih::id &id, uint16 tid, src_transaction_ptr &t) +{ + boost::mutex::scoped_lock lock(_src_mutex); + src_transaction_set::iterator it; + + for(it = _src.begin(); it != _src.end(); it++) { + if(((*it)->peer_mihf_id == id) && (*it)->tid == tid) { + t = *it; + return; + } + } +} + +/** + * Searchs for a destination transaction in the transaction pool. + * + * @param id The MIH source identifier to search for. + * @param tid The transaction identifier to search for. + * @param t The destination transaction pointer. + */ +void transaction_pool::find(const mih::id &id, uint16 tid, dst_transaction_ptr &t) +{ + boost::mutex::scoped_lock lock(_dst_mutex); + dst_transaction_set::iterator it; + + for(it = _dst.begin(); it != _dst.end(); it++) { + if(((*it)->peer_mihf_id == id) && (*it)->tid == tid) { + t = *it; + return; + } + } +} + + +} /* namespace mihf */ } /* namespace odtone */ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/build_all.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/build_all.bash new file mode 100644 index 0000000000000000000000000000000000000000..f9bc814e62163a2f25cf13e6c455a5a9f1c22694 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/build_all.bash @@ -0,0 +1,122 @@ +#! /bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file build_all.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# +########################################################### +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +source $THIS_SCRIPT_PATH/env_802dot21.bash +########################################################### + +echo_success "\n###############################" +echo_success "# Check installed utils and libs" +echo_success "###############################" +test_command_install_package "gccxml" "gccxml" "--force-yes" +test_command_install_package "iptables" "iptables" +#test_command_install_package "ebtables" "ebtables" "--force-yes" +test_command_install_package "ip" "iproute" +test_install_package "openssl" +test_install_package "libblas-dev" +# for itti analyser +test_install_package "libgtk-3-dev" +test_install_package "libxml2" +test_install_package "libxml2-dev" +test_install_package "libforms-bin" "--force-yes" +test_install_package "libforms-dev" +test_install_package "libatlas-dev" +test_install_package "libatlas-base-dev" +test_install_package "libpgm-5.1-0" "--force-yes" +test_install_package "libpgm-dev" "--force-yes" +test_install_package linux-headers-`uname -r` +test_install_package "tshark" "--force-yes" +# for ODTONE git clone +test_install_package "git" + +test_install_asn1c_4_rrc_cellular + + +echo_success "\n###############################" +echo_success "# COMPILE oaisim" +echo_success "###############################" +cd $OPENAIR_TARGETS/SIMU/USER +#echo_success "Executing: make oaisim NAS=1 OAI_NW_DRIVER_TYPE_ETHERNET=1 ENABLE_ITTI=1 USER_MODE=1 OPENAIR2=1 ENABLE_RAL=1 MIH_C_MEDIEVAL_EXTENSIONS=1 USE_3GPP_ADDR_AS_LINK_ADDR=1 RLC_STOP_ON_LOST_PDU=1 Rel10=1 -j`grep -c ^processor /proc/cpuinfo `" +#make --keep-going oaisim NAS=1 OAI_NW_DRIVER_TYPE_ETHERNET=1 ENABLE_ITTI=1 USER_MODE=1 OPENAIR2=1 ENABLE_RAL=1 MIH_C_MEDIEVAL_EXTENSIONS=1 USE_3GPP_ADDR_AS_LINK_ADDR=1 RLC_STOP_ON_LOST_PDU=1 Rel10=1 -j`grep -c ^processor /proc/cpuinfo ` + +echo_success "Executing: make oaisim NAS=1 OAI_NW_DRIVER_TYPE_ETHERNET=1 ENABLE_ITTI=1 USER_MODE=1 OPENAIR2=1 ENABLE_RAL=1 MIH_C_MEDIEVAL_EXTENSIONS=1 USE_3GPP_ADDR_AS_LINK_ADDR=1 RLC_STOP_ON_LOST_PDU=1 -j`grep -c ^processor /proc/cpuinfo `" +#make --keep-going oaisim NAS= +make --keep-going oaisim DEBUG=1 NAS=1 OAI_NW_DRIVER_TYPE_ETHERNET=1 ENABLE_ITTI=1 USER_MODE=1 OPENAIR2=1 ENABLE_RAL=1 MIH_C_MEDIEVAL_EXTENSIONS=1 USE_3GPP_ADDR_AS_LINK_ADDR=1 RLC_STOP_ON_LOST_PDU=1 -j`grep -c ^processor /proc/cpuinfo ` + +if [[ $? -eq 2 ]] ; then + exit 1 +fi + + +echo_success "\n###############################" +echo_success "# COMPILE IP kernel drivers" +echo_success "###############################" +echo_success "Compiling IP Drivers" +cd $OPENAIR2_DIR +make naslite_netlink_ether.ko +cd $OPENAIR2_DIR/NAS/DRIVER/LITE/RB_TOOL/ +make + + +echo_success "\n###############################" +echo_success "# COMPILE MIH-F" +echo_success "###############################" +cd $ODTONE_ROOT +b2 --boost-root=$BOOST_ROOT linkflags=-lpthread + + +echo_success "\n###############################" +echo_success "# COMPILE MIH-USER" +echo_success "###############################" +cd $ODTONE_ROOT/app/lte_test_user/ +b2 --boost-root=$BOOST_ROOT linkflags=-lrt linkflags=-lpthread + + +echo_success "\n###############################" +echo_success "# COMPILE ITTI ANALYSER" +echo_success "###############################" +cd $OPENAIR_HOME/common/utils/itti_analyzer +if [ ! -f $OPENAIR_HOME/common/utils/itti_analyzer/Makefile ] + then + autoreconf -i + mkdir -m 777 objs + cd objs + ../configure + fi +sudo make install -j`grep -c ^processor /proc/cpuinfo ` + + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/env_802dot21.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/env_802dot21.bash new file mode 100644 index 0000000000000000000000000000000000000000..32bc915d99a404b37094c91823a93bc14deeec47 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/env_802dot21.bash @@ -0,0 +1,113 @@ +#!/bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file env_802dot21.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# + +##################################################### +# VARIABLES TO BE FILLED WITH RIGHT VALUES: +##################################################### +export BOOST_ROOT=path_to_boost_folder +export ODTONE_ROOT=path_to_odtone_folder + +export MIH_F=odtone-mihf +export ENB_MIH_F_CONF_FILE=odtone_enb.conf +export UE_MIH_F_CONF_FILE=odtone_ue.conf + +export ODTONE_MIH_USER_DIR=$ODTONE_ROOT/app/lte_test_user +export ODTONE_MIH_EXE_DIR=$ODTONE_ROOT/dist + +#export ENB_MIH_USER=enb2_lte_user +#export ENB_MIH_USER_CONF_FILE=enb2_lte_user.conf +export ENB_MIH_USER=enb_lte_user +export ENB_MIH_USER_CONF_FILE=enb_lte_user.conf + +export UE_MIH_USER=ue_lte_user +export UE_MIH_USER_CONF_FILE=ue_lte_user.conf +##################################################### + + +ENV_SCRIPT_SOURCED="?" +ENV_SCRIPT_ERRORS="no" + +if [[ $BASH_SOURCE != $0 ]]; then + THIS_SCRIPT_PATH=${BASH_SOURCE%env_802dot21.bash} + [[ x"$THIS_SCRIPT_PATH" == x ]] && THIS_SCRIPT_PATH="./" + ENV_SCRIPT_SOURCED="yes" +else + THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +fi + +ENV_SCRIPT_STARTED="yes" +source $THIS_SCRIPT_PATH/utils.bash + +if [ -d $ODTONE_ROOT ]; then + echo_success "ODTONE_ROOT = $ODTONE_ROOT." >&2 +else + echo_error "ODTONE_ROOT variable was not set correctly, please update ($ODTONE_ROOT)." >&2 +fi +if [ ! -d $BOOST_ROOT ]; then + echo_error "BOOST_ROOT variable was not set correctly, please update (may be you also need to install boost), exiting." + ENV_SCRIPT_ERRORS="yes" +else + command -v b2 >/dev/null 2>&1 + if [ $? -ne 0 ]; then + echo_warning "Program b2 is not installed or not in the PATH variable. Trying to resolve..." >&2 + if [[ -x "$BOOST_ROOT/b2" ]]; then + echo_success "Program b2 found in dir $BOOST_ROOT." >&2 + export PATH=$PATH:$BOOST_ROOT + else + echo_warning "Program b2 not found in dir $BOOST_ROOT. Trying to install..." >&2 + cd $BOOST_ROOT; ./bootstrap.sh; + cd - + export PATH=$PATH:$BOOST_ROOT + fi + command -v $BOOST_ROOT/b2 >/dev/null 2>&1 + if [ $? -eq 0 ]; then + echo_success "Program b2 is now reachable by the PATH variable during the execution of this script." >&2 + else + echo_error "Built of b2 failed. Please help yourself" >&2 + fi + fi +fi + + +if [ ! -d $ODTONE_ROOT ]; then + echo_error "ODTONE_ROOT variable was not set correctly, please update (may be you also need to install odtone), exiting." + ENV_SCRIPT_ERRORS="yes" +fi + + +[[ x"$ENV_SCRIPT_ERRORS" == "xyes" ]] && [[ x"$ENV_SCRIPT_SOURCED" == "xyes" ]] && return 1 +[[ x"$ENV_SCRIPT_ERRORS" == "xyes" ]] && [[ x"$ENV_SCRIPT_SOURCED" == "xno" ]] && exit 1 + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/utils.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/utils.bash new file mode 100644 index 0000000000000000000000000000000000000000..9322aa4a6193bac5a37155c7cc1f84c899905e2a --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/common/openair_scripts/utils.bash @@ -0,0 +1,380 @@ +#!/bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file utils.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# + +cidr2mask() { + local i mask="" + local full_octets=$(($1/8)) + local partial_octet=$(($1%8)) + + for ((i=0;i<4;i+=1)); do + if [ $i -lt $full_octets ]; then + mask+=255 + elif [ $i -eq $full_octets ]; then + mask+=$((256 - 2**(8-$partial_octet))) + else + mask+=0 + fi + test $i -lt 3 && mask+=. + done + + echo $mask +} + + +black='\E[30m' +red='\E[31m' +green='\E[32m' +yellow='\E[33m' +blue='\E[34m' +magenta='\E[35m' +cyan='\E[36m' +white='\E[37m' +reset_color='\E[00m' + +ROOT_UID=0 +E_NOTROOT=67 + +HOSTNAME=$(hostname -f) + +trim () +{ + echo "$1" | sed -n '1h;1!H;${;g;s/^[ \t]*//g;s/[ \t]*$//g;p;}' +} + +trim2() +{ + local var=$@ + var="${var#"${var%%[![:space:]]*}"}" # remove leading whitespace characters + var="${var%"${var##*[![:space:]]}"}" # remove trailing whitespace characters + echo -n "$var" +} + +cecho() # Color-echo +# arg1 = message +# arg2 = color +{ + local default_msg="No Message." + message=${1:-$default_msg} + color=${2:-$green} + echo -e -n "$color$message$reset_color" + echo + return +} + +echo_error() { + local my_string="" + until [ -z "$1" ] + do + my_string="$my_string$1" + shift + done + cecho "$my_string" $red +} + +echo_warning() { + local my_string="" + until [ -z "$1" ] + do + my_string="$my_string$1" + shift + done + cecho "$my_string" $yellow +} + +echo_success() { + local my_string="" + until [ -z "$1" ] + do + my_string="$my_string$1" + shift + done + cecho "$my_string" $green +} + +bash_exec() { + output=$($1 2>&1) + result=$? + if [ $result -eq 0 ] + then + echo_success "$1" + else + echo_error "$1: $output" + fi +} + + +extract() { + if [ -f $1 ] ; then + case $1 in + *.tar.bz2) tar xvjf $1 ;; + *.tar.gz) tar xvzf $1 ;; + *.bz2) bunzip2 $1 ;; + *.rar) unrar $1 ;; + *.gz) gunzip $1 ;; + *.tar) tar xvf $1 ;; + *.tbz2) tar xvjf $1 ;; + *.tgz) tar xvzf $1 ;; + *.zip) unzip $1 ;; + *.Z) uncompress $1 ;; + *.7z) 7z x $1 ;; + *) echo_error "'$1' cannot be extracted via >extract<" ; return 1;; + esac + else + echo_error "'$1' is not a valid file" + return 1 + fi + return 0 +} + + +set_openair() { + path=`pwd` + declare -i length_path + declare -i index + length_path=${#path} + + for i in 'openair1' 'openair2' 'openair3' 'openair-cn' 'targets' + do + index=`echo $path | grep -b -o $i | cut -d: -f1` + #echo ${path%$token*} + if [[ $index -lt $length_path && index -gt 0 ]] + then + declare -x OPENAIR_DIR + index=`expr $index - 1` + openair_path=`echo $path | cut -c1-$index` + #openair_path=`echo ${path:0:$index}` + export OPENAIR_DIR=$openair_path + export OPENAIR_HOME=$openair_path + export OPENAIR1_DIR=$openair_path/openair1 + export OPENAIR2_DIR=$openair_path/openair2 + export OPENAIR3_DIR=$openair_path/openair3 + export OPENAIRCN_DIR=$openair_path/openair-cn + export OPENAIR_TARGETS=$openair_path/targets + return 0 + fi + done + return -1 +} + +test_install_asn1c_4_rrc_cellular() { + if [ -d $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c/asn1c ]; then + if [ -x $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c/asn1c/asn1c/asn1c ]; then + if [ -x /usr/local/bin/asn1c ]; then + diff /usr/local/bin/asn1c $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c/asn1c/asn1c/asn1c >/dev/null 2>&1; + if [ $? -eq 0 ]; then + echo_success "asn1c for RRC cellular installed" + return 0 + fi + fi + echo_warning "Installing asn1c for RRC cellular..." + cd $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c/asn1c + sudo make install + return 0 + fi + else + echo_warning "asn1c for RRC cellular is not installed in $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c/. Installing it" + cd $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c + svn co https://asn1c.svn.sourceforge.net/svnroot/asn1c/trunk asn1c + fi + echo_warning "Configuring and building and installing asn1c for RRC cellular..." + cd $OPENAIR2_DIR/RRC/LITE/MESSAGES/asn1c/asn1c + ./configure + make + sudo make install +} + +wait_process_started () { + if [ -z "$1" ] + then + echo_error "WAITING FOR PROCESS START: NO PROCESS" + return 1 + fi + ps -C $1 > /dev/null 2>&1 + while [ $? -ne 0 ]; do + echo_warning "WAITING FOR $1 START" + sleep 2 + ps -C $1 > /dev/null 2>&1 + done + echo_success "PROCESS $1 STARTED" + return 0 +} + +is_process_started () { + if [ -z "$1" ] + then + echo_error "WAITING FOR PROCESS START: ERROR NO PROCESS NAME IN ARGUMENT" + return 1 + fi + ps -C $1 > /dev/null 2>&1 + if [ $? -ne 0 ] + then + echo_success "PROCESS $1 NOT STARTED" + return 1 + fi + echo_success "PROCESS $1 STARTED" + return 0 +} + +assert() { + # If condition false + # exit from script with error message + E_PARAM_ERR=98 + E_PARAM_FAILED=99 + + if [ -z "$2" ] # Not enought parameters passed. + then + return $E_PARAM_ERR + fi + + lineno=$2 + if [ ! $1 ] + then + echo "Assertion failed: \"$1\"" + echo "File \"$0\", line $lineno" + exit $E_ASSERT_FAILED + fi +} + + +test_install_package() { + # usage: test_install_package package_name_to_be_installed optional_option_to_apt_get_install + dpkg --get-selections $1 | grep -i install > /dev/null 2>&1 + if [ $? -ne 0 ]; then + echo_warning "Package $1 is not installed. Installing it." >&2 + apt-get install $2 $1 -y + dpkg --get-selections $1 | grep -i install > /dev/null 2>&1 + if [ $? -ne 0 ]; then + exit 1 + fi + else + echo_success "$1 is installed" + fi + return 0 +} + + + +test_command_install_package() { + # usage: test_command_install_package searched_binary package_to_be_installed_if_binary_not_found optional_option_to_apt_get_install + if [ $# -eq 2 ]; then + command -v $1 >/dev/null 2>&1 || { echo_warning "Program $1 is not installed. Trying installing it." >&2; apt-get install $2 -y; command -v $1 >/dev/null 2>&1 || { echo_error "Program $1 is not installed. Aborting." >&2; exit 1; };} + else + if [ $# -eq 3 ]; then + command -v $1 >/dev/null 2>&1 || { echo_warning "Program $1 is not installed. Trying installing it (apt-get install $3 $2)." >&2; apt-get install $3 $2 -y; command -v $1 >/dev/null 2>&1 || { echo_error "Program $1 is not installed. Aborting." >&2; exit 1; };} + else + echo_success "test_command_install_package: BAD PARAMETER" + exit 1 + fi + fi + echo_success "$1 available" +} + +test_command_install_script() { + # usage: test_command_install_script searched_binary script_to_be_invoked_if_binary_not_found + command -v $1 >/dev/null 2>&1 || { echo_warning "Program $1 is not installed. Trying installing it." >&2; bash $2; command -v $1 >/dev/null 2>&1 || { echo_error "Program $1 is not installed. Aborting." >&2; exit 1; };} + echo_success "$1 available" +} + +start_openswitch_daemon() { + rmmod -s bridge + if [[ -e "/lib/modules/`uname -r`/extra/openvswitch.ko" ]] ; then + bash_exec "insmod /lib/modules/`uname -r`/extra/openvswitch.ko" + else + echo_error "/lib/modules/`uname -r`/extra/openvswitch.ko not found, exiting" + exit -1 + fi + is_process_started "ovsdb-server" + if [ $? -ne 0 ] + then + ovsdb-server --remote=punix:/usr/local/var/run/openvswitch/db.sock --remote=db:Open_vSwitch,manager_options --pidfile --detach + wait_process_started "ovsdb-server" + fi + # To be done after installation + # ovs-vsctl --no-wait init + is_process_started "ovs-vswitchd" + if [ $? -ne 0 ] + then + ovs-vswitchd --pidfile --detach + wait_process_started "ovs-vswitchd" + fi +} + +check_enb_config() { + if [ ! -f $OPENAIR3_DIR/OPENAIRMME/UTILS/CONF/enb_$HOSTNAME.conf ] + then + echo "Cannot find file $OPENAIR3_DIR/OPENAIRMME/UTILS/CONF/enb_$HOSTNAME.conf" + echo "Please make sure to create one that fits your use (you can use mme_default.conf file as template)" + exit -1 + fi +} + + +check_for_root_rights() { + if [[ $EUID -ne 0 ]]; then + echo "This script must be run as root" 1>&2 + exit -1 + fi +} + +rotate_log_file () { + if [ -f $1 ]; then + TIMESTAMP=`date +%Y-%m-%d.%Hh_%Mm_%Ss` + NEWLOGFILE=$1.$TIMESTAMP + mv $1 $NEWLOGFILE + cat /dev/null > $1 + sync + nohup gzip -f -9 $NEWLOGFILE & + fi +} + +########################################################### +declare -x OPENAIR_DIR="" +declare -x OPENAIR_HOME="" +declare -x OPENAIR1_DIR="" +declare -x OPENAIR2_DIR="" +declare -x OPENAIR3_DIR="" +declare -x OPENAIRCN_DIR="" +declare -x OPENAIR_TARGETS="" +########################################################### + +set_openair +cecho "OPENAIR_DIR = $OPENAIR_DIR" $green +cecho "OPENAIR_HOME = $OPENAIR_HOME" $green +cecho "OPENAIR1_DIR = $OPENAIR1_DIR" $green +cecho "OPENAIR2_DIR = $OPENAIR2_DIR" $green +cecho "OPENAIR3_DIR = $OPENAIR3_DIR" $green +cecho "OPENAIRCN_DIR = $OPENAIRCN_DIR" $green +cecho "OPENAIR_TARGETS = $OPENAIR_TARGETS" $green diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/enb_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/enb_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..b9bbf279de1650cfa71dbeef31960214086d0d98 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/enb_lte_user.conf @@ -0,0 +1,40 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_enb + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/link_sap.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/link_sap.conf new file mode 100644 index 0000000000000000000000000000000000000000..c5b63413d0148de7186c2fac37cf9e606dd592df --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/link_sap.conf @@ -0,0 +1,47 @@ +#=============================================================================== +# Brief : Link SAP configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[link] +## +## Link SAP identifier +## +id=link1 + +## +## Link SAP listening +## +port = 1235 + +## +## Link SAP interface technology +## +tec = 802_11 + +## +## Link SAP interface address +## +link_addr = 00:11:22:33:44:55 + +## +## Comma separated list of the Link SAP supported events +## +event_list = link_detected, link_up, link_down, link_parameters_report, link_going_down, link_handover_imminent, link_handover_complete + +[mihf] +ip=127.0.0.1 +local_port=1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/odtone_enb.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/odtone_enb.conf new file mode 100644 index 0000000000000000000000000000000000000000..93d8fa1a9fed5be8b2d7d7179118eff5ca9d4d00 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_conf/odtone_enb.conf @@ -0,0 +1,72 @@ +#=============================================================================== +# Brief : MIHF configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[mihf] +## +## This mihf's id +## +## Usage: id = <MIHF ID> +## +id = mihf1_enb + +## +## Port on localhost that MIH Users and MIH Link SAPs connect to. +## +## Usage: local_port = <port> +## +local_port = 1025 + +## +## Port to which remote peer MIHF connect to +## +## Usage: remote_port = <port> +## +remote_port = 4551 + +## +## Comma seperated list of remote MIHF's +## +## If you want to test remote MIHF communication add an entry here +## with the IP address of the remote MIHF. +## +## Usage: peers = <mihf id> <ip> <port> <transport protocol list>, ... +## +#peers = mihf2_ue 192.168.13.1 4551 udp, mihf3_ue 192.168.56.101 4551 udp +peers = mihf2_ue 10.0.0.2 4551 udp, mihf3_ue 192.168.14.4 4551 udp + +## +## Comma separated list of local MIH User SAPs id's and ports +## +## Usage: users = <user sap id> <port> [<supported commands> <supported queries>], ... +## Note: If no command is specified, the MIHF will assume that the MIH-User +## supports all MIH_***_HO_*** commands and the MIH_Get_Information +## Note: If no query is specified, the MIHF will assume that the MIH-User does +## not support the MIH_Get_Information command. +## +users = user_enb 1635 + +## +## Comma separated list of local MIH Link SAPs id's and ports. +## +## Usage: links = <link sap id> <port> <techonoly type> <interface> [<supported events list> <supported commands list>], ... +## +links = enb_lte_link00 1234 LTE 00:39:18:36:62:00 00:11:33 5, enb_eth_link00 1234 802_3 e0:db:55:eb:33:ac + +## +## Comma separated list of the MIHF's transport protocol +## +transport = udp diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient new file mode 100644 index 0000000000000000000000000000000000000000..29465e592af164bb52a3340f0900e1acead09fa9 Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient.cpp new file mode 100644 index 0000000000000000000000000000000000000000..8e384e776d8eca8d1971d825a8e01a4319574ee6 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient.cpp @@ -0,0 +1,260 @@ +#include <cpprest/http_client.h> +#include <cpprest/json.h> +#include <iostream> +#include <ostream> +#include <sstream> +#include <fstream> +#include "cpprest/basic_types.h" +#include "cpprest/asyncrt_utils.h" +#include "cpprest/uri.h" +#include <string> + +#include "CRMClient.hpp" + +using namespace std; +using namespace web; +using namespace web::http; +using namespace web::http::client; +using namespace utility; + + +CRMClient::CRMClient(char* url) +{ + m_url = (char*) (malloc (200* sizeof (char))); + m_url = url; +} + +CRMClient::CRMClient() +{ +} + +CRMClient::~CRMClient() +{ +} + +void CRMClient::SetUrl(char* url) +{ + m_url = url; +} + + +/****************************************************** +* Retrieves a JSON value from an HTTP request +* The JSON value is stored in the file "outputGET.txt" +*******************************************************/ + +pplx::task<void> CRMClient::RequestJSONValueAsync() +{ + http_client client (U(m_url)); + return client.request(methods::GET).then([](http_response response) -> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + ofstream myfile; + myfile.open ("outputGET.txt"); + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + +//Parsing Begin + + cout << U("Start Parsing...") << endl; + for(auto iterArray = v.cbegin(); iterArray != v.cend(); ++iterArray) + { + const json::value &arrayValue = iterArray->second; + + for(auto iterInner = arrayValue.cbegin(); iterInner != arrayValue.cend(); ++iterInner) + { + const json::value &propertyValue = iterInner->second; + for(auto iterlast = propertyValue.cbegin(); iterlast != propertyValue.cend(); ++iterlast) + { + const json::value &Name = iterlast->first; + const json::value &Value = iterlast->second; + cout<< U("Parameter: ") << Name.to_string()<<endl; + cout<< U("Value: ") << Value.to_string()<< std::endl; + } + } + cout << std::endl; + } + +//parsing End + myfile << jsonString.c_str(); + myfile.close(); + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +/******************************************************* +* Stores a JSON value (a Policy) using a HTTP request +********************************************************/ + +pplx::task<void> CRMClient::StoreJSONValuePolicies(char * param1, char * param2, char * param3, char * param4) +{ + http_client client (U(m_url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("pid"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("description"), json::value(param3))); + putvalue.push_back(make_pair(json::value("value"), json::value(param4))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); +} + + +/********************************************************** +* Stores a JSON value (a Measurement) using a HTTP request +***********************************************************/ + +pplx::task<void> CRMClient::StoreJSONValuemeasurements(char * param1, char * param2, char * param3, char * param4, char * param5, char * param6) +{ + + http_client client (U(m_url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("key"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("type"), json::value(param3))); + putvalue.push_back(make_pair(json::value("unit"), json::value(param4))); + putvalue.push_back(make_pair(json::value("value"), json::value(param5))); + putvalue.push_back(make_pair(json::value("time"), json::value(param6))); + + const string_t& s = ""; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + +/******************************************************** +* Stores a JSON value (a Decision) using a HTTP request +*********************************************************/ + +pplx::task<void> CRMClient::StoreJSONValuedecisions(char * param1, char * param2, char * param3, char * param4, char * param5) +{ + + http_client client (U(m_url)); + json::value::field_map putvalue; + + putvalue.push_back(make_pair(json::value("did"), json::value(param1))); + putvalue.push_back(make_pair(json::value("name"), json::value(param2))); + putvalue.push_back(make_pair(json::value("description"), json::value(param3))); + putvalue.push_back(make_pair(json::value("value"), json::value(param4))); + putvalue.push_back(make_pair(json::value("time"), json::value(param5))); + + const string_t& s = "/"; + json::value object = json::value::object(putvalue); + return client.request(methods::PUT, s, object).then([](http_response response)-> pplx::task<json::value> + { + if(response.status_code() == status_codes::OK) + { + return response.extract_json(); + } + + return pplx::task_from_result(json::value()); + }) + .then([](pplx::task<json::value> previousTask) + { + try + { + const json::value& v = previousTask.get(); + string_t jsonString = v.to_string(); + cout << U("Response...") << jsonString <<endl; + } + catch (const http_exception& e) + { + wostringstream ss; + ss << e.what() << endl; + wcout << ss.str(); + } + }); + +} + + +/******************************************************** +* Deletes entities using a HTTP request +*********************************************************/ + +pplx::task<void> CRMClient::Delete() +{ + char* url = m_url; + return pplx::create_task([url] + { + http_client client (U(url)); + + return client.request(methods::DEL); + + }).then([](http_response response) + { + if(response.status_code() == status_codes::OK) + { + auto body = response.extract_string(); + + std::wcout << L"Deleted: " << body.get().c_str() << std::endl; + } + }); +} diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient.hpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient.hpp new file mode 100644 index 0000000000000000000000000000000000000000..a49f2e62c8ca160514439aa590abb0c10be0d0f8 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClient.hpp @@ -0,0 +1,94 @@ +//============================================================================== +// Brief : MIH-User +// Authors : FATMA HRIZI <hrizi@eurecom.fr> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#ifndef ODTONE_CRM_CLIENT_HPP +#define ODTONE_CRM_CLIENT_HPP + + +#include <cpprest/http_client.h> +#include <cpprest/json.h> +#include <iostream> +#include <ostream> +#include <sstream> +#include "cpprest/basic_types.h" +#include "cpprest/asyncrt_utils.h" +#include "cpprest/uri.h" +#include <string> + + +/** + * CRM Client Class + * + * Defines the required functions to Get/Store Data from/in the CRRM + * + **/ +class CRMClient { + +public: + + CRMClient(); + CRMClient(char* url); + ~CRMClient(); + + void SetUrl(char* url); + + /** + * Get the data from the CRRM + * + * Returns JSON Format + */ + pplx::task<void> RequestJSONValueAsync(); + + /** + * Store Policies data from the CRRM + * + * Takes 4 parameters: P_ID, Name, Description, Value + * Returns the JSON format of the stored entity + */ + pplx::task<void> StoreJSONValuePolicies(char * param1, char * param2, char * param3, char * param4); + + /** + * Store Measurements data from the CRRM + * + * Takes 5 parameters: M_ID, Name, Type, Unit, Value, Time + * Returns the JSON format of the stored entity + */ + pplx::task<void> StoreJSONValuemeasurements(char * param1, char * param2, char * param3, char * param4, char * param5, char * param6); + + /** + * Store Decisions data from the CRRM + * + * Takes 4 parameters: D_ID, Name, Description, Value + * Returns the JSON format of the stored entity + */ + pplx::task<void> StoreJSONValuedecisions(char * param1, char * param2, char * param3, char * param4, char * param5); + + + + /** + * Delete Entities from the CRRM + * + * Returns the JSON format of the deleted entity + */ + pplx::task<void> Delete(); + + +private: + char * m_url; + +}; +#endif diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClientmain b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClientmain new file mode 100644 index 0000000000000000000000000000000000000000..971f08c4fe396ef85e8d71243da00c8afb041f7a Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/CRMClientmain differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/Jamfile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/Jamfile new file mode 100644 index 0000000000000000000000000000000000000000..c2a6aecbc45e3321cb97fafc4f51a161ac7b59ff --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/Jamfile @@ -0,0 +1,44 @@ +#=============================================================================== +# Brief : MIH-User SAP Application Sample Project Build +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +project enb_lte_user + ; + +exe enb_lte_user + : enb_lte_user.cpp + ../../lib/odtone//odtone + /boost//program_options + ; + +install install + : enb_lte_user + enb_lte_user.conf + ue_lte_user + ue_lte_user.conf + : <location>../../dist + ; + +project ue_lte_user + ; + +exe ue_lte_user + : ue_lte_user.cpp + ../../lib/odtone//odtone + /boost//program_options + ; + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/Makefile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..b6837ad7481d9d011d8148120bc6c86122b2d251 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/Makefile @@ -0,0 +1,19 @@ +main: main.cpp + g++-4.8 -Wall main.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -L. -std=c++11 -lcrmclient -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem -o CRMClientmain + +lib: CRMClient.cpp + +# 64 bits compilation +# g++-4.8 -Wall -fPIC -c CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -m64 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem +# g++-4.8 -shared -m64 -Wl,-soname,libcrmclient.so.1 -o libcrmclient.so.1.0 CRMClient.o +# ln -sf libcrmclient.so.1.0 libcrmclient.so +# ln -sf libcrmclient.so.1.0 libcrmclient.so.1 + +# 32 bits compilation + g++-4.8 -Wall -fPIC -c CRMClient.cpp -I /opt/casablanca/Release/include/ -L/usr/local/lib -L/opt/casablanca/Binaries/Release32/ -std=c++11 -lpthread -lboost_system -lboost_thread -lcasablanca -lboost_filesystem + g++-4.8 -shared -Wl,-soname,libcrmclient.so.1 -o libcrmclient.so.1.0 CRMClient.o + ln -sf libcrmclient.so.1.0 libcrmclient.so + ln -sf libcrmclient.so.1.0 libcrmclient.so.1 + +clean: + rm *.o *.so* diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/diffueenb b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/diffueenb new file mode 100644 index 0000000000000000000000000000000000000000..2babd6226795837b20fb8018bbeb827a52479665 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/diffueenb @@ -0,0 +1,201 @@ +difference between UE and eNB implementation: + +net_type_addr_list2string + +send_MIH_Link_Action_Power_Up_plus_scan_request: added in UE + +event_handler: +case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1;; + } + break; + + +No action in the odtone::mih::indication::link_parameters_report + +Define the eNB Scenario: + + +The UE Scenario: + +// The scenario coded in this MIH-USER demo is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// etc etc etc + + + + +the eNB scenario + +// +--------+ +-----+ +---------+ +----------+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | | LINK_SAP | | MIH-F | +// +---+----+ +--+--+ +----+----+ +----+-----+ +---------+ +// | | | | | +// ... (start of MIH-F here) ... ... ... | +// | |---------- Link_Capability_Discover.request --------X| | | +// ... (start of LINK_SAP here) ... ... | | +// | |<--------- Link_Register.indication -----------------| | | +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// ------------------------------------------------------------------------------------------------------------------------ +// Get the measurements from the UE +// ------------------------------------------------------------------------------------------------------------------------ + +Is this necessary? the link is not up by default?? +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +Is this necessary? the link is not up by default?? + +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | + ... +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | + +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| + + +// ------------------------------------------------------------------------------------------------------------------------ +// END OF THE EXECUTION +// ------------------------------------------------------------------------------------------------------------------------ + + + + + + + + + + + + + + + + + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER is the following (with eRALlteDummy and NASRGDummy executables) +// +--------+ +-----+ +// |MIH_USER| |MIH-F| +// +---+----+ +--+--+ +// | | _current_link_action_request = 0 +// |---------- User_Register.indication ---------------->| (supported_commands) Handler next msg=user_reg_handler +// | | +// |---------- Capability_Discover.request ------------->| Handler next msg=receive_MIH_Capability_Discover_confirm +// |<--------- Capability_Discover.confirm --------------| (success) +// | | +// |---------- Event_Subscribe.request ----------------->| Handler next msg=receive_MIH_Event_Subscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 1: Sequentially activate and deactivate each resource +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// |<--------- Link_Actions.confirm ---------------------| (success) +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 2: Activate all resources, then deactivate all resources +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | _current_link_action_request = 0 +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Event_Unsubscribe.request --------------->| Handler next msg=receive_MIH_Event_Unsubscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// | | +/////////////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/dummy_130606.tgz b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/dummy_130606.tgz new file mode 100644 index 0000000000000000000000000000000000000000..11ef083a922376ab0b713c88568902ae7483a5ce Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/dummy_130606.tgz differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/eNB_lte_user_tcs.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/eNB_lte_user_tcs.cpp new file mode 100644 index 0000000000000000000000000000000000000000..72d46e791bee1fde516e9f592ec1f77bdec6d605 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/eNB_lte_user_tcs.cpp @@ -0,0 +1,1412 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +// Fatma HRIZI EURECOM <hrizi@eurecom>fr> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +//This file is the implementation of the MIH user in the eNB + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER is the following (with eRALlteDummy and NASRGDummy executables) +// +--------+ +-----+ +// |MIH_USER| |MIH-F| +// +---+----+ +--+--+ +// | | _current_link_action_request = 0 +// |---------- User_Register.indication ---------------->| (supported_commands) Handler next msg=user_reg_handler +// | | +// |---------- Capability_Discover.request ------------->| Handler next msg=receive_MIH_Capability_Discover_confirm +// |<--------- Capability_Discover.confirm --------------| (success) +// | | +// |---------- Event_Subscribe.request ----------------->| Handler next msg=receive_MIH_Event_Subscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 1: Sequentially activate and deactivate each resource +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// |<--------- Link_Actions.confirm ---------------------| (success) +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 2: Activate all resources, then deactivate all resources +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | _current_link_action_request = 0 +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Event_Unsubscribe.request --------------->| Handler next msg=receive_MIH_Event_Unsubscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// | | +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s; + if(evtP.get(odtone::mih::mih_evt_link_detected)) s = std::string("DETECTED "); + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s; + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s = std::string("Link_Get_Parameters "); + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(const odtone::mih::config& cfg); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES) +//----------------------------------------------------------------------------- +{ + + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + log_(0, ""); + + // + // Local Capability Discover Request + // + odtone::mih::message msg; + _mihfid.assign("mihf2"); + msg << odtone::mih::request(odtone::mih::request::capability_discover, _mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Capability_Discover.request --->]["+msg.destination().to_string()+"]\n"); + _mihf.async_send(msg, boost::bind(&mih_user::receive_MIH_Capability_Discover_confirm, this, _1)); + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)"); + log_(0, ""); + + //****eNB***** + // Remote Capability Discover Request + // + mih_user::send_MIH_Capability_Discover_request(cfg); + + //Trigger Link_Configure_Thresholds Request + // odtone::mih::message m; + //m.destination(msg.source()); + //mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + //****eNB***** +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + } + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + + break; + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} +/* +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + log_(0, "MIH_Link_Detected.indication - End\n"); +} +*/ +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + +boost::system::error_code ec; + for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) + { + log_(0, " - LINK_ID - Link identifier: ", link_id2string(i->id).c_str()); + log_(0, " - Network ID: ", i->network_id); + log_(0, " - SINR: ", i->sinr); + log_(0, " - Data_rate: ", i->data_rate); + } + + send_MIH_Link_Configure_Thresholds_request(msg, ec); + + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (const odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { + + log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available + } + else{ + log_(0, " -Regular Report for BW Threshold "); + //update MEAS + } + } + + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + //boost::optional<odtone::mih::net_type_addr_list> ntal; + //boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::request(odtone::mih::request::capability_discover) + //& odtone::mih::tlv_net_type_addr_list(ntal) + //& odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + _mihfid.assign("mihf2"); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + odtone::mih::threshold th; + std::vector<odtone::mih::threshold> thl; + + odtone::mih::link_tuple_id lti; +// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + + + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp; + odtone::mih::link_param_lte lp; + + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + + odtone::mih::link_det_info_list ldil; + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) + { + + + lti.type = i->id.type; + lti.addr = i->id.addr; + lp = odtone::mih::link_param_lte_bandwidth; + lcp.type = lp; + if ( link_measures_request ==0){ + lcp.timer_interval = 10; + lcp.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } + else{ + lcp.timer_interval = 0; + lcp.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + th.threshold_val = 5; + th.threshold_x_dir = odtone::mih::threshold::above_threshold; + thl.push_back(th); + lcp.threshold_list = thl; + lcpl.push_back(lcp); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + link_id2string(lti).c_str()+ + " --->]["+_mihfid.to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + } + +/* + if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); + + if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", th.threshold_val); + + if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +*/ +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + // T odtone::uint iter; + // T odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + // Todtone::mih::link_cfg_status_list lcsl; + // Todtone::mih::link_cfg_status lcp; + //odtone::mih::link_param_gen lp; + + // T odtone::mih::link_tuple_id lti; + + //msg >> odtone::mih::confirm() + // & odtone::mih::tlv_status(st) + // & odtone::mih::tlv_link_identifier(lti) + // & odtone::mih::tlv_link_cfg_status_list(lcsl); + + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1234), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/enb_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/enb_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..b9bbf279de1650cfa71dbeef31960214086d0d98 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/enb_lte_user.conf @@ -0,0 +1,40 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_enb + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/enb_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/enb_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..80fbfe7522b05fd8b069218bb39f42dea0d55425 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/enb_lte_user.cpp @@ -0,0 +1,1695 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <string> +#include <map> +#include <stdio.h> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +//The thresholds for the Energy Detection Sensing algorithm respectively for 10 and 100 samples +#define ED_THRESHOLD_10 23695432 +#define ED_THRESHOLD_100 230445932 +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER (of the eNB) for the demo of the Scenario 2 of SPECTRA project is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Initiallization of the MIH-USER and the MIHF +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | + +// ------------------------------------------------------------------------------------------------------------------------ +// Locally send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to local MIHF) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | + +// ------------------------------------------------------------------------------------------------------------------------ +// Remotely send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to remote MIHF of the UE) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Detect the connection of the UE to the eNB + Start the measurement report process +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Activate the TVWS link (After running the cognitive algorithms) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |-------------- MIH_Link_Actions.request ------------>|--------------- MIH_Link_Actions.request ----------->| +// | | | +// |<------------- MIH_Link_Actions.confirm -------------|<-------------- MIH_Link_Actions.confirm ------------| +// | | | + +//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + +static const char* const kConf_MIH_Commands = "user.commands"; + +///////////////////////////////////////////////////////////////////////////////////////// + +std::string exec(char* cmd) { + FILE* pipe = popen(cmd, "r"); + if (!pipe) return "ERROR"; + char buffer[128]; + std::string result = ""; + while(!feof(pipe)) { + if(fgets(buffer, 128, pipe) != NULL) + result += buffer; + } + pclose(pipe); + return result; +} + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} + + +/*//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} +*/ + +//Updated from UE code +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + + //Measurements report methods + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + int receive_Sensing_Report(); + void receive_CRRM_Data(); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + void forward_Parameters_Report_indication(odtone::mih::message& m); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + odtone::uint second_link_activated; + odtone::uint count; + odtone::uint sensing_done; + odtone::uint sensing_score; + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + second_link_activated = 0; + count = 0; + sensing_done = 0; + sensing_score = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + + //send a capability discover request to the remote UE + mih_user::send_MIH_Capability_Discover_request_remote(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + boost::system::error_code ec; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} +// mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, /*boost::cref(cfg),*/ _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + //for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + //{ + // rcv_link_id.addr = i->addr; + // rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + //} + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + + std::cout<<"NTALL "<<ntal.get()[0].addr<<std::endl; + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; +// std::ostringstream stream; +// std::stringstream st; +// odtone::mih::net_type_addr net_type_addr; +// net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); +// stream << net_type_addr; +// std::string s; +// std::stringstream st (stream.str()); +// while (getline (st, s, '\n')) +// { +// std::stringstream ss(s); +// getline (ss, name, ':'); +// getline (ss, value, '\n'); +// // ss>>name>>value; +// std::cout<<" name "<<name<<" value "<<value<<std::endl; +// } +/* + st<<i->addr; + getline (st, s, ' '); + std::cout<<"LINK_TUPLE_ID s "<<s<<std::endl; + + odtone::mih::l2_3gpp_addr add; + add.value = s;*/ + li.addr = /*add*/ /*boost::get<odtone::mih::l2_3gpp_addr>(*/i->addr/*)*/; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + if (std::find(_link_id_list.begin(), _link_id_list.end(), li)==_link_id_list.end()) + _link_id_list.push_back(li); // save the link identifier of the network interface + + std::cout<<"LINK_TUPLE_ID - Link identifier:after "<<link_addr2string(&li.addr).c_str() <<" "<<_link_id_list[_link_id_list.size()-1]<<std::endl; + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<li<<std::endl; +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + link_act_req.id = link; +// } + + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + std::cout<<"_link received from parameters report "<<link<<std::endl; + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + } + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + +// link_act_req.action.param.param = res; + +// link_act_req.ex_time = 0; + link_act_req.ex_time = 5000; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + +// odtone::mih::id mid_ue; +// mid_ue.assign("mihf2_ue"); + m.destination(/*mid_ue*/_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st); +// & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); +// if (larl) { +// log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); +// for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) +// { +// log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), +// ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); +// } +// } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +// #ifdef SCENARIO_1 +// if (larl) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// if (rsp->result.get() == odtone::mih::link_ac_success) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// _current_link_action_request += 1; +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_1 +// +// #ifdef SCENARIO_2 +// // 2nd scenario: Activate all resources, then deactivate all resources +// if (larl.get().size() > 0) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (++_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// _current_link_action_request = 0; +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +void mih_user::receive_CRRM_Data() +{ + + //CRRM data report + std::string result = exec ("./CRMClientmain 0 1"); +} + +//----------------------------------------------------------------------------- +int mih_user::receive_Sensing_Report() +//----------------------------------------------------------------------------- +{ + std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); + std::string tmp; + int value; + std::stringstream ss(result); + ss >> tmp >> value; + std::cout<<"Result of Sensing "<<value<<std::endl; + sensing_done = 1; + return value; +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link, link1; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<link<<std::endl; + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value ); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + + + //eNB1: Forward the message to UE 2 + forward_Parameters_Report_indication(msg); + + //eNB2 : Action Power Up the TVWS Link after running the cognitive algorithm + + //First Phase: Collect the data as input fot the cognitive algorithm + //Sensing data + if (sensing_done == 0) + sensing_score = receive_Sensing_Report(); + + //CRRM Data + receive_CRRM_Data(); + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { +// log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// if(*i==link) + odtone::mih::l2_3gpp_addr add; + odtone::mih::link_type type; + type = odtone::mih::link_type_lte; +// add.value = "l2_3gpp_addr"; + link1.addr = add; + link1.type = type; +// send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_power_up); + + //Send ONLY ONCE RRC_Connection_Reconfiguration request to the UE on LTE link 0 + if (second_link_activated==0) + { + if (count == 20) + { + send_MIH_Link_Action_Power_Up_request(_link_id_list[1]);//Link LTE + second_link_activated = 1; + } + } + + count++; +// continue; +// send_MIH_Link_Action_Power_Up_request(link); +// if (link_id2string(link).c_str() == "LTE") +// } + } + } + +} + + +//Forward the MIH_Link_Parameters_Report to the UE of the CPE +//----------------------------------------------------------------------------- +void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + m >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + odtone::mih::message msg; + msg.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + msg.destination(mid_ue); + + //mn_ho_candidate_query is used to constuct/send the message containing parameters reports + msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- before MIH_User_PARAMETERS.indication --->]["+msg.destination().to_string()+"]\n"); + + + _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, "MIH_User_PARAMETERS.indication- SENT (towards UE)\n"); + +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl1; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lcp1.type = lp1; + lcp2.type = lp2; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th2.threshold_val = -19; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + + + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ " --->][ ADDR LINK " + + m.destination().to_string() +"]\n"); + std::cout<<"LINK TUPLE ID "<<lti<<std::endl; + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti); +// & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", (lcsl[iter].thold.threshold_val/256)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("enb_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1_enb"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/ue_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/ue_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..27d56760bea01f85150e9f7e5d21214e5bbf1ae3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/ue_lte_user.conf @@ -0,0 +1,42 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_ue + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/ue_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/ue_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..896958f62bfeace56eff5786582169244001e38d --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/mih_user/lte_test_user/ue_lte_user.cpp @@ -0,0 +1,1399 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +//#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER demo is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// etc etc etc + +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + + + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + void send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + // send Link_Action / Power Up + lid.type = odtone::mih::link_type_lte; + lid.addr = ldi.id.addr; + //send_MIH_Link_Action_Power_Up_plus_scan_request(lid); + log_(0, "MIH_Link_Detected.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + + link_act_req.id = link; + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); + mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(link); + +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th; + std::vector<odtone::mih::threshold> thl; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp; + odtone::mih::link_param_lte lp; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp = odtone::mih::link_param_lte_rsrp; + lcp.type = lp; + + link_measures_request = 0; + if ( link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp.timer_interval = 0; + lcp.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th.threshold_val = -105; + th.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl.push_back(th); + lcp.threshold_list = thl; + lcpl.push_back(lcp); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+_mihfid.to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); + + if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", th.threshold_val); + + if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + // T odtone::uint iter; + // T odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + // Todtone::mih::link_cfg_status_list lcsl; + // Todtone::mih::link_cfg_status lcp; + //odtone::mih::link_param_gen lp; + + // T odtone::mih::link_tuple_id lti; + + //msg >> odtone::mih::confirm() + // & odtone::mih::tlv_status(st) + // & odtone::mih::tlv_link_identifier(lti) + // & odtone::mih::tlv_link_cfg_status_list(lcsl); + + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("ue_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf2_ue"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/oai_conf/enb.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/oai_conf/enb.conf new file mode 100644 index 0000000000000000000000000000000000000000..8acd6d84afc6530bc1c366e11e4424a3913ed2bb --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/oai_conf/enb.conf @@ -0,0 +1,217 @@ +Active_eNBs = ( "eNB_Eurecom_LTEBox"); +# Asn1_verbosity, choice in: none, info, annoying +Asn1_verbosity = "none"; + +eNBs = +( + { + # real_time choice in {hard, rt-preempt, no} + real_time = "no"; + ////////// Identification parameters: + eNB_ID = 0xe00; + + cell_type = "CELL_MACRO_ENB"; + + eNB_name = "eNB_Eurecom_LTEBox"; + + // Tracking area code, 0x0000 and 0xfffe are reserved values + tracking_area_code = "1"; + + mobile_country_code = "208"; + + mobile_network_code = "92"; + + ////////// Physical parameters: + + component_carriers = ( + { + frame_type = "TDD"; + tdd_config = 3; + tdd_config_s = 0; + prefix_type = "NORMAL"; + eutra_band = 33; + downlink_frequency = 1910000000L; + uplink_frequency_offset = 0; + + Nid_cell = 0; + N_RB_DL = 25; + Nid_cell_mbsfn = 0; + nb_antennas_tx = 1; + nb_antennas_rx = 1; + prach_root = 22; + prach_config_index = 3; + prach_high_speed = "DISABLE"; + prach_zero_correlation = 0; + prach_freq_offset = 0; + pucch_delta_shift = 1; + pucch_nRB_CQI = 1; + pucch_nCS_AN = 0; + pucch_n1_AN = 32; + pdsch_referenceSignalPower = -24; + pdsch_p_b = 0; + pusch_n_SB = 1; + pusch_enable64QAM = "DISABLE"; + pusch_hoppingMode = "interSubFrame"; + pusch_hoppingOffset = 4; + pusch_groupHoppingEnabled = "DISABLE"; + pusch_groupAssignment = 0; + pusch_sequenceHoppingEnabled = "DISABLE"; + pusch_nDMRS1 = 1; + phich_duration = "NORMAL"; + phich_resource = "ONESIXTH"; + srs_enable = "DISABLE"; + # srs_BandwidthConfig =; + # srs_SubframeConfig =; + # srs_ackNackST =; + # srs_MaxUpPts =; + + pusch_p0_Nominal = -95; + pusch_alpha = "AL08"; + pucch_p0_Nominal = -117; + msg3_delta_Preamble = 6; + pucch_deltaF_Format1 = "deltaF2"; + pucch_deltaF_Format1b = "deltaF3"; + pucch_deltaF_Format2 = "deltaF0"; + pucch_deltaF_Format2a = "deltaF0"; + pucch_deltaF_Format2b = "deltaF0"; + + rach_numberOfRA_Preambles = 52; + rach_preamblesGroupAConfig = "DISABLE"; + +# rach_sizeOfRA_PreamblesGroupA = ; +# rach_messageSizeGroupA = ; +# rach_messagePowerOffsetGroupB = ; + + rach_powerRampingStep = 2; + rach_preambleInitialReceivedTargetPower = -104; + rach_preambleTransMax = 6; + rach_raResponseWindowSize = 10; + rach_macContentionResolutionTimer = 48; + rach_maxHARQ_Msg3Tx = 4; + + pcch_default_PagingCycle = 128; + pcch_nB = "oneT"; + bcch_modificationPeriodCoeff = 2; + ue_TimersAndConstants_t300 = 1000; + ue_TimersAndConstants_t301 = 1000; + ue_TimersAndConstants_t310 = 1000; + ue_TimersAndConstants_t311 = 10000; + ue_TimersAndConstants_n310 = 20; + ue_TimersAndConstants_n311 = 1; + + }, + { + frame_type = "TDD"; + tdd_config = 3; + tdd_config_s = 0; + prefix_type = "NORMAL"; + eutra_band = 33; + downlink_frequency = 1910000000L; + uplink_frequency_offset = 0; + + Nid_cell = 0; + N_RB_DL = 25; + Nid_cell_mbsfn = 0; + nb_antennas_tx = 1; + nb_antennas_rx = 1; + prach_root = 22; + prach_config_index = 3; + prach_high_speed = "DISABLE"; + prach_zero_correlation = 0; + prach_freq_offset = 0; + pucch_delta_shift = 1; + pucch_nRB_CQI = 1; + pucch_nCS_AN = 0; + pucch_n1_AN = 32; + pdsch_referenceSignalPower = -24; + pdsch_p_b = 0; + pusch_n_SB = 1; + pusch_enable64QAM = "DISABLE"; + pusch_hoppingMode = "interSubFrame"; + pusch_hoppingOffset = 4; + pusch_groupHoppingEnabled = "DISABLE"; + pusch_groupAssignment = 0; + pusch_sequenceHoppingEnabled = "DISABLE"; + pusch_nDMRS1 = 1; + phich_duration = "NORMAL"; + phich_resource = "ONESIXTH"; + srs_enable = "DISABLE"; + /* srs_BandwidthConfig =; + srs_SubframeConfig =; + srs_ackNackST =; + srs_MaxUpPts =;*/ + + pusch_p0_Nominal = -95; + pusch_alpha = "AL08"; + pucch_p0_Nominal = -117; + msg3_delta_Preamble = 6; + pucch_deltaF_Format1 = "deltaF2"; + pucch_deltaF_Format1b = "deltaF3"; + pucch_deltaF_Format2 = "deltaF0"; + pucch_deltaF_Format2a = "deltaF0"; + pucch_deltaF_Format2b = "deltaF0"; + + rach_numberOfRA_Preambles = 52; + rach_preamblesGroupAConfig = "DISABLE"; +/* + rach_sizeOfRA_PreamblesGroupA = ; + rach_messageSizeGroupA = ; + rach_messagePowerOffsetGroupB = ; +*/ + rach_powerRampingStep = 2; + rach_preambleInitialReceivedTargetPower = -104; + rach_preambleTransMax = 6; + rach_raResponseWindowSize = 10; + rach_macContentionResolutionTimer = 48; + rach_maxHARQ_Msg3Tx = 4; + + pcch_default_PagingCycle = 128; + pcch_nB = "oneT"; + bcch_modificationPeriodCoeff = 2; + ue_TimersAndConstants_t300 = 1000; + ue_TimersAndConstants_t301 = 1000; + ue_TimersAndConstants_t310 = 1000; + ue_TimersAndConstants_t311 = 10000; + ue_TimersAndConstants_n310 = 20; + ue_TimersAndConstants_n311 = 1; + + } + ); + + ////////// MME parameters: + mme_ip_address = ( { ipv4 = "192.168.13.11"; + ipv6 = "192:168:30::17"; + active = "yes"; + preference = "ipv4"; + } + ); + + NETWORK_INTERFACES : + { + ENB_INTERFACE_NAME_FOR_S1_MME = "eth1"; + ENB_IPV4_ADDRESS_FOR_S1_MME = "192.168.13.10/24"; + + ENB_INTERFACE_NAME_FOR_S1U = "eth1"; + ENB_IPV4_ADDRESS_FOR_S1U = "192.168.13.10/24"; + }; + + log_config : + { + global_log_level ="debug"; + global_log_verbosity ="medium"; + hw_log_level ="debug"; + hw_log_verbosity ="medium"; + phy_log_level ="info"; + phy_log_verbosity ="medium"; + mac_log_level ="debug"; + mac_log_verbosity ="high"; + rlc_log_level ="info"; + rlc_log_verbosity ="medium"; + pdcp_log_level ="info"; + pdcp_log_verbosity ="medium"; + rrc_log_level ="info"; + rrc_log_verbosity ="medium"; + }; + + } +); diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/oai_conf/start_enb.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/oai_conf/start_enb.bash new file mode 100644 index 0000000000000000000000000000000000000000..70f254ffd275285087ebe8ed5679522b664cef3b --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb1/oai_conf/start_enb.bash @@ -0,0 +1,196 @@ +#!/bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file start_enb.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# +#------------------------------------------------ +# ENB CONFIG FILE +#------------------------------------------------ +#declare -x ENB_CONFIG_FILE="CONF/enb.sfr.yang.conf" +declare -x ENB_CONFIG_FILE="enb.conf" + +#------------------------------------------------ +# OAI NETWORKING +#------------------------------------------------ +declare -x EMULATION_DEV_INTERFACE="eth1" +declare -x EMULATION_DEV_ADDRESS="192.168.13.1" +declare -x IP_DRIVER_NAME="oai_nw_drv" +declare -x LTEIF="oai0" +declare -x ENB_IPv4="10.0.0.1" +declare -x ENB_IPv6="2001:1::1" +declare -x ENB_IPv6_CIDR=$ENB_IPv6"/64" +declare -x ENB_IPv4_CIDR=$ENB_IPv4"/24" +declare -a NAS_IMEI=( 3 9 1 8 3 6 6 2 0 0 0 0 0 0 ) +declare -x IP_DEFAULT_MARK="1" # originally 3 +#------------------------------------------------ +# OAI MIH +#------------------------------------------------ +declare -x ENB_RAL_IP_ADDRESS="127.0.0.1" +declare -x ENB_MIHF_IP_ADDRESS=127.0.0.1 +MIH_LOG_FILE="mih-f_enb.log" + +#------------------------------------------------ +LOG_FILE="/tmp/oai_sim_enb.log" + +########################################################### +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +source $THIS_SCRIPT_PATH/env_802dot21.bash +########################################################### +bash_exec "ifconfig $EMULATION_DEV_INTERFACE up $EMULATION_DEV_ADDRESS netmask 255.255.255.0" +bash_exec "ifconfig eth2 up 192.168.14.3 netmask 255.255.255.0" +bash_exec "ip r d default dev eth0" +bash_exec "ip r a default via 192.168.14.4 dev eth2" +########################################################### +IPTABLES=/sbin/iptables +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +declare -x OPENAIR_DIR="" +declare -x OPENAIR1_DIR="" +declare -x OPENAIR2_DIR="" +declare -x OPENAIR3_DIR="" +declare -x OPENAIR_TARGETS="" +########################################################### + +set_openair +cecho "OPENAIR_DIR = $OPENAIR_DIR" $green +cecho "OPENAIR1_DIR = $OPENAIR1_DIR" $green +cecho "OPENAIR2_DIR = $OPENAIR2_DIR" $green +cecho "OPENAIR3_DIR = $OPENAIR3_DIR" $green +cecho "OPENAIR_TARGETS = $OPENAIR_TARGETS" $green + +bash_exec "/sbin/iptables -t mangle -F" +bash_exec "/sbin/iptables -t nat -F" +bash_exec "/sbin/iptables -t raw -F" +bash_exec "/sbin/iptables -t filter -F" +bash_exec "/sbin/ip6tables -t mangle -F" +bash_exec "/sbin/ip6tables -t filter -F" +bash_exec "/sbin/ip6tables -t raw -F" + +################################################## +# LAUNCH eNB executable +################################################## + +echo "Bringup eNB interface" +pkill oaisim > /dev/null 2>&1 +pkill oaisim > /dev/null 2>&1 +pkill $MIH_F > /dev/null 2>&1 +pkill $ENB_MIH_USER > /dev/null 2>&1 +rmmod -f $IP_DRIVER_NAME > /dev/null 2>&1 + +bash_exec "insmod $OPENAIR2_DIR/NAS/DRIVER/LITE/$IP_DRIVER_NAME.ko oai_nw_drv_IMEI=${NAS_IMEI[0]},${NAS_IMEI[1]},${NAS_IMEI[2]},${NAS_IMEI[3]},${NAS_IMEI[4]},${NAS_IMEI[5]},${NAS_IMEI[6]},${NAS_IMEI[7]},${NAS_IMEI[8]},${NAS_IMEI[9]},${NAS_IMEI[10]},${NAS_IMEI[11]},${NAS_IMEI[12]},${NAS_IMEI[13]}" +bash_exec "ip route flush cache" +bash_exec "ip link set $LTEIF up" +sleep 1 +bash_exec "ip addr add dev $LTEIF $ENB_IPv4_CIDR" +bash_exec "ip addr add dev $LTEIF $ENB_IPv6_CIDR" +sleep 1 +bash_exec "sysctl -w net.ipv4.conf.all.log_martians=1" +assert " `sysctl -n net.ipv4.conf.all.log_martians` -eq 1" $LINENO +bash_exec "sysctl -w net.ipv4.conf.all.rp_filter=0" +assert " `sysctl -n net.ipv4.conf.all.rp_filter` -eq 0" $LINENO +bash_exec "ip route flush cache" +bash_exec "sysctl -w net.ipv4.ip_forward=1" +assert " `sysctl -n net.ipv4.ip_forward` -eq 1" $LINENO + +# Check table 200 lte in /etc/iproute2/rt_tables +fgrep lte /etc/iproute2/rt_tables > /dev/null +if [ $? -ne 0 ]; then + echo '200 lte ' >> /etc/iproute2/rt_tables +fi +ip rule add fwmark $IP_DEFAULT_MARK table lte +ip route add default dev $LTEIF table lte +ip route add 239.0.0.160/28 dev $EMULATION_DEV_INTERFACE + +/sbin/ebtables -t nat -A POSTROUTING -p arp -j mark --mark-set 3 + +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +#All other traffic is sent on the RAB you want (mark = RAB ID) +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK + +rotate_log_file $MIH_LOG_FILE + + +# start MIH-F +xterm -hold -title "[RELAY][eNB1] MIHF" -e $ODTONE_MIH_EXE_DIR/$MIH_F --log 4 --conf.file $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE > $MIH_LOG_FILE 2>&1 & +wait_process_started $MIH_F + +NOW=$(date +"%Y-%m-%d.%Hh_%Mm_%Ss") +rm -f $LOG_FILE + +ENB_RAL_LINK_ID=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +ENB_RAL_LINK_ID=`trim2 $ENB_RAL_LINK_ID` +ENB_RAL_LINK_ID=`echo $ENB_RAL_LINK_ID | cut -d" " -f1` + +ENB_RAL_LISTENING_PORT=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +ENB_RAL_LISTENING_PORT=`trim2 $ENB_RAL_LISTENING_PORT` +ENB_RAL_LISTENING_PORT=`echo $ENB_RAL_LISTENING_PORT | cut -d" " -f2` + +ENB_MIHF_REMOTE_PORT=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep local_port | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` + +ENB_MIHF_ID=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep id | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` +#remove 2 last digits (vitualization, index on 2 digits) +ENB_RAL_LINK_ID_STRIPPED=${ENB_RAL_LINK_ID%%??} + +#xterm -hold -e gdb --args +$OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l9 -u0 -b1 -M0 -p2 -g1 -D $EMULATION_DEV_INTERFACE \ + --enb-ral-listening-port $ENB_RAL_LISTENING_PORT \ + --enb-ral-link-id $ENB_RAL_LINK_ID_STRIPPED \ + --enb-ral-ip-address $ENB_RAL_IP_ADDRESS \ + --enb-mihf-remote-port $ENB_MIHF_REMOTE_PORT \ + --enb-mihf-ip-address $ENB_MIHF_IP_ADDRESS \ + --enb-mihf-id $ENB_MIHF_ID \ + -O $ENB_CONFIG_FILE > log_enb.txt & +# -O $ENB_CONFIG_FILE | grep "RAL\|PDCP" & + +wait_process_started oaisim + +# start MIH-USER +# wait for emulation start +tshark -c 150 -i $EMULATION_DEV_INTERFACE > /dev/null 2>&1 +sudo xterm -hold -title "[RELAY][eNB1] MIH_USER" -e $ODTONE_MIH_EXE_DIR/$ENB_MIH_USER --conf.file $ODTONE_MIH_EXE_DIR/$ENB_MIH_USER_CONF_FILE & +wait_process_started $ENB_MIH_USER + +xterm -hold -title "[RELAY][eNB1] CRM Client" -e tail -f outputGET.txt & + +sleep 100000 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/enb2_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/enb2_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..356a52abbb17636a9cfc05bb51cd6e033198c8d0 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/enb2_lte_user.conf @@ -0,0 +1,40 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user2_enb + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/link_sap.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/link_sap.conf new file mode 100644 index 0000000000000000000000000000000000000000..c5b63413d0148de7186c2fac37cf9e606dd592df --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/link_sap.conf @@ -0,0 +1,47 @@ +#=============================================================================== +# Brief : Link SAP configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[link] +## +## Link SAP identifier +## +id=link1 + +## +## Link SAP listening +## +port = 1235 + +## +## Link SAP interface technology +## +tec = 802_11 + +## +## Link SAP interface address +## +link_addr = 00:11:22:33:44:55 + +## +## Comma separated list of the Link SAP supported events +## +event_list = link_detected, link_up, link_down, link_parameters_report, link_going_down, link_handover_imminent, link_handover_complete + +[mihf] +ip=127.0.0.1 +local_port=1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/odtone_enb.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/odtone_enb.conf new file mode 100644 index 0000000000000000000000000000000000000000..56a1f5e89e1f4b52538d66703c3e4c7975da1271 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_conf/odtone_enb.conf @@ -0,0 +1,73 @@ +#=============================================================================== +# Brief : MIHF configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[mihf] +## +## This mihf's id +## +## Usage: id = <MIHF ID> +## +id = mihf4_enb + +## +## Port on localhost that MIH Users and MIH Link SAPs connect to. +## +## Usage: local_port = <port> +## +local_port = 1025 + +## +## Port to which remote peer MIHF connect to +## +## Usage: remote_port = <port> +## +remote_port = 4551 + +## +## Comma seperated list of remote MIHF's +## +## If you want to test remote MIHF communication add an entry here +## with the IP address of the remote MIHF. +## +## Usage: peers = <mihf id> <ip> <port> <transport protocol list>, ... +## +#peers = mihf3_ue 10.0.0.3 4551 udp +peers = mihf3_ue 10.0.2.3 4551 udp + +## +## Comma separated list of local MIH User SAPs id's and ports +## +## Usage: users = <user sap id> <port> [<supported commands> <supported queries>], ... +## Note: If no command is specified, the MIHF will assume that the MIH-User +## supports all MIH_***_HO_*** commands and the MIH_Get_Information +## Note: If no query is specified, the MIHF will assume that the MIH-User does +## not support the MIH_Get_Information command. +## +users = user_enb 1635 + +## +## Comma separated list of local MIH Link SAPs id's and ports. +## +## Usage: links = <link sap id> <port> <techonoly type> <interface> [<supported events list> <supported commands list>], ... +## +#links = enb_lte_link00 1234 LTE 00:39:18:36:62:00 00:11:33 5, enb_eth_link00 1234 802_3 e0:db:55:eb:33:ac +links = enb_lte_link00 1234 LTE 00:39:18:36:62:00 00:11:33 5 + +## +## Comma separated list of the MIHF's transport protocol +## +transport = udp diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/Jamfile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/Jamfile new file mode 100644 index 0000000000000000000000000000000000000000..4f1c6cb43f727db278cdde12cfc6b732f2aa9882 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/Jamfile @@ -0,0 +1,31 @@ +#=============================================================================== +# Brief : MIH-User SAP Application Sample Project Build +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +project enb2_lte_user + ; + +exe enb2_lte_user + : enb2_lte_user.cpp + ../../lib/odtone//odtone + /boost//program_options + ; + +install install + : enb2_lte_user + : <location>../../dist + ; diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb2_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb2_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..c5bc9e58879edb40db5ecfd83cb7dfafc4332144 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb2_lte_user.cpp @@ -0,0 +1,1786 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <string> +#include <map> +#include <stdio.h> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +//The thresholds for the Energy Detection Sensing algorithm respectively for 10 and 100 samples +#define ED_THRESHOLD_10 23695432 +#define ED_THRESHOLD_100 230445932 +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER (of the eNB) for the demo of the Scenario 2 of SPECTRA project is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Initiallization of the MIH-USER and the MIHF +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | + +// ------------------------------------------------------------------------------------------------------------------------ +// Locally send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to local MIHF) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | + +// ------------------------------------------------------------------------------------------------------------------------ +// Remotely send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to remote MIHF of the UE) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Detect the connection of the UE to the eNB + Start the measurement report process +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Activate the TVWS link (After running the cognitive algorithms) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |-------------- MIH_Link_Actions.request ------------>|--------------- MIH_Link_Actions.request ----------->| +// | | | +// |<------------- MIH_Link_Actions.confirm -------------|<-------------- MIH_Link_Actions.confirm ------------| +// | | | + +//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + +static const char* const kConf_MIH_Commands = "user.commands"; + +///////////////////////////////////////////////////////////////////////////////////////// + +std::string exec(char* cmd) { + FILE* pipe = popen(cmd, "r"); + if (!pipe) return "ERROR"; + char buffer[128]; + std::string result = ""; + while(!feof(pipe)) { + if(fgets(buffer, 128, pipe) != NULL) + result += buffer; + } + pclose(pipe); + return result; +} + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} + + +/*//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} +*/ + +//Updated from UE code +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + + //Measurements report methods + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + int receive_Sensing_Report(); + void receive_CRRM_Data(); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + void forward_Parameters_Report_indication(odtone::mih::message& m); + + void receive_MIH_MN_HO_Candidate_Query_request(odtone::mih::message& msg); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + odtone::uint second_link_activated; + odtone::uint count; + odtone::uint sensing_done; + odtone::uint sensing_score; + + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + second_link_activated = 0; + count = 0; + sensing_done = 0; + sensing_score = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + + //send a capability discover request to the remote UE + mih_user::send_MIH_Capability_Discover_request_remote(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + + + case odtone::mih::indication::mn_ho_candidate_query: + log_(0, "MIH-User has received a request \"mn_ho_candidate_query\""); + mih_user::receive_MIH_MN_HO_Candidate_Query_request(msg); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + boost::system::error_code ec; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} +// mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, /*boost::cref(cfg),*/ _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address +// for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) +// { +// rcv_link_id.addr = i->addr; +// rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); +// } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + + std::cout<<"NTALL "<<ntal.get()[0].addr<<std::endl; + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; +// std::ostringstream stream; +// std::stringstream st; +// odtone::mih::net_type_addr net_type_addr; +// net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); +// stream << net_type_addr; +// std::string s; +// std::stringstream st (stream.str()); +// while (getline (st, s, '\n')) +// { +// std::stringstream ss(s); +// getline (ss, name, ':'); +// getline (ss, value, '\n'); +// // ss>>name>>value; +// std::cout<<" name "<<name<<" value "<<value<<std::endl; +// } +/* + st<<i->addr; + getline (st, s, ' '); + std::cout<<"LINK_TUPLE_ID s "<<s<<std::endl; + + odtone::mih::l2_3gpp_addr add; + add.value = s;*/ +// li.addr = /*add*/ boost::get<odtone::mih::l2_3gpp_addr>(i->addr); +// li.type = boost::get<odtone::mih::link_type>(i->nettype.link); +// if (std::find(_link_id_list.begin(), _link_id_list.end(), li)==_link_id_list.end()) +// _link_id_list.push_back(li); // save the link identifier of the network interface + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + + std::cout<<"LINK_TUPLE_ID - Link identifier:after "<<link_addr2string(&li.addr).c_str() <<" "<<_link_id_list[_link_id_list.size()-1]<<std::endl; + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<li<<std::endl; +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + link_act_req.id = link; +// } + + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + std::cout<<"_link received from parameters report "<<link<<std::endl; + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + } + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + +// link_act_req.action.param.param = res; + +// link_act_req.ex_time = 0; + link_act_req.ex_time = 5000; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + +// odtone::mih::id mid_ue; +// mid_ue.assign("mihf2_ue"); + m.destination(/*mid_ue*/_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st); +// & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); +// if (larl) { +// log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); +// for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) +// { +// log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), +// ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); +// } +// } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +// #ifdef SCENARIO_1 +// if (larl) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// if (rsp->result.get() == odtone::mih::link_ac_success) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// _current_link_action_request += 1; +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_1 +// +// #ifdef SCENARIO_2 +// // 2nd scenario: Activate all resources, then deactivate all resources +// if (larl.get().size() > 0) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (++_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// _current_link_action_request = 0; +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +// void mih_user::cognitive_decision() +// { +// +// //Sensing measurements report +// std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); +// } + + +void mih_user::receive_CRRM_Data() +{ + + //CRRM data report + std::string result = exec ("./CRMClientmain 0 1"); +} + + +//----------------------------------------------------------------------------- +int mih_user::receive_Sensing_Report() +//----------------------------------------------------------------------------- +{ + std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); + std::string tmp; + int value; + std::stringstream ss(result); + ss >> tmp >> value; + sensing_done = 1; +// std::cout<<"Result "<<result<<std::endl; + return value; +} + +void mih_user::receive_MIH_MN_HO_Candidate_Query_request(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::request() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_MN_HO_Candidate_Query.request - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value ); + } + } + +//eNB2 : Action Power Up the TVWS Link after running the cognitive algorithm + + //First Phase: Collect the data as input fot the cognitive algorithm + //Sensing data + if (sensing_done == 0) + sensing_score = receive_Sensing_Report(); + + //CRRM Data + // receive_CRRM_Data(); + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { +// log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// if(*i==link) + odtone::mih::l2_3gpp_addr add; + odtone::mih::link_type type; + type = odtone::mih::link_type_lte; +// add.value = "l2_3gpp_addr"; +// link1.addr = add; +// link1.type = type; +// send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_power_up); + + //Send ONLY ONCE RRC_Connection_Reconfiguration request to the UE on LTE link 0 + log_(0, "COUNT: ", count ); + if (second_link_activated==0) + { + log_(0, "COUNT: ", count ); + if (count >= 30) + { + log_(0, "COUNT: ", count ); + send_MIH_Link_Action_Power_Up_request(_link_id_list[0]);//Link LTE + second_link_activated = 1; + } + } + + count++; +// continue; +// send_MIH_Link_Action_Power_Up_request(link); +// if (link_id2string(link).c_str() == "LTE") +// } + } + } + +} + + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link, link1; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<link<<std::endl; + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value ); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + + + //eNB1: Forward the message to UE 2 + // forward_Parameters_Report_indication(msg); + + //eNB2 : Action Power Up the TVWS Link after running the cognitive algorithm + + //First Phase: Collect the data as input fot the cognitive algorithm + //Sensing data +// int score; +// score = receive_Sensing_Report(); +// + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { +// log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); +// //link_action if free channel is available +// // for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// // if(*i==link) +// odtone::mih::l2_3gpp_addr add; +// odtone::mih::link_type type; +// type = odtone::mih::link_type_lte; +// // add.value = "l2_3gpp_addr"; +// link1.addr = add; +// link1.type = type; +// send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_power_up); + + //Send ONLY ONCE RRC_Connection_Reconfiguration request to the UE on LTE link 0 + if (second_link_activated==0) + { + send_MIH_Link_Action_Power_Up_request(_link_id_list[0]); + second_link_activated = 1; + } +// continue; + // send_MIH_Link_Action_Power_Up_request(link); +// if (link_id2string(link).c_str() == "LTE") +// } + } + } + +} + + +//Forward the MIH_Link_Parameters_Report to the UE of the CPE +//----------------------------------------------------------------------------- +void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + m >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + odtone::mih::message msg; + msg.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + msg.destination(mid_ue); + + //mn_ho_candidate_query is used to constuct/send the message containing parameters reports + msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- before MIH_User_PARAMETERS.indication --->]["+msg.destination().to_string()+"]\n"); + + + _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, "MIH_User_PARAMETERS.indication- SENT (towards UE)\n"); + +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl1; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lcp1.type = lp1; + lcp2.type = lp2; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th2.threshold_val = -19; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + + + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue");//mihf2_ue + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ " --->][ ADDR LINK " + + m.destination().to_string() +"]\n"); + std::cout<<"LINK TUPLE ID "<<lti<<std::endl; + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti); +// & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", (lcsl[iter].thold.threshold_val/256)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("enb_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf4_enb"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..b9bbf279de1650cfa71dbeef31960214086d0d98 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.conf @@ -0,0 +1,40 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_enb + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..b0fb81c99d2ad9a85bc5db264923394ecab6fab0 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.cpp @@ -0,0 +1,1681 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <string> +#include <map> +#include <stdio.h> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +//The thresholds for the Energy Detection Sensing algorithm respectively for 10 and 100 samples +#define ED_THRESHOLD_10 23695432 +#define ED_THRESHOLD_100 230445932 +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER (of the eNB) for the demo of the Scenario 2 of SPECTRA project is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Initiallization of the MIH-USER and the MIHF +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | + +// ------------------------------------------------------------------------------------------------------------------------ +// Locally send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to local MIHF) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | + +// ------------------------------------------------------------------------------------------------------------------------ +// Remotely send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to remote MIHF of the UE) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Detect the connection of the UE to the eNB + Start the measurement report process +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Activate the TVWS link (After running the cognitive algorithms) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |-------------- MIH_Link_Actions.request ------------>|--------------- MIH_Link_Actions.request ----------->| +// | | | +// |<------------- MIH_Link_Actions.confirm -------------|<-------------- MIH_Link_Actions.confirm ------------| +// | | | + +//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + +static const char* const kConf_MIH_Commands = "user.commands"; + +///////////////////////////////////////////////////////////////////////////////////////// + +std::string exec(char* cmd) { + FILE* pipe = popen(cmd, "r"); + if (!pipe) return "ERROR"; + char buffer[128]; + std::string result = ""; + while(!feof(pipe)) { + if(fgets(buffer, 128, pipe) != NULL) + result += buffer; + } + pclose(pipe); + return result; +} + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} + + +/*//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} +*/ + +//Updated from UE code +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + + //Measurements report methods + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + int receive_Sensing_Report(); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + void forward_Parameters_Report_indication(odtone::mih::message& m); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + odtone::uint second_link_activated; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + second_link_activated = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + + //send a capability discover request to the remote UE + mih_user::send_MIH_Capability_Discover_request_remote(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + boost::system::error_code ec; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} +// mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, /*boost::cref(cfg),*/ _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address +// for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) +// { +// rcv_link_id.addr = i->addr; +// rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); +// } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + + std::cout<<"NTALL "<<ntal.get()[0].addr<<std::endl; + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; +// std::ostringstream stream; +// std::stringstream st; +// odtone::mih::net_type_addr net_type_addr; +// net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); +// stream << net_type_addr; +// std::string s; +// std::stringstream st (stream.str()); +// while (getline (st, s, '\n')) +// { +// std::stringstream ss(s); +// getline (ss, name, ':'); +// getline (ss, value, '\n'); +// // ss>>name>>value; +// std::cout<<" name "<<name<<" value "<<value<<std::endl; +// } +/* + st<<i->addr; + getline (st, s, ' '); + std::cout<<"LINK_TUPLE_ID s "<<s<<std::endl; + + odtone::mih::l2_3gpp_addr add; + add.value = s;*/ + li.addr = /*add*/ boost::get<odtone::mih::l2_3gpp_addr>(i->addr); + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + if (std::find(_link_id_list.begin(), _link_id_list.end(), li)==_link_id_list.end()) + _link_id_list.push_back(li); // save the link identifier of the network interface + + std::cout<<"LINK_TUPLE_ID - Link identifier:after "<<link_addr2string(&li.addr).c_str() <<" "<<_link_id_list[_link_id_list.size()-1]<<std::endl; + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<li<<std::endl; +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + link_act_req.id = link; +// } + + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + std::cout<<"_link received from parameters report "<<link<<std::endl; + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + } + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + +// link_act_req.action.param.param = res; + +// link_act_req.ex_time = 0; + link_act_req.ex_time = 5000; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + +// odtone::mih::id mid_ue; +// mid_ue.assign("mihf2_ue"); + m.destination(/*mid_ue*/_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st); +// & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); +// if (larl) { +// log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); +// for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) +// { +// log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), +// ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); +// } +// } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +// #ifdef SCENARIO_1 +// if (larl) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// if (rsp->result.get() == odtone::mih::link_ac_success) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// _current_link_action_request += 1; +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_1 +// +// #ifdef SCENARIO_2 +// // 2nd scenario: Activate all resources, then deactivate all resources +// if (larl.get().size() > 0) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (++_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// _current_link_action_request = 0; +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +// void mih_user::cognitive_decision() +// { +// +// //Sensing measurements report +// std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); +// } + +//----------------------------------------------------------------------------- +int mih_user::receive_Sensing_Report() +//----------------------------------------------------------------------------- +{ + std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); + std::string tmp; + int value; + std::stringstream ss(result); + ss >> tmp >> value; +// std::cout<<"Result "<<result<<std::endl; + return value; +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link, link1; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<link<<std::endl; + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value ); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + + + //eNB1: Forward the message to UE 2 + // forward_Parameters_Report_indication(msg); + + //eNB2 : Action Power Up the TVWS Link after running the cognitive algorithm + + //First Phase: Collect the data as input fot the cognitive algorithm + //Sensing data + int score; + score = receive_Sensing_Report(); + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { + log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// if(*i==link) + odtone::mih::l2_3gpp_addr add; + odtone::mih::link_type type; + type = odtone::mih::link_type_lte; +// add.value = "l2_3gpp_addr"; + link1.addr = add; + link1.type = type; +// send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_power_up); + + //Send ONLY ONCE RRC_Connection_Reconfiguration request to the UE on LTE link 0 + if (second_link_activated==0) + { + send_MIH_Link_Action_Power_Up_request(_link_id_list[0]); + second_link_activated = 1; + } +// continue; +// send_MIH_Link_Action_Power_Up_request(link); +// if (link_id2string(link).c_str() == "LTE") +// } + } + } + +} + + +//Forward the MIH_Link_Parameters_Report to the UE of the CPE +//----------------------------------------------------------------------------- +void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + m >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + odtone::mih::message msg; + msg.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + msg.destination(mid_ue); + + //mn_ho_candidate_query is used to constuct/send the message containing parameters reports + msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- before MIH_User_PARAMETERS.indication --->]["+msg.destination().to_string()+"]\n"); + + + _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, "MIH_User_PARAMETERS.indication- SENT (towards UE)\n"); + +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl1; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lcp1.type = lp1; + lcp2.type = lp2; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th2.threshold_val = -19; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + + + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ " --->][ ADDR LINK " + + m.destination().to_string() +"]\n"); + std::cout<<"LINK TUPLE ID "<<lti<<std::endl; + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti); +// & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", (lcsl[iter].thold.threshold_val/256)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("enb_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1_enb"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.cpp.old b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.cpp.old new file mode 100644 index 0000000000000000000000000000000000000000..b32628931c8b17eec2a0d6159814320876f247e9 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user.cpp.old @@ -0,0 +1,1346 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER is the following (with eRALlteDummy and NASRGDummy executables) +// +--------+ +-----+ +// |MIH_USER| |MIH-F| +// +---+----+ +--+--+ +// | | _current_link_action_request = 0 +// |---------- User_Register.indication ---------------->| (supported_commands) Handler next msg=user_reg_handler +// | | +// |---------- Capability_Discover.request ------------->| Handler next msg=receive_MIH_Capability_Discover_confirm +// |<--------- Capability_Discover.confirm --------------| (success) +// | | +// |---------- Event_Subscribe.request ----------------->| Handler next msg=receive_MIH_Event_Subscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 1: Sequentially activate and deactivate each resource +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// |<--------- Link_Actions.confirm ---------------------| (success) +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 2: Activate all resources, then deactivate all resources +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | _current_link_action_request = 0 +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Event_Unsubscribe.request --------------->| Handler next msg=receive_MIH_Event_Unsubscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// | | +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s; + if(evtP.get(odtone::mih::mih_evt_link_detected)) s = std::string("DETECTED "); + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s; + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s = std::string("Link_Get_Parameters "); + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES) +//----------------------------------------------------------------------------- +{ + + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + } + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + + break; + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + odtone::mih::threshold th; + std::vector<odtone::mih::threshold> thl; + + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp; + odtone::mih::link_param_lte lp; + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + lp = odtone::mih::link_param_lte_bandwidth; + + lcp.type = lp; + + if ( link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp.timer_interval = 0; + lcp.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th.threshold_val = 0; + th.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl.push_back(th); + lcp.threshold_list = thl; + lcpl.push_back(lcp); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+_mihfid.to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); + + if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", th.threshold_val); + + if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + // T odtone::uint iter; + // T odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + // Todtone::mih::link_cfg_status_list lcsl; + // Todtone::mih::link_cfg_status lcp; + //odtone::mih::link_param_gen lp; + + // T odtone::mih::link_tuple_id lti; + + //msg >> odtone::mih::confirm() + // & odtone::mih::tlv_status(st) + // & odtone::mih::tlv_link_identifier(lti) + // & odtone::mih::tlv_link_cfg_status_list(lcsl); + + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("mih_usr.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1234), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user2.cpp.old b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user2.cpp.old new file mode 100644 index 0000000000000000000000000000000000000000..1f64a5d556ab64033ad230ff617708107a1fdce2 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user2.cpp.old @@ -0,0 +1,1565 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER (of the eNB) for the demo of the Scenario 2 of SPECTRA project is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Initiallization of the MIH-USER and the MIHF +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | + +// ------------------------------------------------------------------------------------------------------------------------ +// Locally send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to local MIHF) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | + +// ------------------------------------------------------------------------------------------------------------------------ +// Remotely send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to remote MIHF of the UE) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Detect the connection of the UE to the eNB + Start the measurement report process +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Activate the TVWS link (After running the cognitive algorithms) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |-------------- MIH_Link_Actions.request ------------>|--------------- MIH_Link_Actions.request ----------->| +// | | | +// |<------------- MIH_Link_Actions.confirm -------------|<-------------- MIH_Link_Actions.confirm ------------| +// | | | + +//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} + + +/*//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} +*/ + +//Updated from UE code +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + void do_nothing(); + void forward_Parameters_Report_indication(odtone::mih::message& m); + void receive_MIH_MN_HO_Candidate_Query_request(odtone::mih::message& msg); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + + //send a capability discover request to the remote UE + mih_user::send_MIH_Capability_Discover_request_remote(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + //mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + //_num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::mn_ho_candidate_query: + log_(0, "MIH-User has received a request \"mn_ho_candidate_query\""); + mih_user::receive_MIH_MN_HO_Candidate_Query_request(msg); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + boost::system::error_code ec; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} +// mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, /*boost::cref(cfg),*/ _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +void mih_user::receive_MIH_MN_HO_Candidate_Query_request(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::request() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_MN_HO_Candidate_Query.request - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value /*- 65536*/); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value /*- 65536*/); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + + + //Test 1: Forward the message to UE + // forward_Parameters_Report_indication(msg); + +} + +//----------------------------------------------------------------------------- +void mih_user::do_nothing() +//----------------------------------------------------------------------------- +{ + +} + +//----------------------------------------------------------------------------- +void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + m >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + odtone::mih::message msg; + msg.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + msg.destination(mid_ue); + + + msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- before MIH_User_PARAMETERS.indication --->]["+msg.destination().to_string()+"]\n"); + + + _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, "MIH_User_PARAMETERS.indication- SENT (towards UE)\n"); + +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl1; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lcp1.type = lp1; + lcp2.type = lp2; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th2.threshold_val = -19; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + + + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); +// m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+m.destination().to_string() +"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti); +// & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", (lcsl[iter].thold.threshold_val/256)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("enb2_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf4_enb"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user_tcs.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user_tcs.cpp new file mode 100644 index 0000000000000000000000000000000000000000..72d46e791bee1fde516e9f592ec1f77bdec6d605 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/enb_lte_user_tcs.cpp @@ -0,0 +1,1412 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +// Fatma HRIZI EURECOM <hrizi@eurecom>fr> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +//This file is the implementation of the MIH user in the eNB + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER is the following (with eRALlteDummy and NASRGDummy executables) +// +--------+ +-----+ +// |MIH_USER| |MIH-F| +// +---+----+ +--+--+ +// | | _current_link_action_request = 0 +// |---------- User_Register.indication ---------------->| (supported_commands) Handler next msg=user_reg_handler +// | | +// |---------- Capability_Discover.request ------------->| Handler next msg=receive_MIH_Capability_Discover_confirm +// |<--------- Capability_Discover.confirm --------------| (success) +// | | +// |---------- Event_Subscribe.request ----------------->| Handler next msg=receive_MIH_Event_Subscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 1: Sequentially activate and deactivate each resource +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// |<--------- Link_Actions.confirm ---------------------| (success) +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 2: Activate all resources, then deactivate all resources +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | _current_link_action_request = 0 +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Event_Unsubscribe.request --------------->| Handler next msg=receive_MIH_Event_Unsubscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// | | +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s; + if(evtP.get(odtone::mih::mih_evt_link_detected)) s = std::string("DETECTED "); + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s; + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s = std::string("Link_Get_Parameters "); + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(const odtone::mih::config& cfg); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES) +//----------------------------------------------------------------------------- +{ + + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + log_(0, ""); + + // + // Local Capability Discover Request + // + odtone::mih::message msg; + _mihfid.assign("mihf2"); + msg << odtone::mih::request(odtone::mih::request::capability_discover, _mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Capability_Discover.request --->]["+msg.destination().to_string()+"]\n"); + _mihf.async_send(msg, boost::bind(&mih_user::receive_MIH_Capability_Discover_confirm, this, _1)); + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)"); + log_(0, ""); + + //****eNB***** + // Remote Capability Discover Request + // + mih_user::send_MIH_Capability_Discover_request(cfg); + + //Trigger Link_Configure_Thresholds Request + // odtone::mih::message m; + //m.destination(msg.source()); + //mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + //****eNB***** +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + } + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + + break; + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} +/* +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + log_(0, "MIH_Link_Detected.indication - End\n"); +} +*/ +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + +boost::system::error_code ec; + for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) + { + log_(0, " - LINK_ID - Link identifier: ", link_id2string(i->id).c_str()); + log_(0, " - Network ID: ", i->network_id); + log_(0, " - SINR: ", i->sinr); + log_(0, " - Data_rate: ", i->data_rate); + } + + send_MIH_Link_Configure_Thresholds_request(msg, ec); + + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (const odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { + + log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available + } + else{ + log_(0, " -Regular Report for BW Threshold "); + //update MEAS + } + } + + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + //boost::optional<odtone::mih::net_type_addr_list> ntal; + //boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::request(odtone::mih::request::capability_discover) + //& odtone::mih::tlv_net_type_addr_list(ntal) + //& odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + _mihfid.assign("mihf2"); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + odtone::mih::threshold th; + std::vector<odtone::mih::threshold> thl; + + odtone::mih::link_tuple_id lti; +// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + + + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp; + odtone::mih::link_param_lte lp; + + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + + odtone::mih::link_det_info_list ldil; + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) + { + + + lti.type = i->id.type; + lti.addr = i->id.addr; + lp = odtone::mih::link_param_lte_bandwidth; + lcp.type = lp; + if ( link_measures_request ==0){ + lcp.timer_interval = 10; + lcp.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } + else{ + lcp.timer_interval = 0; + lcp.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + th.threshold_val = 5; + th.threshold_x_dir = odtone::mih::threshold::above_threshold; + thl.push_back(th); + lcp.threshold_list = thl; + lcpl.push_back(lcp); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + link_id2string(lti).c_str()+ + " --->]["+_mihfid.to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + } + +/* + if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); + + if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", th.threshold_val); + + if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +*/ +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + // T odtone::uint iter; + // T odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + // Todtone::mih::link_cfg_status_list lcsl; + // Todtone::mih::link_cfg_status lcp; + //odtone::mih::link_param_gen lp; + + // T odtone::mih::link_tuple_id lti; + + //msg >> odtone::mih::confirm() + // & odtone::mih::tlv_status(st) + // & odtone::mih::tlv_link_identifier(lti) + // & odtone::mih::tlv_link_cfg_status_list(lcsl); + + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1234), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/ue_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/ue_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..27d56760bea01f85150e9f7e5d21214e5bbf1ae3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/ue_lte_user.conf @@ -0,0 +1,42 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_ue + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/ue_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/ue_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..582573a66681c85134e760f218a56a724772a622 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/mih_user/lte_test_user/ue_lte_user.cpp @@ -0,0 +1,1658 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_req +//#define SCENARIO_1 // Sequentially activate and deactivate each resou +//#define SCENARIO_2 // Activate all resources, then deactivate all resour +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER demo is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// etc etc etc + +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +// int& operator [](const link_param_lte_enum value) +// { +// static std::map<link_param_lte_enum, std::string> var; +// if (var.size() == 0) +// { +// var[link_param_lte_rsrp]="link_param_lte_rsrp"; +// var[link_param_lte_rsrq]="link_param_lte_rsrq"; +// var[link_param_lte_cqi]="link_param_lte_cqi"; +// var[link_param_lte_bandwidth]="link_param_lte_bandwidth"; +// var[link_param_lte_pkt_delay]="link_param_lte_pkt_delay"; +// } +// return var[value]; +// } + +const char * MeasurementTypes[]= +{ + "LTE_RSRP", /**< RSRP. */ + "LTE_RSRQ", /**< RSRQ. */ + "link_param_lte_cqi", /**< Multicast packet loss rate.*/ + "link_param_lte_bandwidth", /**< System Load. */ + "link_param_lte_pkt_delay", /**< Number of registered users. */ + "link_param_lte_pkt_loss", /**< Number of active users. */ + "link_param_lte_l2_buffer", /**< Congestion windows of users. */ + "link_param_lte_MN_cap", /**< Congestion windows of users. */ + "link_param_lte_embms", /**< Congestion windows of users. */ + "link_param_lte_jumbo_feasibility", /**< Congestion windows of users. */ + "link_param_lte_jumbo_setup", /**< Congestion windows of users. */ + "link_param_lte_active_embms", /**< Transmission rate of users. */ + "link_param_lte_link_congestion", /**< Link congestion. */ +}; + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)){ + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type"; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + + + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + void send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + //send a capability discover request to the remote eNB + mih_user::send_MIH_Capability_Discover_request_remote(); + +} + + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1;; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + // send Link_Action / Power Up + lid.type = odtone::mih::link_type_lte; + lid.addr = ldi.id.addr; + //send_MIH_Link_Action_Power_Up_plus_scan_request(lid); + log_(0, "MIH_Link_Detected.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + + link_act_req.id = link; + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_enb; + mid_enb.assign("mihf1_enb"); + m.destination(mid_enb); +// m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); + mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(link); + +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl){ + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + +// odtone::mih::link_param_report lpr = lprl.back(); +// +// log_(0, "Meausrement Type: ", lpr.param.type); +// if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&lpr.param.value)){ +// log_(0, "Meausrement Value: ", *value); +// +// }; +// const odtone::mih::threshold *th; + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { +// odtone::mih::link_param_lte * index = boost::get<odtone::mih::link_param_lte>(&i->param.type); +// int val = (short)*index; +// log_(0, "Meausrement Type: index ", *index); + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); +// log_(0, "Meausrement Type: ", MeasurementTypes[index]); +// log_(0, "Meausrement Type: ", &i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value /*- 65536*/); + } +// if (th = boost::get<odtone::mih::threshold> (&i->thold)) +// { +// log_(0, "Threshold crossed ", boost::get<odtone::mih::link_param_val> (i->param.value)); +// } +// else +// { +// log_(0,"Regular Report"); +// } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + std::vector<odtone::mih::threshold> thl1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::threshold th3; + std::vector<odtone::mih::threshold> thl3; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_cfg_param lcp3; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + odtone::mih::link_param_lte lp3; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lp3 = odtone::mih::link_param_lte_cqi; + lcp1.type = lp1; + lcp2.type = lp2; + lcp3.type = lp3; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + lcp3.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + lcp3.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp3.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + lcp3.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_val = -19; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + th3.threshold_val = 0; + th3.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + thl3.push_back(th3); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + lcp3.threshold_list = thl3; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + lcpl.push_back(lcp3); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + if(lp2 == odtone::mih::link_param_lte_rsrq) {log_(0, "\t LTE link parameter LTE RSRQ");} + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + +// //----------------------------------------------------------------------------- +// void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +// //----------------------------------------------------------------------------- +// { +// odtone::mih::message m; +// odtone::mih::threshold th; +// std::vector<odtone::mih::threshold> thl; +// odtone::mih::link_tuple_id lti; +// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; +// //List of the link threshold parameters +// odtone::mih::link_cfg_param_list lcpl; +// odtone::mih::link_cfg_param lcp; +// odtone::mih::link_param_lte lp; +// //odtone::mih::link_param_gen lp; +// +// odtone::mih::link_param_type typr; +// +// log_(0,""); +// log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); +// +// //link_tuple_id +// lti.type = rcv_link_id.type; +// lti.addr = rcv_link_id.addr; +// +// //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); +// +// +// //link_param_gen_data_rate = 0, /**< Data rate. */ +// //link_param_gen_signal_strength = 1, /**< Signal strength. */ +// //link_param_gen_sinr = 2, /**< SINR. */ +// //link_param_gen_throughput = 3, /**< Throughput. */ +// //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ +// //lp = odtone::mih::link_param_lte_bandwidth; +// lp = odtone::mih::link_param_lte_rsrp; +// lcp.type = lp; +// +// link_measures_request = 0; +// if ( link_measures_request ==0){ +// // Set Timer Interval (in ms) +// lcp.timer_interval = 3000; +// //th_action_normal = 0, /**< Set normal threshold. */ +// //th_action_one_shot = 1, /**< Set one-shot threshold. */ +// //th_action_cancel = 2 /**< Cancel threshold. */ +// lcp.action = odtone::mih::th_action_normal; +// link_measures_request = 1; +// } else if ( link_measures_request==1){ +// // Set Timer Interval (in ms) +// lcp.timer_interval = 0; +// lcp.action = odtone::mih::th_action_cancel; +// link_measures_request = 0; +// } +// +// //above_threshold = 0, /**< Above threshold. */ +// //below_threshold = 1, /**< Below threshold. */ +// th.threshold_val = -105; +// th.threshold_x_dir = odtone::mih::threshold::above_threshold; +// +// thl.push_back(th); +// lcp.threshold_list = thl; +// lcpl.push_back(lcp); +// +// m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) +// & odtone::mih::tlv_link_identifier(lti) +// & odtone::mih::tlv_link_cfg_param_list(lcpl); +// +// m.destination(msg.source()); +// +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ +// // link_tupple_id2string(lti).c_str() + +// link_id2string(lti).c_str()+ +// " --->]["+_mihfid.to_string()+"]\n"); +// +// _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); +// +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); +// +// log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); +// +// //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} +// //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} +// //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} +// //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} +// //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} +// if(lp == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} +// +// log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); +// +// if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} +// if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} +// if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} +// +// log_(0, "\t Threshold value: ", th.threshold_val); +// +// if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} +// if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} +// +// log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +// } + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", ((short) lcsl[iter].thold.threshold_val)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +// //----------------------------------------------------------------------------- +// void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +// //----------------------------------------------------------------------------- +// { +// log_(0, ""); +// log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); +// +// // T odtone::uint iter; +// // T odtone::mih::status st; +// +// //boost::optional<odtone::mih::link_cfg_status_list> lcsl; +// // Todtone::mih::link_cfg_status_list lcsl; +// // Todtone::mih::link_cfg_status lcp; +// //odtone::mih::link_param_gen lp; +// +// // T odtone::mih::link_tuple_id lti; +// +// //msg >> odtone::mih::confirm() +// // & odtone::mih::tlv_status(st) +// // & odtone::mih::tlv_link_identifier(lti) +// // & odtone::mih::tlv_link_cfg_status_list(lcsl); +// +// +// log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); +// log_(0,""); +// } + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("ue_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf2_ue"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/oai_conf/enb.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/oai_conf/enb.conf new file mode 100644 index 0000000000000000000000000000000000000000..8acd6d84afc6530bc1c366e11e4424a3913ed2bb --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/oai_conf/enb.conf @@ -0,0 +1,217 @@ +Active_eNBs = ( "eNB_Eurecom_LTEBox"); +# Asn1_verbosity, choice in: none, info, annoying +Asn1_verbosity = "none"; + +eNBs = +( + { + # real_time choice in {hard, rt-preempt, no} + real_time = "no"; + ////////// Identification parameters: + eNB_ID = 0xe00; + + cell_type = "CELL_MACRO_ENB"; + + eNB_name = "eNB_Eurecom_LTEBox"; + + // Tracking area code, 0x0000 and 0xfffe are reserved values + tracking_area_code = "1"; + + mobile_country_code = "208"; + + mobile_network_code = "92"; + + ////////// Physical parameters: + + component_carriers = ( + { + frame_type = "TDD"; + tdd_config = 3; + tdd_config_s = 0; + prefix_type = "NORMAL"; + eutra_band = 33; + downlink_frequency = 1910000000L; + uplink_frequency_offset = 0; + + Nid_cell = 0; + N_RB_DL = 25; + Nid_cell_mbsfn = 0; + nb_antennas_tx = 1; + nb_antennas_rx = 1; + prach_root = 22; + prach_config_index = 3; + prach_high_speed = "DISABLE"; + prach_zero_correlation = 0; + prach_freq_offset = 0; + pucch_delta_shift = 1; + pucch_nRB_CQI = 1; + pucch_nCS_AN = 0; + pucch_n1_AN = 32; + pdsch_referenceSignalPower = -24; + pdsch_p_b = 0; + pusch_n_SB = 1; + pusch_enable64QAM = "DISABLE"; + pusch_hoppingMode = "interSubFrame"; + pusch_hoppingOffset = 4; + pusch_groupHoppingEnabled = "DISABLE"; + pusch_groupAssignment = 0; + pusch_sequenceHoppingEnabled = "DISABLE"; + pusch_nDMRS1 = 1; + phich_duration = "NORMAL"; + phich_resource = "ONESIXTH"; + srs_enable = "DISABLE"; + # srs_BandwidthConfig =; + # srs_SubframeConfig =; + # srs_ackNackST =; + # srs_MaxUpPts =; + + pusch_p0_Nominal = -95; + pusch_alpha = "AL08"; + pucch_p0_Nominal = -117; + msg3_delta_Preamble = 6; + pucch_deltaF_Format1 = "deltaF2"; + pucch_deltaF_Format1b = "deltaF3"; + pucch_deltaF_Format2 = "deltaF0"; + pucch_deltaF_Format2a = "deltaF0"; + pucch_deltaF_Format2b = "deltaF0"; + + rach_numberOfRA_Preambles = 52; + rach_preamblesGroupAConfig = "DISABLE"; + +# rach_sizeOfRA_PreamblesGroupA = ; +# rach_messageSizeGroupA = ; +# rach_messagePowerOffsetGroupB = ; + + rach_powerRampingStep = 2; + rach_preambleInitialReceivedTargetPower = -104; + rach_preambleTransMax = 6; + rach_raResponseWindowSize = 10; + rach_macContentionResolutionTimer = 48; + rach_maxHARQ_Msg3Tx = 4; + + pcch_default_PagingCycle = 128; + pcch_nB = "oneT"; + bcch_modificationPeriodCoeff = 2; + ue_TimersAndConstants_t300 = 1000; + ue_TimersAndConstants_t301 = 1000; + ue_TimersAndConstants_t310 = 1000; + ue_TimersAndConstants_t311 = 10000; + ue_TimersAndConstants_n310 = 20; + ue_TimersAndConstants_n311 = 1; + + }, + { + frame_type = "TDD"; + tdd_config = 3; + tdd_config_s = 0; + prefix_type = "NORMAL"; + eutra_band = 33; + downlink_frequency = 1910000000L; + uplink_frequency_offset = 0; + + Nid_cell = 0; + N_RB_DL = 25; + Nid_cell_mbsfn = 0; + nb_antennas_tx = 1; + nb_antennas_rx = 1; + prach_root = 22; + prach_config_index = 3; + prach_high_speed = "DISABLE"; + prach_zero_correlation = 0; + prach_freq_offset = 0; + pucch_delta_shift = 1; + pucch_nRB_CQI = 1; + pucch_nCS_AN = 0; + pucch_n1_AN = 32; + pdsch_referenceSignalPower = -24; + pdsch_p_b = 0; + pusch_n_SB = 1; + pusch_enable64QAM = "DISABLE"; + pusch_hoppingMode = "interSubFrame"; + pusch_hoppingOffset = 4; + pusch_groupHoppingEnabled = "DISABLE"; + pusch_groupAssignment = 0; + pusch_sequenceHoppingEnabled = "DISABLE"; + pusch_nDMRS1 = 1; + phich_duration = "NORMAL"; + phich_resource = "ONESIXTH"; + srs_enable = "DISABLE"; + /* srs_BandwidthConfig =; + srs_SubframeConfig =; + srs_ackNackST =; + srs_MaxUpPts =;*/ + + pusch_p0_Nominal = -95; + pusch_alpha = "AL08"; + pucch_p0_Nominal = -117; + msg3_delta_Preamble = 6; + pucch_deltaF_Format1 = "deltaF2"; + pucch_deltaF_Format1b = "deltaF3"; + pucch_deltaF_Format2 = "deltaF0"; + pucch_deltaF_Format2a = "deltaF0"; + pucch_deltaF_Format2b = "deltaF0"; + + rach_numberOfRA_Preambles = 52; + rach_preamblesGroupAConfig = "DISABLE"; +/* + rach_sizeOfRA_PreamblesGroupA = ; + rach_messageSizeGroupA = ; + rach_messagePowerOffsetGroupB = ; +*/ + rach_powerRampingStep = 2; + rach_preambleInitialReceivedTargetPower = -104; + rach_preambleTransMax = 6; + rach_raResponseWindowSize = 10; + rach_macContentionResolutionTimer = 48; + rach_maxHARQ_Msg3Tx = 4; + + pcch_default_PagingCycle = 128; + pcch_nB = "oneT"; + bcch_modificationPeriodCoeff = 2; + ue_TimersAndConstants_t300 = 1000; + ue_TimersAndConstants_t301 = 1000; + ue_TimersAndConstants_t310 = 1000; + ue_TimersAndConstants_t311 = 10000; + ue_TimersAndConstants_n310 = 20; + ue_TimersAndConstants_n311 = 1; + + } + ); + + ////////// MME parameters: + mme_ip_address = ( { ipv4 = "192.168.13.11"; + ipv6 = "192:168:30::17"; + active = "yes"; + preference = "ipv4"; + } + ); + + NETWORK_INTERFACES : + { + ENB_INTERFACE_NAME_FOR_S1_MME = "eth1"; + ENB_IPV4_ADDRESS_FOR_S1_MME = "192.168.13.10/24"; + + ENB_INTERFACE_NAME_FOR_S1U = "eth1"; + ENB_IPV4_ADDRESS_FOR_S1U = "192.168.13.10/24"; + }; + + log_config : + { + global_log_level ="debug"; + global_log_verbosity ="medium"; + hw_log_level ="debug"; + hw_log_verbosity ="medium"; + phy_log_level ="info"; + phy_log_verbosity ="medium"; + mac_log_level ="debug"; + mac_log_verbosity ="high"; + rlc_log_level ="info"; + rlc_log_verbosity ="medium"; + pdcp_log_level ="info"; + pdcp_log_verbosity ="medium"; + rrc_log_level ="info"; + rrc_log_verbosity ="medium"; + }; + + } +); diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/oai_conf/start_enb.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/oai_conf/start_enb.bash new file mode 100644 index 0000000000000000000000000000000000000000..c877b9b8435b6014f337e97c40e3dde550b20738 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/enb2/oai_conf/start_enb.bash @@ -0,0 +1,211 @@ +#!/bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file start_enb.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# +#------------------------------------------------ +# ENB CONFIG FILE +#------------------------------------------------ +#declare -x ENB_CONFIG_FILE="CONF/enb.sfr.yang.conf" +declare -x ENB_CONFIG_FILE="enb.conf" + +#------------------------------------------------ +# OAI NETWORKING +#------------------------------------------------ +declare -x EMULATION_DEV_INTERFACE="eth0" +declare -x EMULATION_DEV_ADDRESS="192.168.15.6" +declare -x IP_DRIVER_NAME="oai_nw_drv" +declare -x LTEIF="oai0" +declare -x ENB_IPv4="10.0.2.4" +declare -x ENB_IPv6="2001:2::4" +declare -x ENB_IPv6_CIDR=$ENB_IPv6"/64" +declare -x ENB_IPv4_CIDR=$ENB_IPv4"/24" +declare -a NAS_IMEI=( 3 9 1 8 3 6 6 2 0 0 0 0 0 0 ) +declare -x IP_DEFAULT_MARK="1" # originally 3 +#------------------------------------------------ +# OAI MIH +#------------------------------------------------ +declare -x ENB_RAL_IP_ADDRESS="127.0.0.1" +declare -x ENB_MIHF_IP_ADDRESS=127.0.0.1 +MIH_LOG_FILE="mih-f_enb.log" + +#------------------------------------------------ +LOG_FILE="/tmp/oai_sim_enb.log" + +# EXE options +EXE_MODE="DEBUG" # PROD or DEBUG + +########################################################### +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +source $THIS_SCRIPT_PATH/env_802dot21.bash +########################################################### +bash_exec "service network-manager stop" +bash_exec "ifconfig $EMULATION_DEV_INTERFACE up $EMULATION_DEV_ADDRESS netmask 255.255.255.0" +########################################################### +IPTABLES=/sbin/iptables +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +declare -x OPENAIR_DIR="" +declare -x OPENAIR1_DIR="" +declare -x OPENAIR2_DIR="" +declare -x OPENAIR3_DIR="" +declare -x OPENAIR_TARGETS="" +########################################################### + +set_openair +cecho "OPENAIR_DIR = $OPENAIR_DIR" $green +cecho "OPENAIR1_DIR = $OPENAIR1_DIR" $green +cecho "OPENAIR2_DIR = $OPENAIR2_DIR" $green +cecho "OPENAIR3_DIR = $OPENAIR3_DIR" $green +cecho "OPENAIR_TARGETS = $OPENAIR_TARGETS" $green + +bash_exec "/sbin/iptables -t mangle -F" +bash_exec "/sbin/iptables -t nat -F" +bash_exec "/sbin/iptables -t raw -F" +bash_exec "/sbin/iptables -t filter -F" +bash_exec "/sbin/ip6tables -t mangle -F" +bash_exec "/sbin/ip6tables -t filter -F" +bash_exec "/sbin/ip6tables -t raw -F" + +################################################## +# LAUNCH eNB executable +################################################## + +echo "Bringup eNB interface" +pkill oaisim > /dev/null 2>&1 +pkill oaisim > /dev/null 2>&1 +pkill $MIH_F > /dev/null 2>&1 +pkill $ENB_MIH_USER > /dev/null 2>&1 +rmmod -f $IP_DRIVER_NAME > /dev/null 2>&1 + +bash_exec "insmod $OPENAIR2_DIR/NAS/DRIVER/LITE/$IP_DRIVER_NAME.ko oai_nw_drv_IMEI=${NAS_IMEI[0]},${NAS_IMEI[1]},${NAS_IMEI[2]},${NAS_IMEI[3]},${NAS_IMEI[4]},${NAS_IMEI[5]},${NAS_IMEI[6]},${NAS_IMEI[7]},${NAS_IMEI[8]},${NAS_IMEI[9]},${NAS_IMEI[10]},${NAS_IMEI[11]},${NAS_IMEI[12]},${NAS_IMEI[13]}" +bash_exec "ip route flush cache" +bash_exec "ip link set $LTEIF up" +sleep 1 +bash_exec "ip addr add dev $LTEIF $ENB_IPv4_CIDR" +bash_exec "ip addr add dev $LTEIF $ENB_IPv6_CIDR" +sleep 1 +bash_exec "sysctl -w net.ipv4.conf.all.log_martians=1" +assert " `sysctl -n net.ipv4.conf.all.log_martians` -eq 1" $LINENO +bash_exec "sysctl -w net.ipv4.conf.all.rp_filter=0" +assert " `sysctl -n net.ipv4.conf.all.rp_filter` -eq 0" $LINENO +bash_exec "ip route flush cache" +bash_exec "sysctl -w net.ipv4.ip_forward=1" +assert " `sysctl -n net.ipv4.ip_forward` -eq 1" $LINENO + +# Check table 200 lte in /etc/iproute2/rt_tables +fgrep lte /etc/iproute2/rt_tables > /dev/null +if [ $? -ne 0 ]; then + echo '200 lte ' >> /etc/iproute2/rt_tables +fi +ip rule add fwmark $IP_DEFAULT_MARK table lte +ip route add default dev $LTEIF table lte +ip route add 239.0.0.160/28 dev $EMULATION_DEV_INTERFACE + +/sbin/ebtables -t nat -A POSTROUTING -p arp -j mark --mark-set 3 + +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +#All other traffic is sent on the RAB you want (mark = RAB ID) +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK + +rotate_log_file $MIH_LOG_FILE + + +# start MIH-F +xterm -hold -title "[eNB][eNB2] MIHF" -e $ODTONE_MIH_EXE_DIR/$MIH_F --log 4 --conf.file $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE > $MIH_LOG_FILE 2>&1 & +wait_process_started $MIH_F + +NOW=$(date +"%Y-%m-%d.%Hh_%Mm_%Ss") +rm -f $LOG_FILE + +ENB_RAL_LINK_ID=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +ENB_RAL_LINK_ID=`trim2 $ENB_RAL_LINK_ID` +ENB_RAL_LINK_ID=`echo $ENB_RAL_LINK_ID | cut -d" " -f1` + +ENB_RAL_LISTENING_PORT=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +ENB_RAL_LISTENING_PORT=`trim2 $ENB_RAL_LISTENING_PORT` +ENB_RAL_LISTENING_PORT=`echo $ENB_RAL_LISTENING_PORT | cut -d" " -f2` + +ENB_MIHF_REMOTE_PORT=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep local_port | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` + +ENB_MIHF_ID=`cat $ODTONE_MIH_EXE_DIR/$ENB_MIH_F_CONF_FILE | grep id | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` +#remove 2 last digits (vitualization, index on 2 digits) +ENB_RAL_LINK_ID_STRIPPED=${ENB_RAL_LINK_ID%%??} + +if [ $EXE_MODE = "DEBUG" ] ; then +#xterm -hold -e gdb --args +$OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l9 -u0 -b1 -M0 -p2 -g3 -D $EMULATION_DEV_INTERFACE \ + --enb-ral-listening-port $ENB_RAL_LISTENING_PORT \ + --enb-ral-link-id $ENB_RAL_LINK_ID_STRIPPED \ + --enb-ral-ip-address $ENB_RAL_IP_ADDRESS \ + --enb-mihf-remote-port $ENB_MIHF_REMOTE_PORT \ + --enb-mihf-ip-address $ENB_MIHF_IP_ADDRESS \ + --enb-mihf-id $ENB_MIHF_ID \ + -O $ENB_CONFIG_FILE > log_enb.txt 2> /dev/null & +# -O $ENB_CONFIG_FILE | grep "RAL\|PDCP" & +else +#xterm -hold -e gdb --args +$OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l3 -u0 -b1 -M0 -p2 -g3 -D $EMULATION_DEV_INTERFACE \ + --enb-ral-listening-port $ENB_RAL_LISTENING_PORT \ + --enb-ral-link-id $ENB_RAL_LINK_ID_STRIPPED \ + --enb-ral-ip-address $ENB_RAL_IP_ADDRESS \ + --enb-mihf-remote-port $ENB_MIHF_REMOTE_PORT \ + --enb-mihf-ip-address $ENB_MIHF_IP_ADDRESS \ + --enb-mihf-id $ENB_MIHF_ID \ + -O $ENB_CONFIG_FILE > /dev/null & +# -O $ENB_CONFIG_FILE | grep "RAL\|PDCP" & +fi + +wait_process_started oaisim + +# start MIH-USER +# wait for emulation start +tshark -c 150 -i $EMULATION_DEV_INTERFACE > /dev/null 2>&1 +xterm -hold -title "[eNB][eNB2] MIH_USER" -e $ODTONE_MIH_EXE_DIR/$ENB_MIH_USER --conf.file $ODTONE_MIH_EXE_DIR/$ENB_MIH_USER_CONF_FILE & +wait_process_started $ENB_MIH_USER + +xterm -hold -title "[eNB][eNB2] TVWS Sensing" -e ./server & +#xterm -hold -title "[eNB][eNB2] output" -e tail -f log_enb.txt | grep -i recon & + +sleep 100000 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/link_sap.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/link_sap.conf new file mode 100644 index 0000000000000000000000000000000000000000..c5b63413d0148de7186c2fac37cf9e606dd592df --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/link_sap.conf @@ -0,0 +1,47 @@ +#=============================================================================== +# Brief : Link SAP configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[link] +## +## Link SAP identifier +## +id=link1 + +## +## Link SAP listening +## +port = 1235 + +## +## Link SAP interface technology +## +tec = 802_11 + +## +## Link SAP interface address +## +link_addr = 00:11:22:33:44:55 + +## +## Comma separated list of the Link SAP supported events +## +event_list = link_detected, link_up, link_down, link_parameters_report, link_going_down, link_handover_imminent, link_handover_complete + +[mihf] +ip=127.0.0.1 +local_port=1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/odtone_ue.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/odtone_ue.conf new file mode 100644 index 0000000000000000000000000000000000000000..44545e6206c8f47e0ad1ce8e813349a9f103b076 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/odtone_ue.conf @@ -0,0 +1,73 @@ +#=============================================================================== +# Brief : MIHF configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[mihf] +## +## This mihf's id +## +## Usage: id = <MIHF ID> +## +id = mihf2_ue + +## +## Port on localhost that MIH Users and MIH Link SAPs connect to. +## +## Usage: local_port = <port> +## +local_port = 1025 + +## +## Port to which remote peer MIHF connect to +## +## Usage: remote_port = <port> +## +remote_port = 4551 + +## +## Comma seperated list of remote MIHF's +## +## If you want to test remote MIHF communication add an entry here +## with the IP address of the remote MIHF. +## +## Usage: peers = <mihf id> <ip> <port>, ... +## +peers = mihf1_enb 10.0.0.1 4551 + +## +## Comma separated list of local MIH User SAPs id's and ports +## +## Usage: users = <user sap id> <port> [<supported commands> <supported queries>], ... +## Note: If no command is specified, the MIHF will assume that the MIH-User +## supports all MIH_***_HO_*** commands and the MIH_Get_Information +## Note: If no query is specified, the MIHF will assume that the MIH-User does +## not support the MIH_Get_Information command. +## +#users = user 1235 +users = user_ue 1635 + +## +## Comma separated list of local MIH Link SAPs id's and ports. +## +## Usage: links = <link sap id> <port> <techonoly type> <interface> [<supported events list> <supported commands list>], ... +## +## +links = ue_lte_link00 1234 LTE 00:39:18:36:73:02 00:11:22 5 + +## +## Comma separated list of the MIHF's transport protocol +## +transport = udp diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/ue_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/ue_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..27d56760bea01f85150e9f7e5d21214e5bbf1ae3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_conf/ue_lte_user.conf @@ -0,0 +1,42 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_ue + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/.ue_lte_user_old.cpp.kate-swp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/.ue_lte_user_old.cpp.kate-swp new file mode 100644 index 0000000000000000000000000000000000000000..667b216e6ecc3b70d78a4fe6aa6f9c911d4efc1f Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/.ue_lte_user_old.cpp.kate-swp differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/Jamfile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/Jamfile new file mode 100644 index 0000000000000000000000000000000000000000..c2a6aecbc45e3321cb97fafc4f51a161ac7b59ff --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/Jamfile @@ -0,0 +1,44 @@ +#=============================================================================== +# Brief : MIH-User SAP Application Sample Project Build +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +project enb_lte_user + ; + +exe enb_lte_user + : enb_lte_user.cpp + ../../lib/odtone//odtone + /boost//program_options + ; + +install install + : enb_lte_user + enb_lte_user.conf + ue_lte_user + ue_lte_user.conf + : <location>../../dist + ; + +project ue_lte_user + ; + +exe ue_lte_user + : ue_lte_user.cpp + ../../lib/odtone//odtone + /boost//program_options + ; + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/dummy_130606.tgz b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/dummy_130606.tgz new file mode 100644 index 0000000000000000000000000000000000000000..11ef083a922376ab0b713c88568902ae7483a5ce Binary files /dev/null and b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/dummy_130606.tgz differ diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..b9bbf279de1650cfa71dbeef31960214086d0d98 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user.conf @@ -0,0 +1,40 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_enb + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..4571396bd42632eb3b77ca2d07643036a7031222 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user.cpp @@ -0,0 +1,1695 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <string> +#include <map> +#include <stdio.h> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +//The thresholds for the Energy Detection Sensing algorithm respectively for 10 and 100 samples +#define ED_THRESHOLD_10 23695432 +#define ED_THRESHOLD_100 230445932 +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER (of the eNB) for the demo of the Scenario 2 of SPECTRA project is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Initiallization of the MIH-USER and the MIHF +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | + +// ------------------------------------------------------------------------------------------------------------------------ +// Locally send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to local MIHF) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | + +// ------------------------------------------------------------------------------------------------------------------------ +// Remotely send the Capability Discover and the Event Subscribe primitives +// (from MIH USER to remote MIHF of the UE) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Detect the connection of the UE to the eNB + Start the measurement report process +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... | | +// | | | +// ------------------------------------------------------------------------------------------------------------------------ +// Activate the TVWS link (After running the cognitive algorithms) +// ------------------------------------------------------------------------------------------------------------------------ +// | | | +// |-------------- MIH_Link_Actions.request ------------>|--------------- MIH_Link_Actions.request ----------->| +// | | | +// |<------------- MIH_Link_Actions.confirm -------------|<-------------- MIH_Link_Actions.confirm ------------| +// | | | + +//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + +static const char* const kConf_MIH_Commands = "user.commands"; + +///////////////////////////////////////////////////////////////////////////////////////// + +std::string exec(char* cmd) { + FILE* pipe = popen(cmd, "r"); + if (!pipe) return "ERROR"; + char buffer[128]; + std::string result = ""; + while(!feof(pipe)) { + if(fgets(buffer, 128, pipe) != NULL) + result += buffer; + } + pclose(pipe); + return result; +} + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} + + +/*//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} +*/ + +//Updated from UE code +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + + //Measurements report methods + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + int receive_Sensing_Report(); + void receive_CRRM_Data(); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + void forward_Parameters_Report_indication(odtone::mih::message& m); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + odtone::uint second_link_activated; + odtone::uint count; + odtone::uint sensing_done; + odtone::uint sensing_score; + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + second_link_activated = 0; + count = 0; + sensing_done = 0; + sensing_score = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(/*const odtone::mih::config& cfg,*/ const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + + //send a capability discover request to the remote UE + mih_user::send_MIH_Capability_Discover_request_remote(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + boost::system::error_code ec; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} +// mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, /*boost::cref(cfg),*/ _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address +// for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) +// { +// rcv_link_id.addr = i->addr; +// rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); +// } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + + std::cout<<"NTALL "<<ntal.get()[0].addr<<std::endl; + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; +// std::ostringstream stream; +// std::stringstream st; +// odtone::mih::net_type_addr net_type_addr; +// net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); +// stream << net_type_addr; +// std::string s; +// std::stringstream st (stream.str()); +// while (getline (st, s, '\n')) +// { +// std::stringstream ss(s); +// getline (ss, name, ':'); +// getline (ss, value, '\n'); +// // ss>>name>>value; +// std::cout<<" name "<<name<<" value "<<value<<std::endl; +// } +/* + st<<i->addr; + getline (st, s, ' '); + std::cout<<"LINK_TUPLE_ID s "<<s<<std::endl; + + odtone::mih::l2_3gpp_addr add; + add.value = s;*/ + li.addr = /*add*/ /*boost::get<odtone::mih::l2_3gpp_addr>(*/i->addr/*)*/; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + if (std::find(_link_id_list.begin(), _link_id_list.end(), li)==_link_id_list.end()) + _link_id_list.push_back(li); // save the link identifier of the network interface + + std::cout<<"LINK_TUPLE_ID - Link identifier:after "<<link_addr2string(&li.addr).c_str() <<" "<<_link_id_list[_link_id_list.size()-1]<<std::endl; + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<li<<std::endl; +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + link_act_req.id = link; +// } + + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + std::cout<<"_link received from parameters report "<<link<<std::endl; + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + std::cout<<"_link_id_list addr "<<i->addr<<" TYPE "<<i->type<<std::endl; + } + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + +// link_act_req.action.param.param = res; + +// link_act_req.ex_time = 0; + link_act_req.ex_time = 5000; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + +// odtone::mih::id mid_ue; +// mid_ue.assign("mihf2_ue"); + m.destination(/*mid_ue*/_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st); +// & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); +// if (larl) { +// log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); +// for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) +// { +// log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), +// ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); +// } +// } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +// #ifdef SCENARIO_1 +// if (larl) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// if (rsp->result.get() == odtone::mih::link_ac_success) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// _current_link_action_request += 1; +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_1 +// +// #ifdef SCENARIO_2 +// // 2nd scenario: Activate all resources, then deactivate all resources +// if (larl.get().size() > 0) { +// odtone::mih::link_action_rsp *rsp = &larl->front(); +// if (++_current_link_action_request < _nb_of_link_action_requests) { +// if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// } +// else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { +// _current_link_action_request = 0; +// mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); +// } +// else { // Ends the scenario +// mih_user::send_MIH_Event_Unsubscribe_request(); +// } +// } +// #endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +void mih_user::receive_CRRM_Data() +{ + + //CRRM data report + std::string result = exec ("./CRMClientmain 0 1"); +} + +//----------------------------------------------------------------------------- +int mih_user::receive_Sensing_Report() +//----------------------------------------------------------------------------- +{ + std::string result = exec ("./client 127.0.0.1 4546 10 | grep \"SCORE\""); + std::string tmp; + int value; + std::stringstream ss(result); + ss >> tmp >> value; + std::cout<<"Result of Sensing "<<value<<std::endl; + sensing_done = 1; + return value; +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link, link1; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + std::cout<<"LINK_TUPLE_ID - Link identifier: "<<link<<std::endl; + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value ); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + + + //eNB1: Forward the message to UE 2 + forward_Parameters_Report_indication(msg); + + //eNB2 : Action Power Up the TVWS Link after running the cognitive algorithm + + //First Phase: Collect the data as input fot the cognitive algorithm + //Sensing data + if (sensing_done == 0) + sensing_score = receive_Sensing_Report(); + + //CRRM Data + receive_CRRM_Data(); + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { +// log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available +// for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { +// if(*i==link) + odtone::mih::l2_3gpp_addr add; + odtone::mih::link_type type; + type = odtone::mih::link_type_lte; +// add.value = "l2_3gpp_addr"; + link1.addr = add; + link1.type = type; +// send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_power_up); + + //Send ONLY ONCE RRC_Connection_Reconfiguration request to the UE on LTE link 0 + if (second_link_activated==0) + { + if (count == 20) + { + send_MIH_Link_Action_Power_Up_request(_link_id_list[1]);//Link LTE + second_link_activated = 1; + } + } + + count++; +// continue; +// send_MIH_Link_Action_Power_Up_request(link); +// if (link_id2string(link).c_str() == "LTE") +// } + } + } + +} + + +//Forward the MIH_Link_Parameters_Report to the UE of the CPE +//----------------------------------------------------------------------------- +void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + m >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + odtone::mih::message msg; + msg.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf3_ue"); + msg.destination(mid_ue); + + //mn_ho_candidate_query is used to constuct/send the message containing parameters reports + msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- before MIH_User_PARAMETERS.indication --->]["+msg.destination().to_string()+"]\n"); + + + _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, "MIH_User_PARAMETERS.indication- SENT (towards UE)\n"); + +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl1; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lcp1.type = lp1; + lcp2.type = lp2; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th2.threshold_val = -19; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + + + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.source(_mihuserid); + odtone::mih::id mid_ue; + mid_ue.assign("mihf2_ue"); + m.destination(mid_ue); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ " --->][ ADDR LINK " + + m.destination().to_string() +"]\n"); + std::cout<<"LINK TUPLE ID "<<lti<<std::endl; + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti); +// & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", (lcsl[iter].thold.threshold_val/256)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("enb_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1_enb"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user_tcs.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user_tcs.cpp new file mode 100644 index 0000000000000000000000000000000000000000..72d46e791bee1fde516e9f592ec1f77bdec6d605 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/enb_lte_user_tcs.cpp @@ -0,0 +1,1412 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +// Fatma HRIZI EURECOM <hrizi@eurecom>fr> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +//This file is the implementation of the MIH user in the eNB + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER is the following (with eRALlteDummy and NASRGDummy executables) +// +--------+ +-----+ +// |MIH_USER| |MIH-F| +// +---+----+ +--+--+ +// | | _current_link_action_request = 0 +// |---------- User_Register.indication ---------------->| (supported_commands) Handler next msg=user_reg_handler +// | | +// |---------- Capability_Discover.request ------------->| Handler next msg=receive_MIH_Capability_Discover_confirm +// |<--------- Capability_Discover.confirm --------------| (success) +// | | +// |---------- Event_Subscribe.request ----------------->| Handler next msg=receive_MIH_Event_Subscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 1: Sequentially activate and deactivate each resource +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// |<--------- Link_Actions.confirm ---------------------| (success) +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// Scenario 2: Activate all resources, then deactivate all resources +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Link_Actions.request -------------------->| (activate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | _current_link_action_request = 0 +// |---------- Link_Actions.request -------------------->| (deactivate-resources[_current_link_action_request]) +// | | Handler next msg=receive_MIH_Link_Actions_confirm +// | | _current_link_action_request = _current_link_action_request + 1 +// |<--------- Link_Actions.confirm ---------------------| (success) +// | . | +// | . | +// | . | +// | | +// ------------------------------------------------------------------------------------------------------------------------ +// | | +// |---------- Event_Unsubscribe.request --------------->| Handler next msg=receive_MIH_Event_Unsubscribe_confirm +// |<--------- Event_Subscribe.confirm ------------------| (success) +// | | +// | | +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s; + if(evtP.get(odtone::mih::mih_evt_link_detected)) s = std::string("DETECTED "); + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s; + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s = std::string("Link_Get_Parameters "); + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::link_id link_id; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + link_id.addr = i->addr; + if (i != ntalP->begin()) { + s += " / "; + } + s += link_id2string(link_id); + } + + return s; +} + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(const odtone::mih::config& cfg); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES) +//----------------------------------------------------------------------------- +{ + + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } + else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + log_(0, ""); + + // + // Local Capability Discover Request + // + odtone::mih::message msg; + _mihfid.assign("mihf2"); + msg << odtone::mih::request(odtone::mih::request::capability_discover, _mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Capability_Discover.request --->]["+msg.destination().to_string()+"]\n"); + _mihf.async_send(msg, boost::bind(&mih_user::receive_MIH_Capability_Discover_confirm, this, _1)); + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)"); + log_(0, ""); + + //****eNB***** + // Remote Capability Discover Request + // + mih_user::send_MIH_Capability_Discover_request(cfg); + + //Trigger Link_Configure_Thresholds Request + // odtone::mih::message m; + //m.destination(msg.source()); + //mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + //****eNB***** +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + } + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + + break; + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} +/* +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + log_(0, "MIH_Link_Detected.indication - End\n"); +} +*/ +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + + odtone::mih::link_det_info_list ldil; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + +boost::system::error_code ec; + for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) + { + log_(0, " - LINK_ID - Link identifier: ", link_id2string(i->id).c_str()); + log_(0, " - Network ID: ", i->network_id); + log_(0, " - SINR: ", i->sinr); + log_(0, " - Data_rate: ", i->data_rate); + } + + send_MIH_Link_Configure_Thresholds_request(msg, ec); + + + log_(0, "MIH_Link_Detected.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + + for (odtone::mih::link_param_rpt_list::iterator i = lprl.begin(); i != lprl.end(); i++) + { + if (const odtone::mih::threshold *th = boost::get<odtone::mih::threshold>(&i->thold)) { + + log_(0, " - BW Threshold crossed, Value:", boost::get<odtone::mih::link_param_val>(i->param.value)); + //link_action if free channel is available + } + else{ + log_(0, " -Regular Report for BW Threshold "); + //update MEAS + } + } + + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + //boost::optional<odtone::mih::net_type_addr_list> ntal; + //boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::request(odtone::mih::request::capability_discover) + //& odtone::mih::tlv_net_type_addr_list(ntal) + //& odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + _mihfid.assign("mihf2"); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + odtone::mih::threshold th; + std::vector<odtone::mih::threshold> thl; + + odtone::mih::link_tuple_id lti; +// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + + + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp; + odtone::mih::link_param_lte lp; + + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + + odtone::mih::link_det_info_list ldil; + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_det_info_list(ldil); + for (odtone::mih::link_det_info_list::iterator i = ldil.begin(); i != ldil.end(); i++) + { + + + lti.type = i->id.type; + lti.addr = i->id.addr; + lp = odtone::mih::link_param_lte_bandwidth; + lcp.type = lp; + if ( link_measures_request ==0){ + lcp.timer_interval = 10; + lcp.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } + else{ + lcp.timer_interval = 0; + lcp.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + th.threshold_val = 5; + th.threshold_x_dir = odtone::mih::threshold::above_threshold; + thl.push_back(th); + lcp.threshold_list = thl; + lcpl.push_back(lcp); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + link_id2string(lti).c_str()+ + " --->]["+_mihfid.to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + } + +/* + if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); + + if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", th.threshold_val); + + if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +*/ +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + // T odtone::uint iter; + // T odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + // Todtone::mih::link_cfg_status_list lcsl; + // Todtone::mih::link_cfg_status lcp; + //odtone::mih::link_param_gen lp; + + // T odtone::mih::link_tuple_id lti; + + //msg >> odtone::mih::confirm() + // & odtone::mih::tlv_status(st) + // & odtone::mih::tlv_link_identifier(lti) + // & odtone::mih::tlv_link_cfg_status_list(lcsl); + + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1234), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf1"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..27d56760bea01f85150e9f7e5d21214e5bbf1ae3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user.conf @@ -0,0 +1,42 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_ue + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..40b4891a1865a45cd7390e62cd14a3f899d77b3f --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user.cpp @@ -0,0 +1,1658 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_req +//#define SCENARIO_1 // Sequentially activate and deactivate each resou +//#define SCENARIO_2 // Activate all resources, then deactivate all resour +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER demo is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// etc etc etc + +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +// int& operator [](const link_param_lte_enum value) +// { +// static std::map<link_param_lte_enum, std::string> var; +// if (var.size() == 0) +// { +// var[link_param_lte_rsrp]="link_param_lte_rsrp"; +// var[link_param_lte_rsrq]="link_param_lte_rsrq"; +// var[link_param_lte_cqi]="link_param_lte_cqi"; +// var[link_param_lte_bandwidth]="link_param_lte_bandwidth"; +// var[link_param_lte_pkt_delay]="link_param_lte_pkt_delay"; +// } +// return var[value]; +// } + +const char * MeasurementTypes[]= +{ + "LTE_RSRP", /**< RSRP. */ + "LTE_RSRQ", /**< RSRQ. */ + "link_param_lte_cqi", /**< Multicast packet loss rate.*/ + "link_param_lte_bandwidth", /**< System Load. */ + "link_param_lte_pkt_delay", /**< Number of registered users. */ + "link_param_lte_pkt_loss", /**< Number of active users. */ + "link_param_lte_l2_buffer", /**< Congestion windows of users. */ + "link_param_lte_MN_cap", /**< Congestion windows of users. */ + "link_param_lte_embms", /**< Congestion windows of users. */ + "link_param_lte_jumbo_feasibility", /**< Congestion windows of users. */ + "link_param_lte_jumbo_setup", /**< Congestion windows of users. */ + "link_param_lte_active_embms", /**< Transmission rate of users. */ + "link_param_lte_link_congestion", /**< Link congestion. */ +}; + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)){ + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type"; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + + + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + void send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + //send a capability discover request to the remote eNB + mih_user::send_MIH_Capability_Discover_request_remote(); + +} + + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1;; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + // send Link_Action / Power Up + lid.type = odtone::mih::link_type_lte; + lid.addr = ldi.id.addr; + //send_MIH_Link_Action_Power_Up_plus_scan_request(lid); + log_(0, "MIH_Link_Detected.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + + link_act_req.id = link; + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_enb; + mid_enb.assign("mihf1_enb"); + m.destination(mid_enb); +// m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); + mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(link); + +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); +/* + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl){ + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2*/ + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + +// odtone::mih::link_param_report lpr = lprl.back(); +// +// log_(0, "Meausrement Type: ", lpr.param.type); +// if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&lpr.param.value)){ +// log_(0, "Meausrement Value: ", *value); +// +// }; +// const odtone::mih::threshold *th; + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { +// odtone::mih::link_param_lte * index = boost::get<odtone::mih::link_param_lte>(&i->param.type); +// int val = (short)*index; +// log_(0, "Meausrement Type: index ", *index); + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); +// log_(0, "Meausrement Type: ", MeasurementTypes[index]); +// log_(0, "Meausrement Type: ", &i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value /*- 65536*/); + } +// if (th = boost::get<odtone::mih::threshold> (&i->thold)) +// { +// log_(0, "Threshold crossed ", boost::get<odtone::mih::link_param_val> (i->param.value)); +// } +// else +// { +// log_(0,"Regular Report"); +// } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + std::vector<odtone::mih::threshold> thl1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::threshold th3; + std::vector<odtone::mih::threshold> thl3; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_cfg_param lcp3; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + odtone::mih::link_param_lte lp3; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lp3 = odtone::mih::link_param_lte_cqi; + lcp1.type = lp1; + lcp2.type = lp2; + lcp3.type = lp3; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + lcp3.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + lcp3.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp3.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + lcp3.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_val = -19; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + th3.threshold_val = 0; + th3.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + thl3.push_back(th3); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + lcp3.threshold_list = thl3; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + lcpl.push_back(lcp3); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + if(lp2 == odtone::mih::link_param_lte_rsrq) {log_(0, "\t LTE link parameter LTE RSRQ");} + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + +// //----------------------------------------------------------------------------- +// void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +// //----------------------------------------------------------------------------- +// { +// odtone::mih::message m; +// odtone::mih::threshold th; +// std::vector<odtone::mih::threshold> thl; +// odtone::mih::link_tuple_id lti; +// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; +// //List of the link threshold parameters +// odtone::mih::link_cfg_param_list lcpl; +// odtone::mih::link_cfg_param lcp; +// odtone::mih::link_param_lte lp; +// //odtone::mih::link_param_gen lp; +// +// odtone::mih::link_param_type typr; +// +// log_(0,""); +// log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); +// +// //link_tuple_id +// lti.type = rcv_link_id.type; +// lti.addr = rcv_link_id.addr; +// +// //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); +// +// +// //link_param_gen_data_rate = 0, /**< Data rate. */ +// //link_param_gen_signal_strength = 1, /**< Signal strength. */ +// //link_param_gen_sinr = 2, /**< SINR. */ +// //link_param_gen_throughput = 3, /**< Throughput. */ +// //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ +// //lp = odtone::mih::link_param_lte_bandwidth; +// lp = odtone::mih::link_param_lte_rsrp; +// lcp.type = lp; +// +// link_measures_request = 0; +// if ( link_measures_request ==0){ +// // Set Timer Interval (in ms) +// lcp.timer_interval = 3000; +// //th_action_normal = 0, /**< Set normal threshold. */ +// //th_action_one_shot = 1, /**< Set one-shot threshold. */ +// //th_action_cancel = 2 /**< Cancel threshold. */ +// lcp.action = odtone::mih::th_action_normal; +// link_measures_request = 1; +// } else if ( link_measures_request==1){ +// // Set Timer Interval (in ms) +// lcp.timer_interval = 0; +// lcp.action = odtone::mih::th_action_cancel; +// link_measures_request = 0; +// } +// +// //above_threshold = 0, /**< Above threshold. */ +// //below_threshold = 1, /**< Below threshold. */ +// th.threshold_val = -105; +// th.threshold_x_dir = odtone::mih::threshold::above_threshold; +// +// thl.push_back(th); +// lcp.threshold_list = thl; +// lcpl.push_back(lcp); +// +// m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) +// & odtone::mih::tlv_link_identifier(lti) +// & odtone::mih::tlv_link_cfg_param_list(lcpl); +// +// m.destination(msg.source()); +// +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ +// // link_tupple_id2string(lti).c_str() + +// link_id2string(lti).c_str()+ +// " --->]["+_mihfid.to_string()+"]\n"); +// +// _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); +// +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); +// +// log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); +// +// //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} +// //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} +// //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} +// //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} +// //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} +// if(lp == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} +// +// log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); +// +// if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} +// if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} +// if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} +// +// log_(0, "\t Threshold value: ", th.threshold_val); +// +// if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} +// if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} +// +// log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +// } + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", ((short) lcsl[iter].thold.threshold_val)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +// //----------------------------------------------------------------------------- +// void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +// //----------------------------------------------------------------------------- +// { +// log_(0, ""); +// log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); +// +// // T odtone::uint iter; +// // T odtone::mih::status st; +// +// //boost::optional<odtone::mih::link_cfg_status_list> lcsl; +// // Todtone::mih::link_cfg_status_list lcsl; +// // Todtone::mih::link_cfg_status lcp; +// //odtone::mih::link_param_gen lp; +// +// // T odtone::mih::link_tuple_id lti; +// +// //msg >> odtone::mih::confirm() +// // & odtone::mih::tlv_status(st) +// // & odtone::mih::tlv_link_identifier(lti) +// // & odtone::mih::tlv_link_cfg_status_list(lcsl); +// +// +// log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); +// log_(0,""); +// } + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("ue_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf2_ue"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + \ No newline at end of file diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user_old.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user_old.cpp new file mode 100644 index 0000000000000000000000000000000000000000..40a286434f22d4d86dfcc901e9ee1fd79500dbce --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/mih_user/lte_test_user/ue_lte_user_old.cpp @@ -0,0 +1,1437 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_requests +//#define SCENARIO_1 // Sequentially activate and deactivate each resource +//#define SCENARIO_2 // Activate all resources, then deactivate all resources +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER demo is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// etc etc etc + +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type "; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + + + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + void send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + mih_user::send_MIH_Capability_Discover_request_remote(); +} + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1; + } + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + // send Link_Action / Power Up + lid.type = odtone::mih::link_type_lte; + lid.addr = ldi.id.addr; + //send_MIH_Link_Action_Power_Up_plus_scan_request(lid); + log_(0, "MIH_Link_Detected.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + + link_act_req.id = link; + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + + + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ---- 2=>CQI ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value ); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_enb; + mid_enb.assign ("mihf1_enb"); + m.destination(mid_enb); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), m); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); + mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(link); + +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); // DUMMY +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); // DUMMY +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); // DUMMY +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters +// log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); +// if (larl) { +// log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); +// for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) +// { +// log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), +// ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); +// if (_last_link_action_type == odtone::mih::link_ac_type_power_up ) +// { +// send_MIH_Link_Action_Power_Up_plus_scan_request(i->id); +// } +// } +// } +// log_(0, ""); + +/* + + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + }*/ +// #endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th; + std::vector<odtone::mih::threshold> thl; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp; + odtone::mih::link_param_lte lp; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp = odtone::mih::link_param_lte_rsrp; + lcp.type = lp; + + link_measures_request = 0; + if ( link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp.timer_interval = 0; + lcp.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th.threshold_val = -105; + th.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl.push_back(th); + lcp.threshold_list = thl; + lcpl.push_back(lcp); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+_mihfid.to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + + log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); + + if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", th.threshold_val); + + if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + // T odtone::uint iter; + // T odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + // Todtone::mih::link_cfg_status_list lcsl; + // Todtone::mih::link_cfg_status lcp; + //odtone::mih::link_param_gen lp; + + // T odtone::mih::link_tuple_id lti; + + //msg >> odtone::mih::confirm() + // & odtone::mih::tlv_status(st) + // & odtone::mih::tlv_link_identifier(lti) + // & odtone::mih::tlv_link_cfg_status_list(lcsl); + + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("ue_lte_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf2_ue"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/oai_conf/start_ue.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/oai_conf/start_ue.bash new file mode 100644 index 0000000000000000000000000000000000000000..f2bc5f59f5c7d925aa69334a7fb2291bcbed962b --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue1/oai_conf/start_ue.bash @@ -0,0 +1,206 @@ +#!/bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file start_ue.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# +#------------------------------------------------ +# OAI NETWORKING +#------------------------------------------------ +declare -x EMULATION_DEV_INTERFACE="eth0" +declare -x EMULATION_DEV_ADDRESS="192.168.13.2" + +declare -x IP_DRIVER_NAME="oai_nw_drv" +declare -x LTEIF="oai0" +declare -x UE_IPv4="10.0.0.2" +declare -x UE_IPv6="2001:1::2" +declare -x UE_IPv6_CIDR=$UE_IPv6"/64" +declare -x UE_IPv4_CIDR=$UE_IPv4"/24" +declare -a NAS_IMEI=( 3 9 1 8 3 6 7 3 0 2 0 0 0 0 ) +declare -x IP_DEFAULT_MARK="1" # originally 3 + +#------------------------------------------------ +# OAI MIH +#------------------------------------------------ +declare -x UE_MIHF_IP_ADDRESS="127.0.0.1" +declare -x UE_RAL_IP_ADDRESS="127.0.0.1" +LOG_FILE="/tmp/oai_sim_ue.log" + +#------------------------------------------------ +MIH_LOG_FILE="mih-f_ue.log" + +# EXE options +EXE_MODE="PROD" # PROD or DEBUG + +########################################################### +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +source $THIS_SCRIPT_PATH/env_802dot21.bash +########################################################### +bash_exec "ifconfig $EMULATION_DEV_INTERFACE up $EMULATION_DEV_ADDRESS netmask 255.255.255.0" +########################################################### +IPTABLES=/sbin/iptables +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +declare -x OPENAIR_DIR="" +declare -x OPENAIR1_DIR="" +declare -x OPENAIR2_DIR="" +declare -x OPENAIR3_DIR="" +declare -x OPENAIR_TARGETS="" +########################################################### + +set_openair +cecho "OPENAIR_DIR = $OPENAIR_DIR" $green +cecho "OPENAIR1_DIR = $OPENAIR1_DIR" $green +cecho "OPENAIR2_DIR = $OPENAIR2_DIR" $green +cecho "OPENAIR3_DIR = $OPENAIR3_DIR" $green +cecho "OPENAIR_TARGETS = $OPENAIR_TARGETS" $green + + +################################################## +# LAUNCH UE +################################################## + +echo "Bringup UE interface" +pkill oaisim > /dev/null 2>&1 +pkill oaisim > /dev/null 2>&1 +pkill $MIH_F > /dev/null 2>&1 +pkill $UE_MIH_USER > /dev/null 2>&1 +rmmod -f $IP_DRIVER_NAME > /dev/null 2>&1 + +bash_exec "insmod $OPENAIR2_DIR/NAS/DRIVER/LITE/$IP_DRIVER_NAME.ko oai_nw_drv_IMEI=${NAS_IMEI[0]},${NAS_IMEI[1]},${NAS_IMEI[2]},${NAS_IMEI[3]},${NAS_IMEI[4]},${NAS_IMEI[5]},${NAS_IMEI[6]},${NAS_IMEI[7]},${NAS_IMEI[8]},${NAS_IMEI[9]},${NAS_IMEI[10]},${NAS_IMEI[11]},${NAS_IMEI[12]},${NAS_IMEI[13]}" +bash_exec "ip route flush cache" +bash_exec "ip link set $LTEIF up" +sleep 1 +bash_exec "ip addr add dev $LTEIF $UE_IPv4_CIDR" +bash_exec "ip addr add dev $LTEIF $UE_IPv6_CIDR" +sleep 1 +bash_exec "sysctl -w net.ipv4.conf.all.log_martians=1" +assert " `sysctl -n net.ipv4.conf.all.log_martians` -eq 1" $LINENO +bash_exec "sysctl -w net.ipv4.conf.all.rp_filter=0" +assert " `sysctl -n net.ipv4.conf.all.rp_filter` -eq 0" $LINENO +bash_exec "ip route flush cache" + +# Check table 200 lte in /etc/iproute2/rt_tables +fgrep lte /etc/iproute2/rt_tables > /dev/null +if [ $? -ne 0 ]; then + echo '200 lte ' >> /etc/iproute2/rt_tables +fi +ip rule add fwmark $IP_DEFAULT_MARK table lte +ip -4 route add default dev $LTEIF table lte +ip -6 route add default dev $LTEIF table lte +ip route add 239.0.0.160/28 dev $EMULATION_DEV_INTERFACE + +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +#All other traffic is sent on the RAB you want (mark = RAB ID) +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK + +rotate_log_file $MIH_LOG_FILE + +echo "printing the MIH file path" +echo "$ODTONE_MIH_EXE_DIR/$MIH_F $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE" +echo "$ODTONE_MIH_EXE_DIR/$UE_MIH_USER $ODTONE_MIH_EXE_DIR/$UE_MIH_USER_CONF_FILE" + +# start MIH-F +xterm -hold -title "[UE][UE1] MIHF" -e $ODTONE_MIH_EXE_DIR/$MIH_F --log 4 --conf.file $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE > $MIH_LOG_FILE 2>&1 & +wait_process_started $MIH_F +sleep 3 + +NOW=$(date +"%Y-%m-%d.%Hh_%Mm_%Ss") +rm -f $LOG_FILE + +UE_RAL_LINK_ID=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +UE_RAL_LINK_ID=`trim2 $UE_RAL_LINK_ID` +UE_RAL_LINK_ID=`echo $UE_RAL_LINK_ID | cut -d" " -f1` +UE_RAL_LINK_ID_STRIPPED=${UE_RAL_LINK_ID%%??} + +UE_RAL_LISTENING_PORT=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +UE_RAL_LISTENING_PORT=`trim2 $UE_RAL_LISTENING_PORT` +UE_RAL_LISTENING_PORT=`echo $UE_RAL_LISTENING_PORT | cut -d" " -f2` + +UE_MIHF_REMOTE_PORT=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep local_port | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` + +UE_MIHF_ID=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep id | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` + +#xterm -hold -e gdb --args +# $EMULATION_DEV_INTERFACE -D192.168.13.2 +#sudo ip route add 239.0.0.160/28 dev $EMULATION_DEV_INTERFACE +#$OPENAIR2_DIR/NAS/DRIVER/LITE/RB_TOOL/rb_tool -a -c0 -i0 -z0 -s 10.0.0.2 -t 10.0.0.1 -r 1 + +if [ $EXE_MODE = "DEBUG" ] ; then + echo "$OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l7 -u1 -b0 -M1 -p2 -g1 -D $EMULATION_DEV_ADDRESS --ue-ral-listening-port $UE_RAL_LISTENING_PORT --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED --ue-ral-ip-address $UE_RAL_IP_ADDRESS --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS --ue-mihf-id $UE_MIHF_ID " + + $OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l7 -u1 -b0 -M1 -p2 -g1 -D $EMULATION_DEV_ADDRESS \ + --ue-ral-listening-port $UE_RAL_LISTENING_PORT \ + --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED \ + --ue-ral-ip-address $UE_RAL_IP_ADDRESS \ + --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT \ + --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS \ + --ue-mihf-id $UE_MIHF_ID > log_ue.txt & +else + echo "$OPENAIR_TARGETS/SIMU/USER/oaisim -a -l3 -u1 -b0 -M1 -p2 -g1 -D $EMULATION_DEV_ADDRESS --ue-ral-listening-port $UE_RAL_LISTENING_PORT --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED --ue-ral-ip-address $UE_RAL_IP_ADDRESS --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS --ue-mihf-id $UE_MIHF_ID " + + $OPENAIR_TARGETS/SIMU/USER/oaisim -a -u1 -b0 -M1 -p2 -g1 -D $EMULATION_DEV_ADDRESS \ + --ue-ral-listening-port $UE_RAL_LISTENING_PORT \ + --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED \ + --ue-ral-ip-address $UE_RAL_IP_ADDRESS \ + --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT \ + --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS \ + --ue-mihf-id $UE_MIHF_ID > /dev/null & + +fi + +wait_process_started oaisim + + +# start MIH-USER + +# wait for emulation start +tshark -c 500 -i $EMULATION_DEV_INTERFACE > /dev/null 2>&1 +sleep 5 + +xterm -hold -title "[UE][UE1] MIH_USER" -e $ODTONE_MIH_EXE_DIR/$UE_MIH_USER --conf.file $ODTONE_MIH_EXE_DIR/$UE_MIH_USER_CONF_FILE & +wait_process_started $UE_MIH_USER + +sleep 100000 + + + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/link_sap.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/link_sap.conf new file mode 100644 index 0000000000000000000000000000000000000000..c5b63413d0148de7186c2fac37cf9e606dd592df --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/link_sap.conf @@ -0,0 +1,47 @@ +#=============================================================================== +# Brief : Link SAP configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2013 Universidade Aveiro +# Copyright (C) 2009-2013 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[link] +## +## Link SAP identifier +## +id=link1 + +## +## Link SAP listening +## +port = 1235 + +## +## Link SAP interface technology +## +tec = 802_11 + +## +## Link SAP interface address +## +link_addr = 00:11:22:33:44:55 + +## +## Comma separated list of the Link SAP supported events +## +event_list = link_detected, link_up, link_down, link_parameters_report, link_going_down, link_handover_imminent, link_handover_complete + +[mihf] +ip=127.0.0.1 +local_port=1025 diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/odtone_ue.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/odtone_ue.conf new file mode 100644 index 0000000000000000000000000000000000000000..c2f596c5ccaeaf32b7d90bf7e2ad97a4b2db9294 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/odtone_ue.conf @@ -0,0 +1,74 @@ +#=============================================================================== +# Brief : MIHF configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +[mihf] +## +## This mihf's id +## +## Usage: id = <MIHF ID> +## +id = mihf3_ue + +## +## Port on localhost that MIH Users and MIH Link SAPs connect to. +## +## Usage: local_port = <port> +## +local_port = 1025 + +## +## Port to which remote peer MIHF connect to +## +## Usage: remote_port = <port> +## +remote_port = 4551 + +## +## Comma seperated list of remote MIHF's +## +## If you want to test remote MIHF communication add an entry here +## with the IP address of the remote MIHF. +## +## Usage: peers = <mihf id> <ip> <port>, ... +## +#peers = mihf1_enb 10.0.0.1 4551 udp, mihf4_enb 10.0.0.4 4551 udp +peers = mihf1_enb 192.168.14.3 4551 udp, mihf4_enb 10.0.2.4 4551 udp + +## +## Comma separated list of local MIH User SAPs id's and ports +## +## Usage: users = <user sap id> <port> [<supported commands> <supported queries>], ... +## Note: If no command is specified, the MIHF will assume that the MIH-User +## supports all MIH_***_HO_*** commands and the MIH_Get_Information +## Note: If no query is specified, the MIHF will assume that the MIH-User does +## not support the MIH_Get_Information command. +## +#users = user 1235 +users = user_ue 1635 + +## +## Comma separated list of local MIH Link SAPs id's and ports. +## +## Usage: links = <link sap id> <port> <techonoly type> <interface> [<supported events list> <supported commands list>], ... +## +## +links = ue_lte_link00 1234 LTE 00:39:18:36:73:02 00:11:22 5 + +## +## Comma separated list of the MIHF's transport protocol +## +transport = udp diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/ue2_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/ue2_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..b50aab8e281e7e2d31a12b12b71b4da6297e8281 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_conf/ue2_user.conf @@ -0,0 +1,42 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_ue2 + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/Jamfile b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/Jamfile new file mode 100644 index 0000000000000000000000000000000000000000..42075be3e268935cc90c2b9d47b905857f16f4a3 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/Jamfile @@ -0,0 +1,33 @@ +#=============================================================================== +# Brief : MIH-User SAP Application Sample Project Build +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +project ue2_user + ; + +exe ue2_user + : ue2_user.cpp + ../../lib/odtone//odtone + /boost//program_options + ; + +install install + : ue2_user + ue2_user.conf + : <location>../../dist + ; + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/ue2_user.conf b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/ue2_user.conf new file mode 100644 index 0000000000000000000000000000000000000000..b50aab8e281e7e2d31a12b12b71b4da6297e8281 --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/ue2_user.conf @@ -0,0 +1,42 @@ +#=============================================================================== +# Brief : MIH-User configuration file +# Authors : Carlos Guimaraes <cguimaraes@av.it.pt> +# Bruno Santos <bsantos@av.it.pt> +#------------------------------------------------------------------------------- +# ODTONE - Open Dot Twenty One +# +# Copyright (C) 2009-2012 Universidade Aveiro +# Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +# +# This software is distributed under a license. The full license +# agreement can be found in the file LICENSE in this distribution. +# This software may not be copied, modified, sold or distributed +# other than expressed in the named license agreement. +# +# This software is distributed without any warranty. +#=============================================================================== + +## +## User id +## +[user] +id = user_ue2 + +## +## Commands supported by the MIH-User +## +commands = mih_link_get_parameters, mih_link_configure_thresholds, mih_link_actions, mih_net_ho_candidate_query, mih_net_ho_commit, mih_n2n_ho_query_resources, mih_n2n_ho_commit, mih_n2n_ho_complete, mih_mn_ho_candidate_query, mih_mn_ho_commit, mih_mn_ho_complete + +## +## Port used for communication with MIHF +## +[conf] +port = 1635 + +## +## MIHF configuration. For the default demonstration leave as is. +## +[mihf] +local_port = 1025 + + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/ue2_user.cpp b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/ue2_user.cpp new file mode 100644 index 0000000000000000000000000000000000000000..1e947bb4f6ae57ffd386f5c5058c5d920b84e2df --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/mih_user/lte_test_user/ue2_user.cpp @@ -0,0 +1,1712 @@ +//============================================================================== +// Brief : MIH-User +// Authors : Bruno Santos <bsantos@av.it.pt> +//------------------------------------------------------------------------------ +// ODTONE - Open Dot Twenty One +// +// Copyright (C) 2009-2012 Universidade Aveiro +// Copyright (C) 2009-2012 Instituto de Telecomunicações - Pólo Aveiro +// +// This software is distributed under a license. The full license +// agreement can be found in the file LICENSE in this distribution. +// This software may not be copied, modified, sold or distributed +// other than expressed in the named license agreement. +// +// This software is distributed without any warranty. +//============================================================================== + +#include <odtone/base.hpp> +#include <odtone/debug.hpp> +#include <odtone/logger.hpp> +#include <odtone/mih/request.hpp> +#include <odtone/mih/response.hpp> +#include <odtone/mih/indication.hpp> +#include <odtone/mih/confirm.hpp> +#include <odtone/mih/tlv_types.hpp> +#include <odtone/sap/user.hpp> + +#include <boost/utility.hpp> +#include <boost/bind.hpp> +#include <boost/tokenizer.hpp> +#include <boost/foreach.hpp> +#include <boost/format.hpp> + +#include <iostream> +#include <map> +#include <time.h> + +/////////////////////////////////////////////////////////////////////////////// + +// Definition of the scenario to execute +#define NB_OF_RESOURCES 4 // Should not exceed mih_user::_max_link_action_req +//#define SCENARIO_1 // Sequentially activate and deactivate each resou +//#define SCENARIO_2 // Activate all resources, then deactivate all resour +#define NUM_PARM_REPORT 10 + +/////////////////////////////////////////////////////////////////////////////// +// The scenario coded in this MIH-USER demo is the following +// +--------+ +-----+ +---------+ +// |MIH_USER| |MIH-F| |LINK_SAP | +// +---+----+ +--+--+ +----+----+ +// | | | +// ... (start of MIH-F here) ... ... +// | |---------- Link_Capability_Discover.request --------X| +// ... (start of LINK_SAP here) ... ... +// | |<--------- Link_Register.indication -----------------| +// | |---------- Link_Capability_Discover.request -------->| +// | |<--------- Link_Capability_Discover.confirm ---------| +// | | | +// ... (start of MIH USER here) ... ... +// |---------- MIH_User_Register.indication ------------>| (supported_commands) | +// | | | +// |---------- MIH_Capability_Discover.request --------->| | +// |<--------- MIH_Capability_Discover.confirm ----------| | +// | | | +// |---------- MIH_Event_Subscribe.request ------------->|---------- Link_Event_Subscribe.request ------------>| +// |<--------- MIH_Event_Subscribe.confirm --------------|<--------- Link_Event_Subscribe.confirm -------------| +// | | | +// |---------- MIH_Link_Actions.request ---------------->|---------- Link_Actions.request -------------------->| +// | (POWER UP + SCAN) | (POWER UP + SCAN) | +// ... ... ... +// | | | +// |<--------- Link_Detected.indication -----------------|<--------- Link_Detected.indication -----------------| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | RRC Connection Reestablishment notification) | +// | | | +// |---------- MIH_Link_Configure_Thresholds.request --->|---------- Link_Configure_Thresholds.request ------->| +// | | | +// |<--------- Link_Up.indication -----------------------|<--------- Link_Up.indication -----------------------| +// | | (RRC Connection reconfiguration notification) | +// |<--------- MIH_Link_Configure_Thresholds.confirm ----|<--------- Link_Configure_Thresholds.confirm --------| +// | | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Actions.confirm -----------------|<--------- Link_Actions.confirm ---------------------| +// | (Success) | | +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// |<--------- MIH_Link_Parameters_Report.indication ----|<--------- MIH_Link_Parameters_Report.indication ----| +// ... ... ... +// etc etc etc + +/////////////////////////////////////////////////////////////////////////////// + +static const char* const kConf_MIH_Commands = "user.commands"; + +/////////////////////////////////////////////////////////////////////////////// + +namespace po = boost::program_options; + +using odtone::uint; +using odtone::ushort; +using odtone::sint8; + +odtone::logger log_("[mih_usr]", std::cout); + +/////////////////////////////////////////////////////////////////////////////// + +// int& operator [](const link_param_lte_enum value) +// { +// static std::map<link_param_lte_enum, std::string> var; +// if (var.size() == 0) +// { +// var[link_param_lte_rsrp]="link_param_lte_rsrp"; +// var[link_param_lte_rsrq]="link_param_lte_rsrq"; +// var[link_param_lte_cqi]="link_param_lte_cqi"; +// var[link_param_lte_bandwidth]="link_param_lte_bandwidth"; +// var[link_param_lte_pkt_delay]="link_param_lte_pkt_delay"; +// } +// return var[value]; +// } + +const char * MeasurementTypes[]= +{ + "LTE_RSRP", /**< RSRP. */ + "LTE_RSRQ", /**< RSRQ. */ + "link_param_lte_cqi", /**< Multicast packet loss rate.*/ + "link_param_lte_bandwidth", /**< System Load. */ + "link_param_lte_pkt_delay", /**< Number of registered users. */ + "link_param_lte_pkt_loss", /**< Number of active users. */ + "link_param_lte_l2_buffer", /**< Congestion windows of users. */ + "link_param_lte_MN_cap", /**< Congestion windows of users. */ + "link_param_lte_embms", /**< Congestion windows of users. */ + "link_param_lte_jumbo_feasibility", /**< Congestion windows of users. */ + "link_param_lte_jumbo_setup", /**< Congestion windows of users. */ + "link_param_lte_active_embms", /**< Transmission rate of users. */ + "link_param_lte_link_congestion", /**< Link congestion. */ +}; + +/////////////////////////////////////////////////////////////////////////////// + +//----------------------------------------------------------------------------- +void __trim(odtone::mih::octet_string &str, const char chr) +//----------------------------------------------------------------------------- +{ + str.erase(std::remove(str.begin(), str.end(), chr), str.end()); +} +//----------------------------------------------------------------------------- +template <class T> std::string StringOf(T object) { +//----------------------------------------------------------------------------- + std::ostringstream os; + os << object; + return(os.str()); +} +//----------------------------------------------------------------------------- +std::string getTimeStamp4Log() +//----------------------------------------------------------------------------- +{ + std::stringstream ss (std::stringstream::in | std::stringstream::out); + struct timespec time_spec; + unsigned int time_now_micros; + unsigned int time_now_s; + clock_gettime (CLOCK_REALTIME, &time_spec); + time_now_s = (unsigned int) time_spec.tv_sec % 3600; + time_now_micros = (unsigned int) time_spec.tv_nsec/1000; + ss << time_now_s << ':' << time_now_micros; + return ss.str(); +} +//----------------------------------------------------------------------------- +std::string status2string(odtone::mih::status statusP){ +//----------------------------------------------------------------------------- + switch (statusP.get()) { + case odtone::mih::status_success: return "SUCCESS";break; + case odtone::mih::status_failure: return "UNSPECIFIED_FAILURE";break; + case odtone::mih::status_rejected: return "REJECTED";break; + case odtone::mih::status_authorization_failure: return "AUTHORIZATION_FAILURE";break; + case odtone::mih::status_network_error: return "NETWORK_ERROR";break; + default: return "UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_down_reason2string(odtone::mih::link_dn_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_dn_reason_explicit_disconnect: return "DN_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_dn_reason_packet_timeout: return "DN_REASON_PACKET_TIMEOUT";break; + case odtone::mih::link_dn_reason_no_resource: return "DN_REASON_NO_RESOURCE";break; + case odtone::mih::link_dn_reason_no_broadcast: return "DN_REASON_NO_BROADCAST";break; + case odtone::mih::link_dn_reason_authentication_failure: return "DN_REASON_AUTHENTICATION_FAILURE";break; + case odtone::mih::link_dn_reason_billing_failure: return "DN_REASON_BILLING_FAILURE";break; + default: return "DN_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string link_going_down_reason2string(odtone::mih::link_gd_reason reasonP){ +//----------------------------------------------------------------------------- + switch (reasonP.get()) { + case odtone::mih::link_gd_reason_explicit_disconnect: return "GD_REASON_EXPLICIT_DISCONNECT";break; + case odtone::mih::link_gd_reason_link_parameter_degrading: return "GD_REASON_PARAMETER_DEGRADING";break; + case odtone::mih::link_gd_reason_low_power: return "GD_REASON_LOW_POWER";break; + case odtone::mih::link_gd_reason_no_resource: return "GD_REASON_NO_RESOURCE";break; + default: return "GD_REASON_UNKNOWN"; + } +} +//----------------------------------------------------------------------------- +std::string evt2string(odtone::mih::mih_evt_list evtP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(evtP.get(odtone::mih::mih_evt_link_detected)) s += "DETECTED "; + if(evtP.get(odtone::mih::mih_evt_link_up)) s += "UP "; + if(evtP.get(odtone::mih::mih_evt_link_down)) s += "DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_parameters_report)) s += "PARAMETERS_REPORT "; + if(evtP.get(odtone::mih::mih_evt_link_going_down)) s += "GOING_DOWN "; + if(evtP.get(odtone::mih::mih_evt_link_handover_imminent)) s += "HANDOVER_IMMINENT "; + if(evtP.get(odtone::mih::mih_evt_link_handover_complete)) s += "HANDOVER_COMPLETE "; + if(evtP.get(odtone::mih::mih_evt_link_pdu_transmit_status)) s += "PDU_TRANSMIT_STATUS "; + return s; +} +//----------------------------------------------------------------------------- +std::string cmd2string(odtone::mih::mih_cmd_list cmdP){ +//----------------------------------------------------------------------------- + std::string s=std::string(" "); + if(cmdP.get(odtone::mih::mih_cmd_link_get_parameters)) s += "Link_Get_Parameters "; + if(cmdP.get(odtone::mih::mih_cmd_link_configure_thresholds)) s += "Link_Configure_Thresholds "; + if(cmdP.get(odtone::mih::mih_cmd_link_actions)) s += "Link_Actions "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_candidate_query)) s += "Net_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_net_ho_commit)) s += "Net_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_query_resources)) s += "N2N_HO_Query_Resources "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_commit)) s += "N2N_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_n2n_ho_complete)) s += "N2N_HO_Complete "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_candidate_query)) s += "MN_HO_Candidate_Query "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_commit)) s += "MN_HO_Commit "; + if(cmdP.get(odtone::mih::mih_cmd_mn_ho_complete)) s += "MN_HO_Complete "; + return s; +} +//----------------------------------------------------------------------------- +std::string link_type2string(const odtone::mih::link_type& lt) +//----------------------------------------------------------------------------- +{ + switch (lt.get()) { + case odtone::mih::link_type_gsm: return "GSM"; break; + case odtone::mih::link_type_gprs: return "GPRS"; break; + case odtone::mih::link_type_edge: return "EDGE"; break; + case odtone::mih::link_type_ethernet: return "Ethernet"; break; + case odtone::mih::link_type_wireless_other: return "Other"; break; + case odtone::mih::link_type_802_11: return "IEEE 802.11"; break; + case odtone::mih::link_type_cdma2000: return "CDMA-2000"; break; + case odtone::mih::link_type_umts: return "UMTS"; break; + case odtone::mih::link_type_cdma2000_hrpd: return "CDMA-2000-HRPD"; break; + case odtone::mih::link_type_lte: return "LTE"; break; + case odtone::mih::link_type_802_16: return "IEEE 802.16"; break; + case odtone::mih::link_type_802_20: return "IEEE 802.20"; break; + case odtone::mih::link_type_802_22: return "IEEE 802.22"; break; + default: break; + } + return "Unknown link type"; +} +//----------------------------------------------------------------------------- +std::string link_addr2string(const odtone::mih::link_addr *addr) +//----------------------------------------------------------------------------- +{ + if (const odtone::mih::mac_addr *la = boost::get<odtone::mih::mac_addr>(addr)) { + return la->address(); + } + else if (const odtone::mih::l2_3gpp_3g_cell_id *la = boost::get<odtone::mih::l2_3gpp_3g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d") % plmn % la->_cell_id); + } + else if (const odtone::mih::l2_3gpp_2g_cell_id *la = boost::get<odtone::mih::l2_3gpp_2g_cell_id>(addr)) { + char plmn[16]; + sprintf(plmn, "%hhx:%hhx:%hhx", la->plmn_id[0], la->plmn_id[1], la->plmn_id[2]); + return str(boost::format("%s %d %d") % plmn % la->_lac % la->_ci); + } + else if (const odtone::mih::l2_3gpp_addr *la = boost::get<odtone::mih::l2_3gpp_addr>(addr)){ + return la->value; + } + else if (const odtone::mih::l2_3gpp2_addr *la = boost::get<odtone::mih::l2_3gpp2_addr>(addr)) { + return la->value; + } + else if (const odtone::mih::other_l2_addr *la = boost::get<odtone::mih::other_l2_addr>(addr)) { + return la->value; + } + return "null"; +} +//----------------------------------------------------------------------------- +std::string l2_3gpp_3g_cell_id2string(odtone::mih::l2_3gpp_3g_cell_id& addr) +//----------------------------------------------------------------------------- +{ + char buffer[256]; + int index = 0; + + index += std::sprintf(&buffer[index], "plmn: -%hhx--%hhx--%hhx-\n", addr.plmn_id[0], addr.plmn_id[1], addr.plmn_id[2]); + index += std::sprintf(&buffer[index], "cell_id: %hhx\n", addr._cell_id); + return buffer; +} +//----------------------------------------------------------------------------- +std::string link_id2string(odtone::mih::link_id linkP) +//----------------------------------------------------------------------------- +{ + std::string s; + s = link_type2string(linkP.type.get()) + " " + link_addr2string(&linkP.addr); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_addr2string(odtone::mih::ip_addr ip_addrP) { +//----------------------------------------------------------------------------- + std::string s; + switch (ip_addrP.type()) { + case odtone::mih::ip_addr::ipv4: s = "ipv4 "; break; + case odtone::mih::ip_addr::ipv6: s = "ipv6 "; break; + default: s = "Unkown type"; + } + s += ip_addrP.address(); + return s; +} +//----------------------------------------------------------------------------- +std::string ip_tuple2string(odtone::mih::ip_tuple ip_tupleP) { +//----------------------------------------------------------------------------- + char buffer[128]; + std::snprintf(buffer, 128, "%s/%d", ip_addr2string(ip_tupleP.ip).c_str(), ip_tupleP.port_val); + return buffer; +} +//----------------------------------------------------------------------------- +std::string ip_proto2string(odtone::mih::proto ip_protoP) { +//----------------------------------------------------------------------------- + switch (ip_protoP.get()) { + case odtone::mih::proto_tcp: return "TCP"; + case odtone::mih::proto_udp: return "UDP"; + default: break; + } + return "Unknown IP protocol"; +} +// TEMP : next 2 functions are commented to restore flow_id as a uint32 +// full structure will be updated later +/*//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::flow_id flowP) { +//----------------------------------------------------------------------------- + std::string s; + odtone::mih::ip_tuple ip; + ip = flowP.src; + s = "SRC = " + ip_tuple2string(flowP.src); + s += ", DST = " + ip_tuple2string(flowP.dst); + s += ", PROTO = " + ip_proto2string(flowP.transport); + return s; +} +//----------------------------------------------------------------------------- +std::string flow_id2string(odtone::mih::link_ac_param link_ac_paramP) { +//----------------------------------------------------------------------------- + if (odtone::mih::resource_desc *res = boost::get<odtone::mih::resource_desc>(&link_ac_paramP.param)) { + return flow_id2string(res->fid); + } + else if (odtone::mih::flow_attribute *flow = boost::get<odtone::mih::flow_attribute>(&link_ac_paramP.param)) { + return flow_id2string(flow->id); + } + return "null"; +}*/ +//----------------------------------------------------------------------------- +std::string link_ac_result2string(odtone::mih::link_ac_result resultP) +//----------------------------------------------------------------------------- +{ + switch (resultP.get()) { + case odtone::mih::link_ac_success: return "SUCCESS"; break; + case odtone::mih::link_ac_failure: return "FAILURE"; break; + case odtone::mih::link_ac_refused: return "REFUSED"; break; + case odtone::mih::link_ac_incapable: return "INCAPABLE"; break; + default: break; + } + return "Unknown action result"; +} +//----------------------------------------------------------------------------- +std::string link_actions_req2string(odtone::mih::link_action_req link_act_reqP) { +//----------------------------------------------------------------------------- + std::string s; + + s = link_id2string(link_act_reqP.id); + + if(link_act_reqP.action.type == odtone::mih::link_ac_type_none) s += ", AC_TYPE_NONE"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_disconnect) s += ", AC_TYPE_DISCONNECT"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_low_power) s += ", AC_TYPE_LOW_POWER"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_down) s += ", AC_TYPE_POWER_DOWN"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_power_up) s += ", AC_TYPE_POWER_UP"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_flow_attr) s += ", AC_TYPE_FLOW_ATTR"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_activate_resources) s += ", AC_TYPE_ACTIVATE_RESOURCES"; + if(link_act_reqP.action.type == odtone::mih::link_ac_type_link_deactivate_resources) s += ", AC_TYPE_DEACTIVATE_RESOURCES"; + + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_data_fwd_req)) s += ", AC_ATTR_DATA_FWD_REQ"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_scan)) s += ", AC_ATTR_SCAN"; + if(link_act_reqP.action.attr.get(odtone::mih::link_ac_attr_res_retain)) s += ", AC_ATTR_RES_RETAIN"; + + s += ", " + StringOf(link_act_reqP.ex_time) + " ms"; + return s; +} +//----------------------------------------------------------------------------- +std::string net_type_addr_list2string(boost::optional<odtone::mih::net_type_addr_list> ntalP) { +//----------------------------------------------------------------------------- + std::string s; + std::ostringstream stream; + odtone::mih::net_type_addr net_type_addr; + + for (odtone::mih::net_type_addr_list::iterator i = ntalP->begin(); i != ntalP->end(); i++) + { + net_type_addr = boost::get<odtone::mih::net_type_addr>(*i); + stream << net_type_addr; + if (i != ntalP->begin()) { + stream << " / "; + } + } + s = stream.str(); + return s; +} + + + +/** + * Parse supported commands. + * + * @param cfg Configuration options. + * @return An optional list of supported commands. + */ +boost::optional<odtone::mih::mih_cmd_list> parse_supported_commands(const odtone::mih::config &cfg) +{ + using namespace boost; + + odtone::mih::mih_cmd_list commands; + + std::map<std::string, odtone::mih::mih_cmd_list_enum> enum_map; + enum_map["mih_link_get_parameters"] = odtone::mih::mih_cmd_link_get_parameters; + enum_map["mih_link_configure_thresholds"] = odtone::mih::mih_cmd_link_configure_thresholds; + enum_map["mih_link_actions"] = odtone::mih::mih_cmd_link_actions; + enum_map["mih_net_ho_candidate_query"] = odtone::mih::mih_cmd_net_ho_candidate_query; + enum_map["mih_net_ho_commit"] = odtone::mih::mih_cmd_net_ho_commit; + enum_map["mih_n2n_ho_query_resources"] = odtone::mih::mih_cmd_n2n_ho_query_resources; + enum_map["mih_n2n_ho_commit"] = odtone::mih::mih_cmd_n2n_ho_commit; + enum_map["mih_n2n_ho_complete"] = odtone::mih::mih_cmd_n2n_ho_complete; + enum_map["mih_mn_ho_candidate_query"] = odtone::mih::mih_cmd_mn_ho_candidate_query; + enum_map["mih_mn_ho_commit"] = odtone::mih::mih_cmd_mn_ho_commit; + enum_map["mih_mn_ho_complete"] = odtone::mih::mih_cmd_mn_ho_complete; + + std::string tmp = cfg.get<std::string>(kConf_MIH_Commands); + __trim(tmp, ' '); + + char_separator<char> sep1(","); + tokenizer< char_separator<char> > list_tokens(tmp, sep1); + + BOOST_FOREACH(std::string str, list_tokens) { + if(enum_map.find(str) != enum_map.end()) { + commands.set((odtone::mih::mih_cmd_list_enum) enum_map[str]); + } + } + + return commands; +} + +/////////////////////////////////////////////////////////////////////////////// +/** + * This class provides an implementation of an IEEE 802.21 MIH-User. + */ +class mih_user : boost::noncopyable { +public: + /** + * Construct the MIH-User. + * + * @param cfg Configuration options. + * @param io The io_service object that the MIH-User will use to + * dispatch handlers for any asynchronous operations performed on the socket. + */ + mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io); + + /** + * Destruct the MIH-User. + */ + ~mih_user(); + +protected: + /** + * User registration handler. + * + * @param cfg Configuration options. + * @param ec Error Code. + */ + void user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec); + /** + * Default MIH event handler. + * + * @param msg Received event notification. + * @param ec Error code. + */ + void event_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + /** + * MIH receive message handler. + * + * @param msg Received message. + * @param ec Error code. + */ + void receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec); + + void send_MIH_User_Register_indication(const odtone::mih::config& cfg); + + void send_MIH_Capability_Discover_request(void); + void send_MIH_Capability_Discover_request_remote(void); + void receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest); + void receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Event_Unsubscribe_request(void); + void send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt); + void receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg); + + void send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type); + void receive_MIH_Link_Actions_confirm(odtone::mih::message& msg); + void receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec); + void forward_Parameters_Report_indication(odtone::mih::message& msg); + + void send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec); + void receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec); + +private: + odtone::sap::user _mihf; /**< User SAP helper. */ + odtone::mih::id _mihfid; /**< MIHF destination ID. */ + odtone::mih::id _mihuserid; /**< MIH_USER ID. */ + + odtone::mih::ip_addr _mihf_ip; /**< MIHF IP address */ + odtone::mih::port _mihf_lport; /**< MIHF local port number */ + + odtone::mih::link_id_list _link_id_list; /**< List of network link identifiers */ + odtone::mih::mih_evt_list _subs_evt_list; /**< List of subscribed link events */ + + odtone::mih::link_ac_type _last_link_action_type; + odtone::uint _current_link_action_request, _nb_of_link_action_requests; + odtone::uint link_threshold_request, link_measures_request, link_measures_counter; + odtone::mih::link_id rcv_link_id; + + static const odtone::uint _max_link_action_requests = 4; + odtone::uint _num_thresholds_request; + + void receive_MIH_Link_Detected_indication(odtone::mih::message& msg); + void send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link); + void receive_MIH_Link_Up_indication(odtone::mih::message& msg); + + void receive_MIH_Link_Down_indication(odtone::mih::message& msg); + void receive_MIH_MN_HO_Candidate_Query_request(odtone::mih::message& msg); + + void receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg); + +}; + +//----------------------------------------------------------------------------- +mih_user::mih_user(const odtone::mih::config& cfg, boost::asio::io_service& io) + : _mihf(cfg, io, boost::bind(&mih_user::event_handler, this, _1, _2)), + _last_link_action_type(odtone::mih::link_ac_type_none), + _current_link_action_request(0), _nb_of_link_action_requests(NB_OF_RESOURCES), _num_thresholds_request(0) +//----------------------------------------------------------------------------- +{ + odtone::mih::octet_string user_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_id); + _mihuserid.assign(user_id.c_str()); + + odtone::mih::octet_string dest_id = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIH_SAP_dest); + _mihfid.assign(dest_id.c_str()); + + odtone::mih::octet_string src = cfg.get<odtone::mih::octet_string>(odtone::sap::kConf_MIHF_Ip); + boost::asio::ip::address ip = boost::asio::ip::address::from_string(src); + if (ip.is_v4()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv4, src); + _mihf_ip = ip_addr; + } else if (ip.is_v6()) { + odtone::mih::ip_addr ip_addr(odtone::mih::ip_addr::ipv6, src); + _mihf_ip = ip_addr; + } + + _mihf_lport = cfg.get<odtone::mih::port>(odtone::sap::kConf_MIHF_Local_Port); + + //_nb_of_link_action_requests = NB_OF_RESOURCES; + if (_nb_of_link_action_requests > _max_link_action_requests) { + _nb_of_link_action_requests = _max_link_action_requests; + } + + _link_id_list.clear(); + _subs_evt_list.clear(); + link_threshold_request = 0; + link_measures_request =0; + link_measures_counter =0; + log_(0, "[MSC_NEW]["+getTimeStamp4Log()+"][MIH-USER="+_mihuserid.to_string()+"]\n"); + + // Send MEDIEVAL specific MIH_User_Register.indication message to the MIH-F + mih_user::send_MIH_User_Register_indication(cfg); +} + +//----------------------------------------------------------------------------- +mih_user::~mih_user() +//----------------------------------------------------------------------------- +{ +} + +//----------------------------------------------------------------------------- +void mih_user::user_reg_handler(const odtone::mih::config& cfg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH-User register result: ", ec.message(), "\n"); + + // + // Let's fire a capability discover request to get things moving + // + mih_user::send_MIH_Capability_Discover_request(); + //send a capability discover request to the remote eNB + mih_user::send_MIH_Capability_Discover_request_remote(); + +} + + +//----------------------------------------------------------------------------- +void mih_user::event_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + case odtone::mih::indication::link_detected: + mih_user::receive_MIH_Link_Detected_indication(msg); + break; + + case odtone::mih::indication::link_up: + mih_user::receive_MIH_Link_Up_indication(msg); + if (_num_thresholds_request == 0) { + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + _num_thresholds_request += 1;; + } + break; + + case odtone::mih::indication::mn_ho_candidate_query: + log_(0, "MIH-User has received a request \"mn_ho_candidate_query\"", msg.mid()); + mih_user::receive_MIH_MN_HO_Candidate_Query_request(msg); + break; + + case odtone::mih::indication::link_down: + mih_user::receive_MIH_Link_Down_indication(msg); + break; + + case odtone::mih::indication::link_going_down: + mih_user::receive_MIH_Link_Going_Down_indication(msg); + break; + + case odtone::mih::indication::link_handover_imminent: + log_(0, "MIH-User has received a local event \"link_handover_imminent\""); + break; + + case odtone::mih::indication::link_handover_complete: + log_(0, "MIH-User has received a local event \"link_handover_complete\""); + break; + + case odtone::mih::indication::link_parameters_report: + //log_(0, "MIH-User has received a local event \"link_parameters_report\""); + mih_user::receive_MIH_Link_Parameters_Report(msg, ec); + /*if (link_threshold_request == 0){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + link_threshold_request =1; + } else if (link_threshold_request == 1){ + link_measures_counter ++; + // Stop measures after 5 reports + if (link_measures_counter == NUM_PARM_REPORT){ + mih_user::send_MIH_Link_Configure_Thresholds_request(msg, ec); + } + }*/ + break; + + case odtone::mih::indication::link_pdu_transmit_status: + log_(0, "MIH-User has received a local event \"link_pdu_transmit_status\""); + break; + + case odtone::mih::confirm::link_configure_thresholds: + mih_user::receive_MIH_Link_Configure_Thresholds_confirm(msg, ec); + break; + + default: + log_(0, "MIH-User has received UNKNOWN local event", msg.mid()); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_handler(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + if (ec) { + log_(0, __FUNCTION__, " error: ", ec.message()); + return; + } + + switch (msg.mid()) { + + case odtone::mih::confirm::capability_discover: + mih_user::receive_MIH_Capability_Discover_confirm(msg); + break; + + case odtone::mih::confirm::event_subscribe: + mih_user::receive_MIH_Event_Subscribe_confirm(msg); + break; + + case odtone::mih::confirm::event_unsubscribe: + mih_user::receive_MIH_Event_Unsubscribe_confirm(msg); + break; + + case odtone::mih::confirm::link_actions: + mih_user::receive_MIH_Link_Actions_confirm(msg); + break; + + default: + log_(0, "MIH-User has received UNKNOWN message (", msg.mid(), ")\n"); + break; + } +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Detected_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Detected.indication - RECEIVED - Begin\n"); + odtone::mih::link_det_info ldi; + odtone::mih::link_det_info_list ldil; + odtone::mih::link_det_info_list::iterator it_ldil; + odtone::mih::link_id lid; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_det_info_list(ldil); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Detected.indication --->]["+msg.destination().to_string()+"]\n"); + for(it_ldil = ldil.begin(); it_ldil != ldil.end(); it_ldil++) { + ldi = *it_ldil; + log_(0, "\tMIH_Link_Detected.indication - network_id:........", ldi.network_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - net_aux_id:........", ldi.net_aux_id.c_str()); + log_(0, "\tMIH_Link_Detected.indication - sig_strength:......TO DO");//, ldi.signal); + log_(0, "\tMIH_Link_Detected.indication - sinr:..............", ldi.sinr); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - data_rate:.........", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - mih_capabilities:..", ldi.data_rate); + log_(0, "\tMIH_Link_Detected.indication - net_capabilities:..TO DO");//, ldi.net_capabilities); + + } + // Display message parameters + // TODO: for each link_det_info in the list {display LINK_DET_INFO} + + // send Link_Action / Power Up + lid.type = odtone::mih::link_type_lte; + lid.addr = ldi.id.addr; + send_MIH_Link_Action_Power_Up_plus_scan_request(lid); + log_(0, "MIH_Link_Detected.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(const odtone::mih::link_id& link) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + //struct null n; + + link_act_req.id = link; + link_act_req.action.type = odtone::mih::link_ac_type_power_up; + link_act_req.action.attr.clear(); + link_act_req.action.attr.set(odtone::mih::link_ac_attr_scan); + + link_act_req.ex_time = 5000; // in ms + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Action_Power_Up_request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Up_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Up.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; +// odtone::mih::tlv_old_access_router oldAR; + + msg >> odtone::mih::indication() & odtone::mih::tlv_link_identifier(link); +// & odtone::mih::tlv_old_access_router(oar); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Up.indication --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); + + log_(0, "MIH_Link_Up.indication - End\n"); +} + + +//MIH_MN_HO_Candidate_Query_request used to forward the parameters report from eNB1 to eNB2 +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_MN_HO_Candidate_Query_request(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::request() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "MIH_MN_HO_Candidate_Query.request - RECEIVED - Begin\n"); + + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + +for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value /*- 65536*/); + } + + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); + + //forward the Link parameters report to eNB2 + forward_Parameters_Report_indication(msg); +} + + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::link_addr> addr; + odtone::mih::link_dn_reason ldr; + + log_(0, "MIH_Link_Down.indication - RECEIVED - Begin\n"); + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_old_access_router(addr) + & odtone::mih::tlv_link_dn_reason(ldr); + +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Down.indication\\n"+link_down_reason2string(ldr).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + //Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str(), "\n"); +// log_(0, " - LINK_DN_REASON - Link down reason: ", link_down_reason2string(ldr).c_str(), "\n"); + + log_(0, "MIH_Link_Down.indication - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Going_Down_indication(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Going_Down.indication - RECEIVED - Begin\n"); + + odtone::mih::link_tuple_id link; + odtone::mih::link_gd_reason lgd; + odtone::mih::link_ac_ex_time ex_time; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_time_interval(ex_time) + & odtone::mih::tlv_link_gd_reason(lgd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Going_Down.indication\\n"+link_going_down_reason2string(lgd).c_str()+" --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - Time Interval:", (ex_time/256)); + log_(0, " - LINK_GD_REASON - Link going down reason: ", link_going_down_reason2string(lgd).c_str(), "\n"); + + log_(0, "MIH_Link_Going_Down.indication - End\n"); +} + + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_User_Register_indication(const odtone::mih::config& cfg) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + boost::optional<odtone::mih::mih_cmd_list> supp_cmd = parse_supported_commands(cfg); + + m << odtone::mih::indication(odtone::mih::indication::user_register) + & odtone::mih::tlv_command_list(supp_cmd); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_User_Register.indication --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::user_reg_handler, this, boost::cref(cfg), _2)); + + log_(0, "MIH_User_Register.indication - SENT (towards its local MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Capability_Discover_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards its local MIHF)\n"); +} + +void mih_user::send_MIH_Capability_Discover_request_remote(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + m << odtone::mih::request(odtone::mih::request::capability_discover); + m.source(_mihuserid); + odtone::mih::id mid_enb; + mid_enb.assign("mihf4_enb"); + m.destination(mid_enb); +// m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Capability_Discover.request --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH_Capability_Discover.request - SENT (towards the remote MIHF)\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Capability_Discover_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Capability_Discover.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::net_type_addr_list> ntal; + boost::optional<odtone::mih::mih_evt_list> evt; + boost::optional<odtone::mih::mih_cmd_list> cmd; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_net_type_addr_list(ntal) + & odtone::mih::tlv_event_list(evt) + & odtone::mih::tlv_command_list(cmd); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Capability_Discover.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + "\\nNet type addr list=" + net_type_addr_list2string(ntal).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + if (cmd) { + log_(0, " - MIH_CMD_LIST - Command List: ", cmd2string(cmd.get()).c_str()); + } + if (ntal) { + log_(0, " - LIST(NET_TYPE_ADDR) - Network Types and Link Address: ", net_type_addr_list2string(ntal).c_str()); + //Store link address + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) + { + rcv_link_id.addr = i->addr; + rcv_link_id.type = boost::get<odtone::mih::link_type>(i->nettype.link); + } + } + log_(0, ""); + + // + // event subscription + // + // For every interface the MIHF sent in the + // Capability_Discover.response send an Event_Subscribe.request + // for all availabe events + // + if (ntal && evt) { + _subs_evt_list = evt.get(); // save the list of subscribed link events + for (odtone::mih::net_type_addr_list::iterator i = ntal->begin(); i != ntal->end(); i++) { + if (i->nettype.link.which() == 1) + { + odtone::mih::link_tuple_id li; + + li.addr = i->addr; + li.type = boost::get<odtone::mih::link_type>(i->nettype.link); + _link_id_list.push_back(li); // save the link identifier of the network interface + + mih_user::send_MIH_Event_Subscribe_request(li, evt.get(), msg.source()); + } + } + } + + log_(0, "MIH_Capability_Discover.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Subscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt, odtone::mih::id dest) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_subscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(dest); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Subscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Subscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Subscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Subscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Subscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Subscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + //mih_user::send_MIH_Link_Actions_request(link, odtone::mih::link_ac_type_link_activate_resources); + //log_(0, "TEMP : Resource scenario deactivated\n"); + + log_(0, "MIH_Event_Subscribe.confirm - End\n"); +// + + if (link.type == odtone::mih::link_type_lte) + mih_user::send_MIH_Link_Action_Power_Up_plus_scan_request(link); + +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(void) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id li; + + // For every interface the MIH user received in the + // Capability_Discover.confirm, send an Event_Unsubscribe.request + // for all subscribed events + for (odtone::mih::link_id_list::iterator i = _link_id_list.begin(); i != _link_id_list.end(); i++) { + li.type = i->type; + li.addr = i->addr; + mih_user::send_MIH_Event_Unsubscribe_request(li, _subs_evt_list); + } +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Event_Unsubscribe_request(odtone::mih::link_tuple_id& li, odtone::mih::mih_evt_list& evt) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + + m << odtone::mih::request(odtone::mih::request::event_unsubscribe) + & odtone::mih::tlv_link_identifier(li) + & odtone::mih::tlv_event_list(evt); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Event_Unsubscribe.request"+ + "\\nLink="+link_id2string(li).c_str()+ + "\\nEvent list="+evt2string(evt).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Event_Unsubscribe.request to ", m.destination().to_string()); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(li).c_str()); + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt).c_str(), "\n"); + + log_(0, "MIH_Event_Unsubscribe.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Event_Unsubscribe_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Event_Unsubscribe.confirm(", msg.tid(), ") - RECEIVED - Begin\n"); + + odtone::mih::status st; + odtone::mih::link_tuple_id link; + boost::optional<odtone::mih::mih_evt_list> evt; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_event_list(evt); + + if (evt) { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + "\\nEvent list="+evt2string(evt.get()).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } else { + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Event_Unsubscribe.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + "\\nLink="+link_id2string(link).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + } + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + if (evt) { + log_(0, " - MIH_EVT_LIST - Event List: ", evt2string(evt.get()).c_str()); + } + log_(0, ""); + + log_(0, "MIH_Event_Unsubscribe.confirm - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Actions_request(const odtone::mih::link_id& link, odtone::mih::link_ac_type type) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::link_action_list lal; + odtone::mih::link_action_req link_act_req; + + link_act_req.id = link; + link_act_req.action.type = type; + + _last_link_action_type = type; + + // Initialize resource parameters + odtone::mih::resource_desc res; + + res.lid = link; // Link identifier + res.data_rate = 128000; // bit rate + res.jumbo = false; // jumbo disable + res.multicast = false; // multicast disable + + odtone::mih::qos qos; // Class Of Service + qos.value = 56; + res.qos_val = qos; + res.fid = 555 + _current_link_action_request; + +// // Flow identifier +// res.fid.src.ip = _mihf_ip; +// res.fid.src.port_val = _mihf_lport; +// +// if (mih_user::_current_link_action_request == 0) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9150"); +// } +// else if (mih_user::_current_link_action_request == 1) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9151"); +// } +// else if (mih_user::_current_link_action_request == 2) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "FF3E:0020:2001:0DB8:0000:0000:0000:0043"); +// res.multicast = true; +// } +// else if (mih_user::_current_link_action_request == 3) { +// res.fid.dst.ip = odtone::mih::ip_addr(odtone::mih::ip_addr::ipv6, +// "2001:0660:0382:0014:0335:0600:8014:9153"); +// } +// res.fid.dst.port_val = 1235; // DUMMY +// res.fid.transport = odtone::mih::proto_udp; + + link_act_req.action.param.param = res; + + link_act_req.ex_time = 0; + + lal.push_back(link_act_req); + + m << odtone::mih::request(odtone::mih::request::link_actions) + & odtone::mih::tlv_link_action_list(lal); + m.source(_mihuserid); + m.destination(_mihfid); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ m.source().to_string() +"][--- MIH_Link_Actions.request\\n"+link_actions_req2string(link_act_req)+" --->]["+m.destination().to_string()+"]\n"); + + _mihf.async_send(m, boost::bind(&mih_user::receive_handler, this, _1, _2)); + + log_(0, "MIH-User has sent Link_Actions.request to ", m.destination().to_string()); + log_(0, " - LINK_ID - Link identifier: ", link_id2string(link).c_str()); + log_(0, " - FLOW_ID - Flow identifier: ", res.fid); +//TEMP log_(0, " - FLOW_ID - Flow identifier: ", flow_id2string(link_act_req.action.param).c_str()); + log_(0, " - LINK_ACTIONS - Link Actions: " + link_actions_req2string(link_act_req) + "\n"); + + log_(0, "MIH_Link_Actions.request - SENT\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Actions_confirm(odtone::mih::message& msg) +//----------------------------------------------------------------------------- +{ + log_(0, "MIH_Link_Actions.confirm - RECEIVED - Begin\n"); + + odtone::mih::status st; + boost::optional<odtone::mih::link_action_rsp_list> larl; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_action_rsp_list(larl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Actions.confirm"+ + "\\nstatus="+status2string(st).c_str()+ + " --->]["+msg.destination().to_string()+"]\n"); + + // Display message parameters + log_(0, " - STATUS: ", status2string(st).c_str(), " (", st.get(), ")"); + if (larl) { + log_(0, " - LINK ACTION RSP LIST - Length:", larl.get().size()); + for (odtone::mih::link_action_rsp_list::iterator i = larl->begin(); i != larl->end(); i++) + { + log_(0, "\tLINK_ID: ", link_id2string(i->id).c_str(), + ", LINK_AC_RESULT: ", link_ac_result2string(i->result).c_str()); + } + } + log_(0, ""); + + // 1st scenario: Sequentially activate and deactivate each resource +#ifdef SCENARIO_1 + if (larl){ + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + if (rsp->result.get() == odtone::mih::link_ac_success) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + _current_link_action_request += 1; + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_1 + +#ifdef SCENARIO_2 + // 2nd scenario: Activate all resources, then deactivate all resources + if (larl.get().size() > 0) { + odtone::mih::link_action_rsp *rsp = &larl->front(); + if (++_current_link_action_request < _nb_of_link_action_requests) { + if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_activate_resources); + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_deactivate_resources) { + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + } + else if (_last_link_action_type == odtone::mih::link_ac_type_link_activate_resources) { + _current_link_action_request = 0; + mih_user::send_MIH_Link_Actions_request(rsp->id, odtone::mih::link_ac_type_link_deactivate_resources); + } + else { // Ends the scenario + mih_user::send_MIH_Event_Unsubscribe_request(); + } + } +#endif // SCENARIO_2 + + log_(0, "MIH_Link_Actions.confirm - End\n"); +} + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Parameters_Report(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + msg >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, ""); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lprl.size()); + log_(0, "MIH_Link_Parameters_Report.indication - RECEIVED - Begin"); + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Parameters_Report.indication --->]["+msg.destination().to_string()+"]\n"); + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(link).c_str()); + for (odtone::mih::link_param_rpt_list::iterator i=lprl.begin(); i!=lprl.end(); i++) + { + log_(0, "Meausrement Type: --- 0 => RSRP ----- 1=>RSRQ ", i->param.type); + if(odtone::mih::link_param_val *value = boost::get<odtone::mih::link_param_val>(&i->param.value)) + { + log_(0, "Meausrement Value: ", (short) *value /*- 65536*/); + } + } + log_(0, "MIH_Link_Parameters_Report.indication - End"); +} + +void mih_user::forward_Parameters_Report_indication(odtone::mih::message& m) +{ + + odtone::mih::link_tuple_id link; + odtone::mih::link_param_rpt_list lprl; + + log_(0, "Forward_Parameters_Report.indication - Start"); + + m >> odtone::mih::indication() + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + odtone::mih::message msg; + msg.source(_mihuserid); + odtone::mih::id mid_enb; + mid_enb.assign("mihf4_enb"); + msg.destination(mid_enb); + + + msg << odtone::mih::request(odtone::mih::request::mn_ho_candidate_query) + & odtone::mih::tlv_link_identifier(link) + & odtone::mih::tlv_link_param_rpt_list(lprl); + + log_(0, "[MSC_MSG] ["+getTimeStamp4Log()+"]["+msg.source().to_string()+"][-------- Forward MIH_User_Parameters.indication ---->]["+msg.destination().to_string()+"]\n"); + + _mihf.async_send(msg, boost::bind(&mih_user::event_handler, this, _1, _2)); + log_(0, "Forward_Parameters_Report.indication - End"); +} + +//----------------------------------------------------------------------------- +void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + odtone::mih::message m; + odtone::mih::threshold th1; + std::vector<odtone::mih::threshold> thl1; + odtone::mih::threshold th2; + std::vector<odtone::mih::threshold> thl2; + odtone::mih::threshold th3; + std::vector<odtone::mih::threshold> thl3; + odtone::mih::link_tuple_id lti; + odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; + //List of the link threshold parameters + odtone::mih::link_cfg_param_list lcpl; + odtone::mih::link_cfg_param lcp1; + odtone::mih::link_cfg_param lcp2; + odtone::mih::link_cfg_param lcp3; + odtone::mih::link_param_lte lp1; + odtone::mih::link_param_lte lp2; + odtone::mih::link_param_lte lp3; + //odtone::mih::link_param_gen lp; + + odtone::mih::link_param_type typr; + + log_(0,""); + log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); + + //link_tuple_id + lti.type = rcv_link_id.type; + lti.addr = rcv_link_id.addr; + + //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); + + //link_param_gen_data_rate = 0, /**< Data rate. */ + //link_param_gen_signal_strength = 1, /**< Signal strength. */ + //link_param_gen_sinr = 2, /**< SINR. */ + //link_param_gen_throughput = 3, /**< Throughput. */ + //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ + //lp = odtone::mih::link_param_lte_bandwidth; + lp1 = odtone::mih::link_param_lte_rsrp; + lp2 = odtone::mih::link_param_lte_rsrq; + lp3 = odtone::mih::link_param_lte_cqi; + lcp1.type = lp1; + lcp2.type = lp2; + lcp3.type = lp3; + + link_measures_request = 0; + if (link_measures_request ==0){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 3000; + lcp2.timer_interval = 3000; + lcp3.timer_interval = 3000; + //th_action_normal = 0, /**< Set normal threshold. */ + //th_action_one_shot = 1, /**< Set one-shot threshold. */ + //th_action_cancel = 2 /**< Cancel threshold. */ + lcp1.action = odtone::mih::th_action_normal; + lcp2.action = odtone::mih::th_action_normal; + lcp3.action = odtone::mih::th_action_normal; + link_measures_request = 1; + } else if ( link_measures_request==1){ + // Set Timer Interval (in ms) + lcp1.timer_interval = 0; + lcp2.timer_interval = 0; + lcp3.timer_interval = 0; + lcp1.action = odtone::mih::th_action_cancel; + lcp2.action = odtone::mih::th_action_cancel; + lcp3.action = odtone::mih::th_action_cancel; + link_measures_request = 0; + } + + //above_threshold = 0, /**< Above threshold. */ + //below_threshold = 1, /**< Below threshold. */ + th1.threshold_val = -105; + th1.threshold_x_dir = odtone::mih::threshold::above_threshold; + th2.threshold_val = -19; + th2.threshold_x_dir = odtone::mih::threshold::above_threshold; + th3.threshold_val = 0; + th3.threshold_x_dir = odtone::mih::threshold::above_threshold; + + thl1.push_back(th1); + thl2.push_back(th2); + thl3.push_back(th3); + lcp1.threshold_list = thl1; + lcp2.threshold_list = thl2; + lcp3.threshold_list = thl3; + + lcpl.push_back(lcp1); + lcpl.push_back(lcp2); + lcpl.push_back(lcp3); + + m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_param_list(lcpl); + + m.destination(msg.source()); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ + // link_tupple_id2string(lti).c_str() + + link_id2string(lti).c_str()+ + " --->]["+m.destination().to_string()+"]\n"); + _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); + + + log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); + + log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); + + //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} + //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} + //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} + //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} + //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} + if(lp1 == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} + if(lp2 == odtone::mih::link_param_lte_rsrq) {log_(0, "\t LTE link parameter LTE RSRQ");} + log_(0, "\t- TIMER INTERVAL - Value: ", lcp1.timer_interval); + + if(lcp1.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} + if(lcp1.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} + if(lcp1.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} + + log_(0, "\t Threshold value: ", (short) th1.threshold_val); + + if(th1.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(th1.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + + log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +} + +// //----------------------------------------------------------------------------- +// void mih_user::send_MIH_Link_Configure_Thresholds_request(odtone::mih::message& msg, const boost::system::error_code& ec) +// //----------------------------------------------------------------------------- +// { +// odtone::mih::message m; +// odtone::mih::threshold th; +// std::vector<odtone::mih::threshold> thl; +// odtone::mih::link_tuple_id lti; +// odtone::mih::l2_3gpp_addr local_l2_3gpp_addr; +// //List of the link threshold parameters +// odtone::mih::link_cfg_param_list lcpl; +// odtone::mih::link_cfg_param lcp; +// odtone::mih::link_param_lte lp; +// //odtone::mih::link_param_gen lp; +// +// odtone::mih::link_param_type typr; +// +// log_(0,""); +// log_(0, "send_MIH_Link_Configure_Thresholds_request - Begin"); +// +// //link_tuple_id +// lti.type = rcv_link_id.type; +// lti.addr = rcv_link_id.addr; +// +// //local_l2_3gpp_addr = boost::get<odtone::mih::l2_3gpp_addr>(lti.addr); +// +// +// //link_param_gen_data_rate = 0, /**< Data rate. */ +// //link_param_gen_signal_strength = 1, /**< Signal strength. */ +// //link_param_gen_sinr = 2, /**< SINR. */ +// //link_param_gen_throughput = 3, /**< Throughput. */ +// //link_param_gen_packet_error_rate = 4, /**< Packet error rate. */ +// //lp = odtone::mih::link_param_lte_bandwidth; +// lp = odtone::mih::link_param_lte_rsrp; +// lcp.type = lp; +// +// link_measures_request = 0; +// if ( link_measures_request ==0){ +// // Set Timer Interval (in ms) +// lcp.timer_interval = 3000; +// //th_action_normal = 0, /**< Set normal threshold. */ +// //th_action_one_shot = 1, /**< Set one-shot threshold. */ +// //th_action_cancel = 2 /**< Cancel threshold. */ +// lcp.action = odtone::mih::th_action_normal; +// link_measures_request = 1; +// } else if ( link_measures_request==1){ +// // Set Timer Interval (in ms) +// lcp.timer_interval = 0; +// lcp.action = odtone::mih::th_action_cancel; +// link_measures_request = 0; +// } +// +// //above_threshold = 0, /**< Above threshold. */ +// //below_threshold = 1, /**< Below threshold. */ +// th.threshold_val = -105; +// th.threshold_x_dir = odtone::mih::threshold::above_threshold; +// +// thl.push_back(th); +// lcp.threshold_list = thl; +// lcpl.push_back(lcp); +// +// m << odtone::mih::request(odtone::mih::request::link_configure_thresholds) +// & odtone::mih::tlv_link_identifier(lti) +// & odtone::mih::tlv_link_cfg_param_list(lcpl); +// +// m.destination(msg.source()); +// +// log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ _mihuserid.to_string() +"][--- MIH_Link_Configure_Thresholds.request\\nlink="+ +// // link_tupple_id2string(lti).c_str() + +// link_id2string(lti).c_str()+ +// " --->]["+_mihfid.to_string()+"]\n"); +// +// _mihf.async_send(m, boost::bind(&mih_user::event_handler, this, _1, _2)); +// +// log_(0, " - LINK_TUPLE_ID - Link identifier: ", link_id2string(lti).c_str()); +// +// log_(0, "\t- LINK CFG PARAM LIST - Length: ", lcpl.size()); +// +// //if(lp == odtone::mih::link_param_gen_data_rate) {log_(0, "\t Generic link parameter DATA RATE ");} +// //if(lp == odtone::mih::link_param_gen_signal_strength) {log_(0, "\t Generic link parameter SIGNAL STRENGTH");} +// //if(lp == odtone::mih::link_param_gen_sinr) {log_(0, "\t Generic link parameter SINR");} +// //if(lp == odtone::mih::link_param_gen_throughput) {log_(0, "\t Generic link parameter THROUGHPUT");} +// //if(lp == odtone::mih::link_param_lte_bandwidth) {log_(0, "\t LTE link parameter BANDWIDTH");} +// if(lp == odtone::mih::link_param_lte_rsrp) {log_(0, "\t LTE link parameter LTE RSRP");} +// +// log_(0, "\t- TIMER INTERVAL - Value: ", lcp.timer_interval); +// +// if(lcp.action == odtone::mih::th_action_normal) {log_(0, "\t Normal Threshold");} +// if(lcp.action == odtone::mih::th_action_one_shot) {log_(0, "\t One Shot Threshold");} +// if(lcp.action == odtone::mih::th_action_cancel) {log_(0, "\t Threshold to be canceled");} +// +// log_(0, "\t Threshold value: ", th.threshold_val); +// +// if(th.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} +// if(th.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} +// +// log_(0, "send_MIH_Link_Configure_Thresholds_request - End"); +// } + +//----------------------------------------------------------------------------- +void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +//----------------------------------------------------------------------------- +{ + log_(0, ""); + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); + + odtone::uint iter; + odtone::mih::status st; + + //boost::optional<odtone::mih::link_cfg_status_list> lcsl; + odtone::mih::link_cfg_status_list lcsl; + odtone::mih::link_cfg_status lcp; + odtone::mih::link_param_gen lp; + + odtone::mih::link_tuple_id lti; + + msg >> odtone::mih::confirm() + & odtone::mih::tlv_status(st) + & odtone::mih::tlv_link_identifier(lti) + & odtone::mih::tlv_link_cfg_status_list(lcsl); + + log_(0, "[MSC_MSG]["+getTimeStamp4Log()+"]["+ msg.source().to_string() +"][--- MIH_Link_Configure_Thresholds.confirm\\nstatus="+status2string(st.get()).c_str()+" --->]["+_mihuserid.to_string()+"]\n"); + log_(0, "\t- STATUS: ", status2string(st.get()), " " ,st.get()); + + log_(0, "\t- LINK CFG STATUS LIST - Length: ", lcsl.size()); + + for(iter=0; iter<lcsl.size(); iter++) + { + log_(0, "\t Link Param Type: ", lcsl[iter].type); + log_(0, "\t Threshold Val: ", ((short) lcsl[iter].thold.threshold_val)); + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::below_threshold) {log_(0, "\t Threshold direction BELOW");} + if(lcsl[iter].thold.threshold_x_dir == odtone::mih::threshold::above_threshold) {log_(0, "\t Threshold direction ABOVE");} + if(lcsl[iter].status == odtone::mih::status_success){log_(0, "\t Config Status: Success ");} + else {log_(0, "\t Config Status: ", lcsl[iter].status);} + } + + log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); + log_(0,""); +} + +// //----------------------------------------------------------------------------- +// void mih_user::receive_MIH_Link_Configure_Thresholds_confirm(odtone::mih::message& msg, const boost::system::error_code& ec) +// //----------------------------------------------------------------------------- +// { +// log_(0, ""); +// log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - Begin"); +// +// // T odtone::uint iter; +// // T odtone::mih::status st; +// +// //boost::optional<odtone::mih::link_cfg_status_list> lcsl; +// // Todtone::mih::link_cfg_status_list lcsl; +// // Todtone::mih::link_cfg_status lcp; +// //odtone::mih::link_param_gen lp; +// +// // T odtone::mih::link_tuple_id lti; +// +// //msg >> odtone::mih::confirm() +// // & odtone::mih::tlv_status(st) +// // & odtone::mih::tlv_link_identifier(lti) +// // & odtone::mih::tlv_link_cfg_status_list(lcsl); +// +// +// log_(0, "receive_MIH_Link_Configure_Thresholds_confirm - End"); +// log_(0,""); +// } + +//----------------------------------------------------------------------------- +int main(int argc, char** argv) +//----------------------------------------------------------------------------- +{ + odtone::setup_crash_handler(); + + try { + boost::asio::io_service ios; + + // declare MIH Usr available options + po::options_description desc(odtone::mih::octet_string("MIH Usr Configuration")); + desc.add_options() + ("help", "Display configuration options") + (odtone::sap::kConf_File, po::value<std::string>()->default_value("ue2_user.conf"), "Configuration file") + (odtone::sap::kConf_Receive_Buffer_Len, po::value<uint>()->default_value(4096), "Receive buffer length") + (odtone::sap::kConf_Port, po::value<ushort>()->default_value(1635), "Listening port") + (odtone::sap::kConf_MIH_SAP_id, po::value<std::string>()->default_value("user"), "MIH-User ID") + (kConf_MIH_Commands, po::value<std::string>()->default_value(""), "MIH-User supported commands") + (odtone::sap::kConf_MIHF_Ip, po::value<std::string>()->default_value("127.0.0.1"), "Local MIHF IP address") + (odtone::sap::kConf_MIHF_Local_Port, po::value<ushort>()->default_value(1025), "Local MIHF communication port") + (odtone::sap::kConf_MIH_SAP_dest, po::value<std::string>()->default_value("mihf3_ue"), "MIHF destination"); + + odtone::mih::config cfg(desc); + cfg.parse(argc, argv, odtone::sap::kConf_File); + + if (cfg.help()) { + std::cerr << desc << std::endl; + return EXIT_SUCCESS; + } + + mih_user usr(cfg, ios); + + ios.run(); + + } catch(std::exception& e) { + log_(0, "exception: ", e.what()); + } +} + +// EOF //////////////////////////////////////////////////////////////////////// + diff --git a/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/oai_conf/start_ue.bash b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/oai_conf/start_ue.bash new file mode 100644 index 0000000000000000000000000000000000000000..0052bd3352ff1979b5bf125ca5d1382af56c968e --- /dev/null +++ b/targets/PROJECTS/SPECTRA/DEMO_SPECTRA/spectra_demo_src/ue2/oai_conf/start_ue.bash @@ -0,0 +1,208 @@ +#!/bin/bash +################################################################################ +# OpenAirInterface +# Copyright(c) 1999 - 2014 Eurecom +# +# OpenAirInterface is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# +# OpenAirInterface is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with OpenAirInterface.The full GNU General Public License is +# included in this distribution in the file called "COPYING". If not, +# see <http://www.gnu.org/licenses/>. +# +# Contact Information +# OpenAirInterface Admin: openair_admin@eurecom.fr +# OpenAirInterface Tech : openair_tech@eurecom.fr +# OpenAirInterface Dev : openair4g-devel@eurecom.fr +# +# Address : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE +# +################################################################################ +# file start_ue.bash +# brief +# author Lionel Gauthier +# company Eurecom +# email: lionel.gauthier@eurecom.fr +# +#------------------------------------------------ +# OAI NETWORKING +#------------------------------------------------ +declare -x EMULATION_DEV_INTERFACE="eth5" +declare -x EMULATION_DEV_ADDRESS="192.168.15.5" + +declare -x IP_DRIVER_NAME="oai_nw_drv" +declare -x LTEIF="oai0" +declare -x UE_IPv4="10.0.2.3" +declare -x UE_IPv6="2001:2::3" +declare -x UE_IPv6_CIDR=$UE_IPv6"/64" +declare -x UE_IPv4_CIDR=$UE_IPv4"/24" +declare -a NAS_IMEI=( 3 9 1 8 3 6 7 3 0 2 0 0 0 0 ) +declare -x IP_DEFAULT_MARK="1" # originally 3 + +#------------------------------------------------ +# OAI MIH +#------------------------------------------------ +declare -x UE_MIHF_IP_ADDRESS="127.0.0.1" +declare -x UE_RAL_IP_ADDRESS="127.0.0.1" +LOG_FILE="/tmp/oai_sim_ue.log" + +#------------------------------------------------ +MIH_LOG_FILE="mih-f_ue.log" + +# EXE options +EXE_MODE="DEBUG" # "PROD" or "DEBUG" + +########################################################### +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +source $THIS_SCRIPT_PATH/env_802dot21.bash +########################################################### +bash_exec "service network-manager stop" +bash_exec "ifconfig $EMULATION_DEV_INTERFACE up $EMULATION_DEV_ADDRESS netmask 255.255.255.0" +bash_exec "ifconfig eth0 up 192.168.14.4 netmask 255.255.255.0" +########################################################### +IPTABLES=/sbin/iptables +THIS_SCRIPT_PATH=$(dirname $(readlink -f $0)) +declare -x OPENAIR_DIR="" +declare -x OPENAIR1_DIR="" +declare -x OPENAIR2_DIR="" +declare -x OPENAIR3_DIR="" +declare -x OPENAIR_TARGETS="" +########################################################### + +set_openair +cecho "OPENAIR_DIR = $OPENAIR_DIR" $green +cecho "OPENAIR1_DIR = $OPENAIR1_DIR" $green +cecho "OPENAIR2_DIR = $OPENAIR2_DIR" $green +cecho "OPENAIR3_DIR = $OPENAIR3_DIR" $green +cecho "OPENAIR_TARGETS = $OPENAIR_TARGETS" $green + + +################################################## +# LAUNCH UE +################################################## + +echo "Bringup UE interface" +pkill oaisim > /dev/null 2>&1 +pkill oaisim > /dev/null 2>&1 +pkill $MIH_F > /dev/null 2>&1 +pkill $UE_MIH_USER > /dev/null 2>&1 +rmmod -f $IP_DRIVER_NAME > /dev/null 2>&1 + +bash_exec "insmod $OPENAIR2_DIR/NAS/DRIVER/LITE/$IP_DRIVER_NAME.ko oai_nw_drv_IMEI=${NAS_IMEI[0]},${NAS_IMEI[1]},${NAS_IMEI[2]},${NAS_IMEI[3]},${NAS_IMEI[4]},${NAS_IMEI[5]},${NAS_IMEI[6]},${NAS_IMEI[7]},${NAS_IMEI[8]},${NAS_IMEI[9]},${NAS_IMEI[10]},${NAS_IMEI[11]},${NAS_IMEI[12]},${NAS_IMEI[13]}" +bash_exec "ip route flush cache" +bash_exec "ip link set $LTEIF up" +sleep 1 +bash_exec "ip addr add dev $LTEIF $UE_IPv4_CIDR" +bash_exec "ip addr add dev $LTEIF $UE_IPv6_CIDR" +sleep 1 +bash_exec "sysctl -w net.ipv4.conf.all.log_martians=1" +assert " `sysctl -n net.ipv4.conf.all.log_martians` -eq 1" $LINENO +bash_exec "sysctl -w net.ipv4.conf.all.rp_filter=0" +assert " `sysctl -n net.ipv4.conf.all.rp_filter` -eq 0" $LINENO +bash_exec "ip route flush cache" + +# Check table 200 lte in /etc/iproute2/rt_tables +fgrep lte /etc/iproute2/rt_tables > /dev/null +if [ $? -ne 0 ]; then + echo '200 lte ' >> /etc/iproute2/rt_tables +fi +ip rule add fwmark $IP_DEFAULT_MARK table lte +ip -4 route add default dev $LTEIF table lte +ip -6 route add default dev $LTEIF table lte +ip route add 239.0.0.160/28 dev $EMULATION_DEV_INTERFACE + +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type broadcast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type multicast -j MARK --set-mark $IP_DEFAULT_MARK + +#All other traffic is sent on the RAB you want (mark = RAB ID) +/sbin/ip6tables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/ip6tables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A POSTROUTING -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK +/sbin/iptables -A OUTPUT -t mangle -o oai0 -m pkttype --pkt-type unicast -j MARK --set-mark $IP_DEFAULT_MARK + +rotate_log_file $MIH_LOG_FILE + +echo "printing the MIH file path" +echo "$ODTONE_MIH_EXE_DIR/$MIH_F $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE" +echo "$ODTONE_MIH_EXE_DIR/$UE_MIH_USER $ODTONE_MIH_EXE_DIR/$UE_MIH_USER_CONF_FILE" + +# start MIH-F +xterm -hold -title "[RELAY][UE] MIHF" -e $ODTONE_MIH_EXE_DIR/$MIH_F --log 4 --conf.file $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE > $MIH_LOG_FILE 2>&1 & +wait_process_started $MIH_F +sleep 3 + +NOW=$(date +"%Y-%m-%d.%Hh_%Mm_%Ss") +rm -f $LOG_FILE + +UE_RAL_LINK_ID=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +UE_RAL_LINK_ID=`trim2 $UE_RAL_LINK_ID` +UE_RAL_LINK_ID=`echo $UE_RAL_LINK_ID | cut -d" " -f1` +UE_RAL_LINK_ID_STRIPPED=${UE_RAL_LINK_ID%%??} + +UE_RAL_LISTENING_PORT=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep links | grep \= | grep -v \# | cut -d"=" -f2` +UE_RAL_LISTENING_PORT=`trim2 $UE_RAL_LISTENING_PORT` +UE_RAL_LISTENING_PORT=`echo $UE_RAL_LISTENING_PORT | cut -d" " -f2` + +UE_MIHF_REMOTE_PORT=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep local_port | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` + +UE_MIHF_ID=`cat $ODTONE_MIH_EXE_DIR/$UE_MIH_F_CONF_FILE | grep id | grep \= | grep -v \# | tr -d " " | cut -d'=' -f2` + +#xterm -hold -e gdb --args +# $EMULATION_DEV_INTERFACE -D192.168.13.2 +#sudo ip route add 239.0.0.160/28 dev $EMULATION_DEV_INTERFACE +#$OPENAIR2_DIR/NAS/DRIVER/LITE/RB_TOOL/rb_tool -a -c0 -i0 -z0 -s 10.0.0.2 -t 10.0.0.1 -r 1 + +if [ $EXE_MODE = "DEBUG" ] ; then + echo "$OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l7 -u1 -b0 -M1 -p2 -g3 -D $EMULATION_DEV_ADDRESS --ue-ral-listening-port $UE_RAL_LISTENING_PORT --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED --ue-ral-ip-address $UE_RAL_IP_ADDRESS --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS --ue-mihf-id $UE_MIHF_ID " + + $OPENAIR_TARGETS/SIMU/USER/oaisim -a -K $LOG_FILE -l7 -u1 -b0 -M1 -p2 -g3 -D $EMULATION_DEV_INTERFACE \ + --ue-ral-listening-port $UE_RAL_LISTENING_PORT \ + --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED \ + --ue-ral-ip-address $UE_RAL_IP_ADDRESS \ + --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT \ + --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS \ + --ue-mihf-id $UE_MIHF_ID > log_ue.txt & +else + echo "$OPENAIR_TARGETS/SIMU/USER/oaisim -a -l3 -u1 -b0 -M1 -p2 -g3 -D $EMULATION_DEV_ADDRESS --ue-ral-listening-port $UE_RAL_LISTENING_PORT --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED --ue-ral-ip-address $UE_RAL_IP_ADDRESS --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS --ue-mihf-id $UE_MIHF_ID " + + $OPENAIR_TARGETS/SIMU/USER/oaisim -a -u1 -b0 -M1 -p2 -g1 -D $EMULATION_DEV_INTERFACE \ + --ue-ral-listening-port $UE_RAL_LISTENING_PORT \ + --ue-ral-link-id $UE_RAL_LINK_ID_STRIPPED \ + --ue-ral-ip-address $UE_RAL_IP_ADDRESS \ + --ue-mihf-remote-port $UE_MIHF_REMOTE_PORT \ + --ue-mihf-ip-address $UE_MIHF_IP_ADDRESS \ + --ue-mihf-id $UE_MIHF_ID > /dev/null & + +fi + +wait_process_started oaisim + + +# start MIH-USER + +# wait for emulation start +tshark -c 500 -i $EMULATION_DEV_INTERFACE > /dev/null 2>&1 +sleep 5 + +xterm -hold -title "[RELAY][UE] MIH_USER" -e $ODTONE_MIH_EXE_DIR/$UE_MIH_USER --conf.file $ODTONE_MIH_EXE_DIR/$UE_MIH_USER_CONF_FILE & +wait_process_started $UE_MIH_USER + +sleep 100000 + + + +