Modest/source/myencoding/detect.c

998 lines
29 KiB
C
Raw Permalink Normal View History

2016-08-29 00:20:40 +03:00
/*
Copyright (C) 2015-2017 Alexander Borisov
2016-08-29 00:20:40 +03:00
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Author: lex.borisov@gmail.com (Alexander Borisov)
*/
#include "myencoding/encoding.h"
#include "myencoding/detect_resource.h"
#include "mycore/utils/resources.h"
2016-08-29 00:20:40 +03:00
myencoding_trigram_result_t myencoding_detect_by_trigram(unsigned const char *u_text, size_t length,
const myencoding_trigram_t *list, size_t list_length,
2016-08-29 00:20:40 +03:00
size_t max_sum_for_break)
{
myencoding_trigram_result_t res = {0, 0};
2016-08-29 00:20:40 +03:00
for (size_t i = 0; i < (length - 3); i++) {
if(u_text[i] > 127)
{
for (size_t j = 0; j < list_length; j++)
{
if(memcmp(list[j].trigram, &u_text[i], 3) == 0) {
res.value += list[j].value;
res.count++;
if(res.value >= max_sum_for_break)
i = length;
break;
}
}
}
}
return res;
}
bool myencoding_detect_russian_has_end(myencoding_trigram_result_t *res, size_t min_count, size_t min_value)
2016-08-29 00:20:40 +03:00
{
if(res->value >= min_value || res->count >= min_count)
return true;
return false;
}
bool myencoding_detect_unicode_has_end(myencoding_unicode_result_t *res, size_t max_bad_percent)
2016-08-29 00:20:40 +03:00
{
if(res->count_good == 0) {
if(res->count_bad)
return false;
return true;
}
else if(res->count_bad == 0)
return true;
size_t percent_bad = (res->count_bad * 100) / res->count_good;
if(percent_bad < max_bad_percent)
return true;
return false;
}
myencoding_unicode_result_t myencoding_detect_utf_8(unsigned const char *u_text, size_t length)
2016-08-29 00:20:40 +03:00
{
size_t i = 0;
myencoding_unicode_result_t res = {0, 0, 0};
2016-08-29 00:20:40 +03:00
while(i < length)
{
if((u_text[i] & 0x80) == 0x00) {
i++;
res.count_ascii++;
}
else if((u_text[i] & 0xE0) == 0xC0) {
i += 2;
if(i >= length)
break;
if(((u_text[i - 1] & 0x80) && (u_text[i - 1] & 0x40) == 0))
res.count_good++;
else
res.count_bad++;
}
else if((u_text[i] & 0xF0) == 0xE0) {
i += 3;
if(i >= length)
break;
if(((u_text[i - 1] & 0x80) && (u_text[i - 1] & 0x40) == 0) &&
((u_text[i - 2] & 0x80) && (u_text[i - 2] & 0x40) == 0)) {
res.count_good++;
}
else
res.count_bad++;
}
else if((u_text[i] & 0xF8) == 0xF0) {
i += 4;
if(i >= length)
break;
if(((u_text[i - 1] & 0x80) && (u_text[i - 1] & 0x40) == 0) &&
((u_text[i - 2] & 0x80) && (u_text[i - 2] & 0x40) == 0) &&
((u_text[i - 3] & 0x80) && (u_text[i - 3] & 0x40) == 0)) {
res.count_good++;
}
else
res.count_bad++;
}
else {
i++;
res.count_bad++;
}
}
return res;
}
myencoding_unicode_result_t myencoding_detect_utf_16(unsigned const char *u_text, size_t length)
2016-08-29 00:20:40 +03:00
{
size_t i = 0;
myencoding_unicode_result_t res = {0, 0, 0};
2016-08-29 00:20:40 +03:00
while(i < length)
{
if(u_text[i] == 0x00) {
if((i % 2) == 0) {
i++;
if(u_text[i] > 0x1F && u_text[i] < 0x7F)
res.count_bad++;
}
else {
if(u_text[(i - 1)] > 0x1F && u_text[(i - 1)] < 0x7F)
res.count_good++;
i++;
}
}
else
i++;
}
return res;
}
bool myencoding_detect_bom(const char *text, size_t length, myencoding_t *encoding)
2016-08-29 00:20:40 +03:00
{
unsigned const char *u_text = (unsigned const char*)text;
if(length > 2) {
if(u_text[0] == 0xEF &&
u_text[1] == 0xBB &&
u_text[2] == 0xBF)
{
*encoding = MyENCODING_UTF_8;
2016-08-29 00:20:40 +03:00
return true;
}
}
if(length > 1) {
if(u_text[0] == 0xFE && u_text[1] == 0xFF) {
*encoding = MyENCODING_UTF_16BE;
2016-08-29 00:20:40 +03:00
return true;
}
if(u_text[0] == 0xFF && u_text[1] == 0xFE) {
*encoding = MyENCODING_UTF_16LE;
2016-08-29 00:20:40 +03:00
return true;
}
}
// //for UTF-32
// if(length > 3) {
// if(u_text[0] == 0x00 &&
// u_text[1] == 0x00 &&
// u_text[2] == 0xFE &&
// u_text[3] == 0xFF)
// {
// *encoding = MyENCODING_UTF_32BE;
2016-08-29 00:20:40 +03:00
// return true;
// }
//
// if(u_text[0] == 0xFF &&
// u_text[1] == 0xFE &&
// u_text[2] == 0x00 &&
// u_text[3] == 0x00)
// {
// *encoding = MyENCODING_UTF_32LE;
2016-08-29 00:20:40 +03:00
// return true;
// }
// }
return false;
}
bool myencoding_detect_and_cut_bom(const char *text, size_t length, myencoding_t *encoding, const char **new_text, size_t *new_size)
2016-08-29 00:20:40 +03:00
{
if(myencoding_detect_bom(text, length, encoding))
2016-08-29 00:20:40 +03:00
{
if(*encoding == MyENCODING_UTF_8) {
2016-08-29 00:20:40 +03:00
*new_text = &text[3];
*new_size = length - 3;
}
else {
*new_text = &text[2];
*new_size = length - 2;
}
return true;
}
return false;
}
bool myencoding_detect_unicode(const char *text, size_t length, myencoding_t *encoding)
2016-08-29 00:20:40 +03:00
{
unsigned const char *u_text = (unsigned const char*)text;
*encoding = MyENCODING_DEFAULT;
2016-08-29 00:20:40 +03:00
myencoding_unicode_result_t res = myencoding_detect_utf_16(u_text, length);
2016-08-29 00:20:40 +03:00
if(res.count_bad == 0 && res.count_good >= 3) {
*encoding = MyENCODING_UTF_16LE;
2016-08-29 00:20:40 +03:00
return true;
}
else if(res.count_bad >= 3 && res.count_good == 0) {
*encoding = MyENCODING_UTF_16BE;
2016-08-29 00:20:40 +03:00
return true;
}
res = myencoding_detect_utf_8(u_text, length);
if(myencoding_detect_unicode_has_end(&res, 10)) {
*encoding = MyENCODING_UTF_8;
2016-08-29 00:20:40 +03:00
return true;
}
return false;
}
bool myencoding_detect_russian(const char *text, size_t length, myencoding_t *encoding)
2016-08-29 00:20:40 +03:00
{
unsigned const char *u_text = (unsigned const char*)text;
size_t min_count = 50;
size_t min_value = 100000;
size_t max_value = 0;
*encoding = MyENCODING_DEFAULT;
2016-08-29 00:20:40 +03:00
myencoding_trigram_result_t
res = myencoding_detect_by_trigram(u_text, length, myencoding_detect_trigrams_index_windows_1251, 1000, min_value);
if(myencoding_detect_russian_has_end(&res, min_count, min_value)) {
*encoding = MyENCODING_WINDOWS_1251;
2016-08-29 00:20:40 +03:00
return true;
}
max_value = res.value;
if(max_value) {
*encoding = MyENCODING_WINDOWS_1251;
2016-08-29 00:20:40 +03:00
}
res = myencoding_detect_by_trigram(u_text, length, myencoding_detect_trigrams_index_koi8_r, 1000, min_value);
if(myencoding_detect_russian_has_end(&res, min_count, min_value)) {
*encoding = MyENCODING_KOI8_R;
2016-08-29 00:20:40 +03:00
return true;
}
if(max_value < res.value) {
*encoding = MyENCODING_KOI8_R;
2016-08-29 00:20:40 +03:00
max_value = res.value;
}
res = myencoding_detect_by_trigram(u_text, length, myencoding_detect_trigrams_index_iso_8859_5, 1000, min_value);
if(myencoding_detect_russian_has_end(&res, min_count, min_value)) {
*encoding = MyENCODING_ISO_8859_5;
2016-08-29 00:20:40 +03:00
return true;
}
if(max_value < res.value) {
*encoding = MyENCODING_ISO_8859_5;
2016-08-29 00:20:40 +03:00
max_value = res.value;
}
res = myencoding_detect_by_trigram(u_text, length, myencoding_detect_trigrams_index_x_mac_cyrillic, 1000, min_value);
if(myencoding_detect_russian_has_end(&res, min_count, min_value)) {
*encoding = MyENCODING_X_MAC_CYRILLIC;
2016-08-29 00:20:40 +03:00
return true;
}
if(max_value < res.value) {
*encoding = MyENCODING_X_MAC_CYRILLIC;
2016-08-29 00:20:40 +03:00
max_value = res.value;
}
res = myencoding_detect_by_trigram(u_text, length, myencoding_detect_trigrams_index_ibm866, 1000, min_value);
if(myencoding_detect_russian_has_end(&res, min_count, min_value)) {
*encoding = MyENCODING_IBM866;
2016-08-29 00:20:40 +03:00
return true;
}
if(max_value < res.value) {
*encoding = MyENCODING_IBM866;
2016-08-29 00:20:40 +03:00
}
return false;
}
bool myencoding_detect(const char *text, size_t length, myencoding_t *encoding)
2016-08-29 00:20:40 +03:00
{
*encoding = MyENCODING_DEFAULT;
2016-08-29 00:20:40 +03:00
if(myencoding_detect_unicode(text, length, encoding))
2016-08-29 00:20:40 +03:00
return true;
if(myencoding_detect_russian(text, length, encoding))
2016-08-29 00:20:40 +03:00
return true;
return false;
}
const myencoding_detect_name_entry_t * myencoding_name_entry_by_name(const char* name, size_t length)
2016-08-29 00:20:40 +03:00
{
size_t idx = ((mycore_string_chars_lowercase_map[ (const unsigned char)name[0] ] *
mycore_string_chars_lowercase_map[ (const unsigned char)name[(length - 1)] ] *
2016-08-29 00:20:40 +03:00
length)
% MyENCODING_DETECT_NAME_STATIC_SIZE) + 1;
2016-08-29 00:20:40 +03:00
while (myencoding_detect_name_entry_static_list_index[idx].label)
2016-08-29 00:20:40 +03:00
{
if(myencoding_detect_name_entry_static_list_index[idx].label_length == length) {
if(mycore_strncasecmp(myencoding_detect_name_entry_static_list_index[idx].label, name, length) == 0)
return &myencoding_detect_name_entry_static_list_index[idx];
2016-08-29 00:20:40 +03:00
if(myencoding_detect_name_entry_static_list_index[idx].next)
idx = myencoding_detect_name_entry_static_list_index[idx].next;
2016-08-29 00:20:40 +03:00
else
return NULL;
}
else if(myencoding_detect_name_entry_static_list_index[idx].label_length > length) {
2016-08-29 00:20:40 +03:00
return NULL;
}
else {
idx = myencoding_detect_name_entry_static_list_index[idx].next;
2016-08-29 00:20:40 +03:00
}
}
return NULL;
}
bool myencoding_by_name(const char *name, size_t length, myencoding_t *encoding)
2016-08-29 00:20:40 +03:00
{
const myencoding_detect_name_entry_t *entry = myencoding_name_entry_by_name(name, length);
2016-08-29 00:20:40 +03:00
if(entry) {
if(encoding)
*encoding = entry->encoding;
return true;
}
return false;
}
const char * myencoding_name_by_id(myencoding_t encoding, size_t *length)
2017-02-14 14:49:05 +03:00
{
if(encoding >= MyENCODING_LAST_ENTRY) {
2017-02-14 14:49:05 +03:00
if(length) {
*length = 0;
}
return NULL;
}
const myencoding_entry_name_index_t *entry = &myencoding_entry_name_index_static_list_index[encoding];
2017-02-14 14:49:05 +03:00
if(length) {
*length = entry->length;
}
return entry->name;
}
/*
When an algorithm requires a user agent to prescan a byte stream to determine its encoding,
given some defined end condition, then it must run the following steps.
These steps either abort unsuccessfully or return a character encoding.
If at any point during these steps (including during instances of the get an attribute algorithm invoked by this one)
the user agent either runs out of bytes (meaning the position pointer created in the first step below goes beyond the end of the byte stream obtained so far)
or reaches its end condition, then abort the prescan a byte stream to determine its encoding algorithm unsuccessfully.
*/
bool myencoding_extracting_character_encoding_from_charset(const char *data, size_t data_size, myencoding_t *encoding)
2017-06-16 17:22:46 +03:00
{
return myencoding_extracting_character_encoding_from_charset_with_found(data, data_size, encoding, NULL, NULL);
}
2019-01-12 19:38:11 +03:00
bool myencoding_extracting_character_encoding_from_charset_with_found(const char *data, size_t data_size, myencoding_t *encoding, const char **found, size_t *found_length)
2017-02-14 14:49:05 +03:00
{
*encoding = MyENCODING_NOT_DETERMINED;
2017-02-14 14:49:05 +03:00
2017-06-16 17:22:46 +03:00
if(found)
*found = NULL;
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = 0;
2017-06-16 17:22:46 +03:00
2017-02-14 14:49:05 +03:00
/* 1 */
size_t length = 0;
size_t charset_length = strlen("charset");
bool is_get_pos = false;
const unsigned char *udata = (const unsigned char *)data;
/* 2 */
while((length + charset_length) < data_size) {
if(mycore_ustrcasecmp_without_checks_by_secondary((const unsigned char*)"charset", &udata[length]))
2017-02-14 14:49:05 +03:00
{
length += charset_length;
/* 2 */
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20)
{
break;
}
length++;
}
/* 4 */
if(udata[length] == 0x3D) { /* EQUALS SIGN (=) */
is_get_pos = true;
length++;
break;
}
}
length++;
}
if(is_get_pos == false || length >= data_size)
return false;
/* 5 */
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20)
{
break;
}
length++;
}
if(length >= data_size)
return false;
/* 6 */
/* " */
if(udata[length] == 0x22)
{
length++;
size_t begin = length;
while(length < data_size) {
if(udata[length] == 0x22)
2017-06-16 17:22:46 +03:00
{
if(found)
*found = &data[begin];
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = (length - begin);
2017-06-16 17:22:46 +03:00
return myencoding_by_name(&data[begin], (length - begin), encoding);
2017-06-16 17:22:46 +03:00
}
2017-02-14 14:49:05 +03:00
length++;
}
return false;
}
/* ' */
if(udata[length] == 0x27)
{
length++;
size_t begin = length;
while(length < data_size) {
if(udata[length] == 0x27)
2017-06-16 17:22:46 +03:00
{
if(found)
*found = &data[begin];
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = (length - begin);
2017-06-16 17:22:46 +03:00
return myencoding_by_name(&data[begin], (length - begin), encoding);
2017-06-16 17:22:46 +03:00
}
2017-02-14 14:49:05 +03:00
length++;
}
return false;
}
/* other */
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20)
{
size_t begin = length;
while(length < data_size) {
/* SEMICOLON character (;) */
2017-06-16 17:22:46 +03:00
if(udata[length] == 0x3B)
{
if(found)
*found = &data[begin];
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = (length - begin);
2017-06-16 17:22:46 +03:00
return myencoding_by_name(&data[begin], (length - begin), encoding);
2017-02-14 14:49:05 +03:00
}
length++;
}
2017-06-16 17:22:46 +03:00
if(found)
*found = &data[begin];
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = (length - begin);
2017-06-16 17:22:46 +03:00
return myencoding_by_name(&data[begin], (length - begin), encoding);
2017-02-14 14:49:05 +03:00
}
length++;
}
return false;
}
bool myencoding_prescan_stream_to_determine_encoding_get_attr_spaces(const unsigned char *udata, size_t *data_length, size_t data_size, myencoding_detect_attr_t *attr)
2017-02-14 14:49:05 +03:00
{
size_t length = *data_length;
/* set position */
attr->key_length = length - attr->key_begin;
/* 6 */
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20 && udata[length] != 0x2F)
{
break;
}
length++;
}
if(length >= data_size) {
*data_length = length;
return false;
}
/* 7 */
if(udata[length] != 0x3D) {
*data_length = length;
return false;
}
/* 8 */
*data_length = (length + 1);
return true;
}
size_t myencoding_prescan_stream_to_determine_encoding_get_attr_value(const unsigned char *udata, size_t length, size_t data_size, myencoding_detect_attr_t *attr, bool *it_last)
2017-02-14 14:49:05 +03:00
{
/* 9 */
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20)
{
break;
}
length++;
}
if(length >= data_size) {
*it_last = true;
return length;
}
/* 10 */
switch (udata[length]) {
case 0x22: /* (ASCII ") */
length++;
attr->value_begin = length;
while(length < data_size) {
if(udata[length] == 0x22)
{
attr->value_length = length - attr->value_begin;
return (length + 1);
}
length++;
}
break;
case 0x27: /* (ASCII ') */
length++;
attr->value_begin = length;
while(length < data_size) {
if(udata[length] == 0x27)
{
attr->value_length = length - attr->value_begin;
return (length + 1);
}
length++;
}
break;
case 0x3E: /* (ASCII >) */
*it_last = true;
return (length + 1);
default:
attr->value_begin = length;
while(length < data_size) {
if(udata[length] == 0x09 || udata[length] == 0x0A || udata[length] == 0x0C ||
udata[length] == 0x0D || udata[length] == 0x20 || udata[length] == 0x3E)
{
attr->value_length = length - attr->value_begin;
return (length + 1);
}
length++;
}
break;
}
attr->value_length = length - attr->value_begin;
return length;
}
size_t myencoding_prescan_stream_to_determine_encoding_get_attr(const unsigned char *udata, size_t length, size_t data_size, myencoding_detect_attr_t *attr, bool *it_last)
2017-02-14 14:49:05 +03:00
{
memset(attr, 0, sizeof(myencoding_detect_attr_t));
2017-02-14 14:49:05 +03:00
/*
If the byte at position is one of 0x09 (ASCII TAB), 0x0A (ASCII LF), 0x0C (ASCII FF), 0x0D (ASCII CR),
0x20 (ASCII space), or 0x2F (ASCII /) then advance position to the next byte and redo this step.
*/
/* 1 */
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20 && udata[length] != 0x2F)
{
break;
}
length++;
}
if(length >= data_size) {
*it_last = true;
return length;
}
/* 2 */
if(udata[length] == 0x3E) { /* (ASCII >) */
*it_last = true;
return (length + 1);
}
attr->key_begin = length;
/* 3, 4 */
while(length < data_size) {
switch (udata[length]) {
case 0x3D: /* (ASCII =) */
if(attr->key_begin != (length - 1)) {
attr->key_length = length - attr->key_begin;
length++;
return myencoding_prescan_stream_to_determine_encoding_get_attr_value(udata, length, data_size, attr, it_last);
2017-02-14 14:49:05 +03:00
}
/* fall through */
case 0x09: /* (ASCII TAB) */
case 0x0A: /* (ASCII LF) */
case 0x0C: /* (ASCII FF) */
case 0x0D: /* (ASCII CR) */
case 0x20: /* (ASCII space) */
length++;
if(myencoding_prescan_stream_to_determine_encoding_get_attr_spaces(udata, &length, data_size, attr) == false) {
2017-02-14 14:49:05 +03:00
*it_last = true;
return length;
}
return myencoding_prescan_stream_to_determine_encoding_get_attr_value(udata, length, data_size, attr, it_last);
2017-02-14 14:49:05 +03:00
case 0x2F: /* (ASCII /) */
case 0x3E: /* (ASCII >) */
*it_last = true;
attr->key_length = length - attr->key_begin;
return (length + 1);
default:
break;
}
length++;
}
if(myencoding_prescan_stream_to_determine_encoding_get_attr_spaces(udata, &length, data_size, attr) == false) {
2017-02-14 14:49:05 +03:00
*it_last = true;
return length;
}
return myencoding_prescan_stream_to_determine_encoding_get_attr_value(udata, length, data_size, attr, it_last);
2017-02-14 14:49:05 +03:00
}
2019-01-12 19:38:11 +03:00
bool myencoding_prescan_stream_to_determine_encoding_check_meta(const unsigned char *udata, size_t *length, size_t data_size, myencoding_t *encoding, const char **found, size_t *found_length)
2017-02-14 14:49:05 +03:00
{
myencoding_detect_attr_t attr;
2017-02-14 14:49:05 +03:00
bool got_pragma = false;
bool it_last = false;
unsigned int need_pragma = 0; /* 0 = NULL, 1 = false, 2 = true */
/*
http-equiv = 1
content = 2
charset = 4
*/
/* If the attribute's name is already in attribute list, then return to the step labeled attributes. */
size_t is_exists = 0;
while(*length < data_size) {
*length = myencoding_prescan_stream_to_determine_encoding_get_attr(udata, *length, data_size, &attr, &it_last);
2017-02-14 14:49:05 +03:00
/* 9 */
if(attr.key_length == strlen("http-equiv") &&
mycore_ustrcasecmp_without_checks_by_secondary((const unsigned char*)"http-equiv", &udata[ attr.key_begin ]))
2017-02-14 14:49:05 +03:00
{
if((is_exists & 1) == 0) {
is_exists |= 1;
if(attr.value_length == strlen("content-type") &&
mycore_ustrcasecmp_without_checks_by_secondary((const unsigned char*)"content-type", &udata[ attr.value_begin ]))
2017-02-14 14:49:05 +03:00
{
got_pragma = true;
}
}
}
else if(attr.key_length == strlen("content") &&
mycore_ustrcasecmp_without_checks_by_secondary((const unsigned char*)"content", &udata[ attr.key_begin ]))
2017-02-14 14:49:05 +03:00
{
if((is_exists & 2) == 0) {
is_exists |= 2;
2019-01-12 19:38:11 +03:00
if(myencoding_extracting_character_encoding_from_charset_with_found((const char*)(&udata[ attr.value_begin ]), attr.value_length, encoding, found, found_length)) {
2017-02-14 14:49:05 +03:00
need_pragma = 2;
}
}
}
else if(attr.key_length == strlen("charset") &&
mycore_ustrcasecmp_without_checks_by_secondary((const unsigned char*)"charset", &udata[ attr.key_begin ]))
2017-02-14 14:49:05 +03:00
{
if((is_exists & 4) == 0) {
is_exists |= 4;
2017-06-16 17:22:46 +03:00
if(found)
*found = (const char*)(&udata[ attr.value_begin ]);
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = attr.value_length;
2017-06-16 17:22:46 +03:00
myencoding_by_name((const char*)(&udata[ attr.value_begin ]), attr.value_length, encoding);
2017-02-14 14:49:05 +03:00
need_pragma = 1;
}
}
if(it_last)
break;
}
/* 11, 12, 13 */
if(need_pragma == 0 || (need_pragma == 2 && got_pragma == false)) {
*encoding = MyENCODING_NOT_DETERMINED;
2017-02-14 14:49:05 +03:00
return false;
}
/* 14 */
if(*encoding == MyENCODING_UTF_16BE || *encoding == MyENCODING_UTF_16LE) {
*encoding = MyENCODING_UTF_8;
2017-02-14 14:49:05 +03:00
}
/* 15 */
if(*encoding == MyENCODING_X_USER_DEFINED) {
*encoding = MyENCODING_WINDOWS_1252;
2017-02-14 14:49:05 +03:00
}
/* 16 */
return true;
}
size_t myencoding_prescan_stream_to_determine_encoding_skip_name(const unsigned char *udata, size_t length, size_t data_size)
2017-02-14 14:49:05 +03:00
{
while(length < data_size) {
if(udata[length] != 0x09 && udata[length] != 0x0A && udata[length] != 0x0C &&
udata[length] != 0x0D && udata[length] != 0x20)
{
break;
}
length++;
}
if(length >= data_size)
return length;
if(udata[length] == 0x3E) {
return (length + 1);
}
myencoding_detect_attr_t attr;
2017-02-14 14:49:05 +03:00
bool it_last = false;
while(length < data_size) {
length = myencoding_prescan_stream_to_determine_encoding_get_attr(udata, length, data_size, &attr, &it_last);
2017-02-14 14:49:05 +03:00
if(it_last) {
return length;
}
}
return length;
}
2016-08-29 00:20:40 +03:00
size_t myencoding_prescan_stream_to_determine_encoding_skip_other(const unsigned char *udata, size_t length, size_t data_size)
2017-02-14 14:49:05 +03:00
{
if(udata[length] == 0x2F) { /* / */
length++;
if(length >= data_size)
return length;
if(mycore_tokenizer_chars_map[ udata[length] ] == MyCORE_STRING_MAP_CHAR_A_Z_a_z) {
return myencoding_prescan_stream_to_determine_encoding_skip_name(udata, length, data_size);
2017-02-14 14:49:05 +03:00
}
while(length < data_size) {
if(udata[length] != 0x3E) {
return (length + 1);
}
length++;
}
return length;
}
else if(udata[length] == 0x21) { /* ! */
length++;
if((length + 2) < data_size && udata[length] == 0x2D && udata[(length+1)] == 0x2D) {
while(length < data_size) {
if(udata[length] != 0x3E) {
if(udata[(length - 1)] == 0x2D && udata[(length - 2)] == 0x2D)
return (length + 1);
}
length++;
2017-02-14 14:49:05 +03:00
}
return length;
}
while(length < data_size) {
if(udata[length] != 0x3E) {
return (length + 1);
}
length++;
}
return length;
}
else if(udata[length] == 0x3F) { /* ? */
length++;
while(length < data_size) {
if(udata[length] != 0x3E) {
return (length + 1);
}
length++;
}
return length;
}
return myencoding_prescan_stream_to_determine_encoding_skip_name(udata, length, data_size);
2017-02-14 14:49:05 +03:00
}
2016-08-29 00:20:40 +03:00
myencoding_t myencoding_prescan_stream_to_determine_encoding(const char *data, size_t data_size)
2017-06-16 17:22:46 +03:00
{
return myencoding_prescan_stream_to_determine_encoding_with_found(data, data_size, NULL, NULL);
}
2019-01-12 19:38:11 +03:00
myencoding_t myencoding_prescan_stream_to_determine_encoding_with_found(const char *data, size_t data_size, const char **found, size_t *found_length)
2017-02-14 14:49:05 +03:00
{
const unsigned char* udata = (const unsigned char*)data;
myencoding_t encoding = MyENCODING_NOT_DETERMINED;
2017-02-14 14:49:05 +03:00
2017-06-16 17:22:46 +03:00
if(found)
*found = NULL;
2019-01-12 19:38:11 +03:00
if(found_length)
*found_length = 0;
2017-06-16 17:22:46 +03:00
2017-02-14 14:49:05 +03:00
size_t i = 0;
while(i < data_size) {
/* 0x3C = '<' */
if(data[i] == 0x3C)
{
if((i + 5) >= data_size)
return encoding;
i++;
switch (data[i]) {
2017-06-16 17:22:46 +03:00
/*
A sequence of bytes starting with:
0x3C, 0x4D or 0x6D, 0x45 or 0x65, 0x54 or 0x74, 0x41 or 0x61,
and one of 0x09, 0x0A, 0x0C, 0x0D, 0x20, 0x2F
(case-insensitive ASCII '<meta' followed by a space or slash)
*/
2017-02-14 14:49:05 +03:00
case 0x4D:
case 0x6D:
if(mycore_ustrcasecmp_without_checks_by_secondary((const unsigned char*)"meta", &udata[i])) {
2017-02-14 14:49:05 +03:00
i += 4;
if(udata[i] == 0x09 || udata[i] == 0x0A || udata[i] == 0x0C ||
udata[i] == 0x0D || udata[i] == 0x20 || udata[i] == 0x2F)
{
i++;
2019-01-12 19:38:11 +03:00
if(myencoding_prescan_stream_to_determine_encoding_check_meta(udata, &i, data_size, &encoding, found, found_length))
2017-02-14 14:49:05 +03:00
return encoding;
}
}
break;
default:
i = myencoding_prescan_stream_to_determine_encoding_skip_other(udata, i, data_size);
2017-02-14 14:49:05 +03:00
break;
}
}
else {
i++;
}
}
return encoding;
}
2016-08-29 00:20:40 +03:00