/*
* 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 NLP_SAFT_COMPONENTS_LANG_ID_MOBILE_SCRIPT_TINY_SCRIPT_DETECTOR_H_
#define NLP_SAFT_COMPONENTS_LANG_ID_MOBILE_SCRIPT_TINY_SCRIPT_DETECTOR_H_
#include "lang_id/script/script-detector.h"
namespace libtextclassifier3 {
namespace mobile {
namespace lang_id {
// Unicode scripts we care about. To get compact and fast code, we detect only
// a few Unicode scripts that offer a strong indication about the language of
// the text (e.g., Hiragana -> Japanese).
enum Script {
// Special value to indicate internal errors in the script detection code.
kScriptError,
// Special values for all Unicode scripts that we do not detect. One special
// value for Unicode characters of 1, 2, 3, respectively 4 bytes (as we
// already have that information, we use it). kScriptOtherUtf8OneByte means
// ~Latin and kScriptOtherUtf8FourBytes means ~Han.
kScriptOtherUtf8OneByte,
kScriptOtherUtf8TwoBytes,
kScriptOtherUtf8ThreeBytes,
kScriptOtherUtf8FourBytes,
kScriptGreek,
kScriptCyrillic,
kScriptHebrew,
kScriptArabic,
kScriptHangulJamo, // Used primarily for Korean.
kScriptHiragana, // Used primarily for Japanese.
kScriptKatakana, // Used primarily for Japanese.
// Add new scripts here.
// Do not add any script after kNumRelevantScripts. This value indicates the
// number of elements in this enum Script (except this value) such that we can
// easily iterate over the scripts.
kNumRelevantScripts,
};
template<typename IntType>
inline bool InRange(IntType value, IntType low, IntType hi) {
return (value >= low) && (value <= hi);
}
// Returns Script for the UTF8 character that starts at address p.
// Precondition: p points to a valid UTF8 character of num_bytes bytes.
inline Script GetScript(const unsigned char *p, int num_bytes) {
switch (num_bytes) {
case 1:
return kScriptOtherUtf8OneByte;
case 2: {
// 2-byte UTF8 characters have 11 bits of information. unsigned int has
// at least 16 bits (http://en.cppreference.com/w/cpp/language/types) so
// it's enough. It's also usually the fastest int type on the current
// CPU, so it's better to use than int32.
static const unsigned int kGreekStart = 0x370;
// Commented out (unsued in the code): kGreekEnd = 0x3FF;
static const unsigned int kCyrillicStart = 0x400;
static const unsigned int kCyrillicEnd = 0x4FF;
static const unsigned int kHebrewStart = 0x590;
// Commented out (unsued in the code): kHebrewEnd = 0x5FF;
static const unsigned int kArabicStart = 0x600;
static const unsigned int kArabicEnd = 0x6FF;
const unsigned int codepoint = ((p[0] & 0x1F) << 6) | (p[1] & 0x3F);
if (codepoint > kCyrillicEnd) {
if (codepoint >= kArabicStart) {
if (codepoint <= kArabicEnd) {
return kScriptArabic;
}
} else {
// At this point, codepoint < kArabicStart = kHebrewEnd + 1, so
// codepoint <= kHebrewEnd.
if (codepoint >= kHebrewStart) {
return kScriptHebrew;
}
}
} else {
if (codepoint >= kCyrillicStart) {
return kScriptCyrillic;
} else {
// At this point, codepoint < kCyrillicStart = kGreekEnd + 1, so
// codepoint <= kGreekEnd.
if (codepoint >= kGreekStart) {
return kScriptGreek;
}
}
}
return kScriptOtherUtf8TwoBytes;
}
case 3: {
// 3-byte UTF8 characters have 16 bits of information. unsigned int has
// at least 16 bits.
static const unsigned int kHangulJamoStart = 0x1100;
static const unsigned int kHangulJamoEnd = 0x11FF;
static const unsigned int kHiraganaStart = 0x3041;
static const unsigned int kHiraganaEnd = 0x309F;
// Commented out (unsued in the code): kKatakanaStart = 0x30A0;
static const unsigned int kKatakanaEnd = 0x30FF;
const unsigned int codepoint =
((p[0] & 0x0F) << 12) | ((p[1] & 0x3F) << 6) | (p[2] & 0x3F);
if (codepoint > kHiraganaEnd) {
// On this branch, codepoint > kHiraganaEnd = kKatakanaStart - 1, so
// codepoint >= kKatakanaStart.
if (codepoint <= kKatakanaEnd) {
return kScriptKatakana;
}
} else {
if (codepoint >= kHiraganaStart) {
return kScriptHiragana;
} else {
if (InRange(codepoint, kHangulJamoStart, kHangulJamoEnd)) {
return kScriptHangulJamo;
}
}
}
return kScriptOtherUtf8ThreeBytes;
}
case 4:
return kScriptOtherUtf8FourBytes;
default:
return kScriptError;
}
}
// Returns Script for the UTF8 character that starts at address p. Similar to
// the previous version of GetScript, except for "char" vs "unsigned char".
// Most code works with "char *" pointers, ignoring the fact that char is
// unsigned (by default) on most platforms, but signed on iOS. This code takes
// care of making sure we always treat chars as unsigned.
inline Script GetScript(const char *p, int num_bytes) {
return GetScript(reinterpret_cast<const unsigned char *>(p),
num_bytes);
}
class TinyScriptDetector : public ScriptDetector {
public:
~TinyScriptDetector() override = default;
int GetScript(const char *s, int num_bytes) const override {
// Add the namespace in indicate that we want to call the method outside
// this class, instead of performing an infinite recursive call.
return libtextclassifier3::mobile::lang_id::GetScript(s, num_bytes);
}
int GetMaxScript() const override {
return kNumRelevantScripts - 1;
}
SAFTM_DEFINE_REGISTRATION_METHOD("tiny-script-detector", TinyScriptDetector);
};
} // namespace lang_id
} // namespace mobile
} // namespace nlp_saft
#endif // NLP_SAFT_COMPONENTS_LANG_ID_MOBILE_SCRIPT_TINY_SCRIPT_DETECTOR_H_