/*
* Copyright (C) 2018 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef RIL_MNC_H
#define RIL_MNC_H
#include <climits>
#include <cstdio>
#include <string>
namespace ril {
namespace util {
namespace mnc {
/**
* Decode an MNC with an optional length indicator provided in the most-significant nibble.
*
* @param mnc an encoded MNC value; if no encoding is provided, then the string is returned
* as a minimum length string representing the provided integer.
*
* @return string representation of an encoded MNC or an empty string if the MNC is not a valid
* MNC value.
*/
static inline std::string decode(int mnc) {
if (mnc == INT_MAX || mnc < 0) return "";
unsigned umnc = mnc;
char mncNumDigits = (umnc >> (sizeof(int) * 8 - 4)) & 0xF;
umnc = (umnc << 4) >> 4;
if (umnc > 999) return "";
char mncStr[4] = {0};
switch (mncNumDigits) {
case 0:
// Legacy MNC report hasn't set the number of digits; preserve current
// behavior and make a string of the minimum number of required digits.
return std::to_string(umnc);
case 2:
snprintf(mncStr, sizeof(mncStr), "%03.3u", umnc);
return mncStr + 1;
case 3:
snprintf(mncStr, sizeof(mncStr), "%03.3u", umnc);
return mncStr;
default:
// Error case
return "";
}
}
/**
* Encode an MNC of the given value and a given number of digits
*
* @param mnc an MNC value 0-999 or INT_MAX if unknown
* @param numDigits the number of MNC digits {2, 3} or 0 if unknown
*
* @return an encoded MNC with embedded length information
*/
static inline int encode(int mnc, int numDigits) {
if (mnc > 999 || mnc < 0) return INT_MAX;
switch (numDigits) {
case 0: // fall through
case 2: // fall through
case 3:
break;
default:
return INT_MAX;
};
return (numDigits << (sizeof(int) * 8 - 4)) | mnc;
}
/**
* Encode an MNC of the given value
*
* @param mnc the string representation of the MNC, with the length equal to the length of the
* provided string.
*
* @return an encoded MNC with embedded length information
*/
static inline int encode(const std::string & mnc) {
return encode(std::stoi(mnc), mnc.length());
}
// echo -e "#include \"hardware/ril/include/telephony/ril_mnc.h\"\nint main()"\
// "{ return ril::util::mnc::test(); }" > ril_test.cpp \
// && g++ -o /tmp/ril_test -DTEST_RIL_MNC ril_test.cpp; \
// rm ril_test.cpp; /tmp/ril_test && [ $? ] && echo "passed"
#ifdef TEST_RIL_MNC
static int test() {
const struct mnc_strings { const char * in; const char * out; } mncs[] = {
{"0001",""},
{"9999",""},
{"0",""},
{"9",""},
{"123","123"},
{"000","000"},
{"001","001"},
{"011","011"},
{"111","111"},
{"00","00"},
{"01","01"},
{"11","11"},
{"09","09"},
{"099","099"},
{"999", "999"}};
for (int i=0; i< sizeof(mncs) / sizeof(struct mnc_strings); i++) {
if (decode(encode(mncs[i].in)).compare(mncs[i].out)) return 1;
}
const struct mnc_ints { const int in; const char * out; } legacy_mncs[] = {
{INT_MAX, ""},
{1, "1"},
{11, "11"},
{111, "111"},
{0, "0"},
{9999, ""},
};
for (int i=0; i < sizeof(legacy_mncs) / sizeof(struct mnc_ints); i++) {
if (decode(legacy_mncs[i].in).compare(legacy_mncs[i].out)) return 1;
}
return 0;
}
#endif
}
}
}
#endif /* !defined(RIL_MNC_H) */