• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2011 Google Inc. All Rights Reserved.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "sfntly/table/core/name_table.h"
18 
19 #include <stdio.h>
20 #include <string.h>
21 
22 #include <unicode/unistr.h>
23 
24 #include "sfntly/font.h"
25 #include "sfntly/port/exception_type.h"
26 
27 namespace sfntly {
28 /******************************************************************************
29  * NameTable::NameEntryId class
30  ******************************************************************************/
NameEntryId()31 NameTable::NameEntryId::NameEntryId()
32     : platform_id_(0),
33       encoding_id_(0),
34       language_id_(0),
35       name_id_(0) {
36 }
37 
NameEntryId(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)38 NameTable::NameEntryId::NameEntryId(int32_t platform_id,
39                                     int32_t encoding_id,
40                                     int32_t language_id,
41                                     int32_t name_id)
42     : platform_id_(platform_id),
43       encoding_id_(encoding_id),
44       language_id_(language_id),
45       name_id_(name_id) {
46 }
47 
NameEntryId(const NameTable::NameEntryId & rhs)48 NameTable::NameEntryId::NameEntryId(const NameTable::NameEntryId& rhs) {
49   *this = rhs;
50 }
51 
52 const NameTable::NameEntryId&
operator =(const NameTable::NameEntryId & rhs) const53     NameTable::NameEntryId::operator=(const NameTable::NameEntryId& rhs) const {
54   platform_id_ = rhs.platform_id_;
55   encoding_id_ = rhs.encoding_id_;
56   language_id_ = rhs.language_id_;
57   name_id_ = rhs.name_id_;
58   return *this;
59 }
60 
operator ==(const NameEntryId & rhs) const61 bool NameTable::NameEntryId::operator==(const NameEntryId& rhs) const {
62   return platform_id_ == rhs.platform_id_ &&
63          encoding_id_ == rhs.encoding_id_ &&
64          language_id_ == rhs.language_id_ &&
65          name_id_ == rhs.name_id_;
66 }
67 
operator <(const NameEntryId & rhs) const68 bool NameTable::NameEntryId::operator<(const NameEntryId& rhs) const {
69   if (platform_id_ != rhs.platform_id_) return platform_id_ < rhs.platform_id_;
70   if (encoding_id_ != rhs.encoding_id_) return encoding_id_ < rhs.encoding_id_;
71   if (language_id_ != rhs.language_id_) return language_id_ < rhs.language_id_;
72   return name_id_ < rhs.name_id_;
73 }
74 
75 /******************************************************************************
76  * NameTable::NameEntry class
77  ******************************************************************************/
NameEntry()78 NameTable::NameEntry::NameEntry() {
79   Init(0, 0, 0, 0, NULL);
80 }
81 
NameEntry(const NameEntryId & name_entry_id,const std::vector<uint8_t> & name_bytes)82 NameTable::NameEntry::NameEntry(const NameEntryId& name_entry_id,
83                                 const std::vector<uint8_t>& name_bytes) {
84   Init(name_entry_id.platform_id(),
85        name_entry_id.encoding_id(),
86        name_entry_id.language_id(),
87        name_entry_id.name_id(),
88        &name_bytes);
89 }
90 
NameEntry(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id,const std::vector<uint8_t> & name_bytes)91 NameTable::NameEntry::NameEntry(int32_t platform_id,
92                                 int32_t encoding_id,
93                                 int32_t language_id,
94                                 int32_t name_id,
95                                 const std::vector<uint8_t>& name_bytes) {
96   Init(platform_id, encoding_id, language_id, name_id, &name_bytes);
97 }
98 
~NameEntry()99 NameTable::NameEntry::~NameEntry() {}
100 
NameAsBytes()101 std::vector<uint8_t>* NameTable::NameEntry::NameAsBytes() {
102   return &name_bytes_;
103 }
104 
NameBytesLength()105 int32_t NameTable::NameEntry::NameBytesLength() {
106   return name_bytes_.size();
107 }
108 
Name()109 UChar* NameTable::NameEntry::Name() {
110   return NameTable::ConvertFromNameBytes(&name_bytes_,
111                                          platform_id(),
112                                          encoding_id());
113 }
114 
operator ==(const NameEntry & rhs) const115 bool NameTable::NameEntry::operator==(const NameEntry& rhs) const {
116   return (name_entry_id_ == rhs.name_entry_id_ &&
117           name_bytes_ == rhs.name_bytes_);
118 }
119 
Init(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id,const std::vector<uint8_t> * name_bytes)120 void NameTable::NameEntry::Init(int32_t platform_id,
121                                 int32_t encoding_id,
122                                 int32_t language_id,
123                                 int32_t name_id,
124                                 const std::vector<uint8_t>* name_bytes) {
125   name_entry_id_ = NameEntryId(platform_id, encoding_id, language_id, name_id);
126   if (name_bytes) {
127     name_bytes_ = *name_bytes;
128   } else {
129     name_bytes_.clear();
130   }
131 }
132 
133 /******************************************************************************
134  * NameTable::NameEntryBuilder class
135  ******************************************************************************/
NameEntryBuilder()136 NameTable::NameEntryBuilder::NameEntryBuilder() {
137   Init(0, 0, 0, 0, NULL);
138 }
139 
NameEntryBuilder(const NameEntryId & name_entry_id,const std::vector<uint8_t> & name_bytes)140 NameTable::NameEntryBuilder::NameEntryBuilder(const NameEntryId& name_entry_id,
141                                               const std::vector<uint8_t>& name_bytes) {
142   Init(name_entry_id.platform_id(),
143        name_entry_id.encoding_id(),
144        name_entry_id.language_id(),
145        name_entry_id.name_id(),
146        &name_bytes);
147 }
148 
NameEntryBuilder(const NameEntryId & name_entry_id)149 NameTable::NameEntryBuilder::NameEntryBuilder(
150     const NameEntryId& name_entry_id) {
151   Init(name_entry_id.platform_id(),
152        name_entry_id.encoding_id(),
153        name_entry_id.language_id(),
154        name_entry_id.name_id(),
155        NULL);
156 }
157 
NameEntryBuilder(NameEntry * b)158 NameTable::NameEntryBuilder::NameEntryBuilder(NameEntry* b) {
159   Init(b->platform_id(),
160        b->encoding_id(),
161        b->language_id(),
162        b->name_id(),
163        b->NameAsBytes());
164 }
165 
~NameEntryBuilder()166 NameTable::NameEntryBuilder::~NameEntryBuilder() {}
167 
SetName(const UChar * name)168 void NameTable::NameEntryBuilder::SetName(const UChar* name) {
169   if (name == NULL) {
170     name_entry_->name_bytes_.clear();
171     return;
172   }
173   NameTable::ConvertToNameBytes(name,
174                                 name_entry_->platform_id(),
175                                 name_entry_->encoding_id(),
176                                 &name_entry_->name_bytes_);
177 }
178 
SetName(const std::vector<uint8_t> & name_bytes)179 void NameTable::NameEntryBuilder::SetName(const std::vector<uint8_t>& name_bytes) {
180   name_entry_->name_bytes_.clear();
181   std::copy(name_bytes.begin(),
182             name_bytes.end(),
183             name_entry_->name_bytes_.begin());
184 }
185 
SetName(const std::vector<uint8_t> & name_bytes,int32_t offset,int32_t length)186 void NameTable::NameEntryBuilder::SetName(const std::vector<uint8_t>& name_bytes,
187                                           int32_t offset,
188                                           int32_t length) {
189   name_entry_->name_bytes_.clear();
190   std::copy(name_bytes.begin() + offset,
191             name_bytes.begin() + offset + length,
192             name_entry_->name_bytes_.begin());
193 }
194 
Init(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id,const std::vector<uint8_t> * name_bytes)195 void NameTable::NameEntryBuilder::Init(int32_t platform_id,
196                                        int32_t encoding_id,
197                                        int32_t language_id,
198                                        int32_t name_id,
199                                        const std::vector<uint8_t>* name_bytes) {
200   name_entry_ = new NameEntry();
201   name_entry_->Init(platform_id, encoding_id, language_id, name_id, name_bytes);
202 }
203 
204 /******************************************************************************
205  * NameTable::NameEntryFilterInPlace class (C++ port only)
206  ******************************************************************************/
NameEntryFilterInPlace(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)207 NameTable::NameEntryFilterInPlace::NameEntryFilterInPlace(int32_t platform_id,
208                                                           int32_t encoding_id,
209                                                           int32_t language_id,
210                                                           int32_t name_id)
211     : platform_id_(platform_id),
212       encoding_id_(encoding_id),
213       language_id_(language_id),
214       name_id_(name_id) {
215 }
216 
Accept(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)217 bool NameTable::NameEntryFilterInPlace::Accept(int32_t platform_id,
218                                                int32_t encoding_id,
219                                                int32_t language_id,
220                                                int32_t name_id) {
221   return (platform_id_ == platform_id &&
222           encoding_id_ == encoding_id &&
223           language_id_ == language_id &&
224           name_id_ == name_id);
225 }
226 
227 /******************************************************************************
228  * NameTable::NameEntryIterator class
229  ******************************************************************************/
NameEntryIterator(NameTable * table)230 NameTable::NameEntryIterator::NameEntryIterator(NameTable* table)
231     : RefIterator<NameEntry, NameTable>(table),
232       name_index_(0),
233       filter_(NULL) {
234 }
235 
NameEntryIterator(NameTable * table,NameEntryFilter * filter)236 NameTable::NameEntryIterator::NameEntryIterator(NameTable* table,
237                                                 NameEntryFilter* filter)
238     : RefIterator<NameEntry, NameTable>(table),
239       name_index_(0),
240       filter_(filter) {
241 }
242 
HasNext()243 bool NameTable::NameEntryIterator::HasNext() {
244   if (!filter_) {
245     if (name_index_ < container()->NameCount()) {
246       return true;
247     }
248     return false;
249   }
250   for (; name_index_ < container()->NameCount(); ++name_index_) {
251     if (filter_->Accept(container()->PlatformId(name_index_),
252                         container()->EncodingId(name_index_),
253                         container()->LanguageId(name_index_),
254                         container()->NameId(name_index_))) {
255       return true;
256     }
257   }
258   return false;
259 }
260 
Next()261 CALLER_ATTACH NameTable::NameEntry* NameTable::NameEntryIterator::Next() {
262   if (!HasNext())
263     return NULL;
264   return container()->GetNameEntry(name_index_++);
265 }
266 
267 /******************************************************************************
268  * NameTable::Builder class
269  ******************************************************************************/
Builder(Header * header,WritableFontData * data)270 NameTable::Builder::Builder(Header* header, WritableFontData* data)
271     : SubTableContainerTable::Builder(header, data) {
272 }
273 
Builder(Header * header,ReadableFontData * data)274 NameTable::Builder::Builder(Header* header, ReadableFontData* data)
275     : SubTableContainerTable::Builder(header, data) {
276 }
277 
278 CALLER_ATTACH NameTable::Builder*
CreateBuilder(Header * header,WritableFontData * data)279     NameTable::Builder::CreateBuilder(Header* header,
280                                       WritableFontData* data) {
281   Ptr<NameTable::Builder> builder;
282   builder = new NameTable::Builder(header, data);
283   return builder.Detach();
284 }
285 
RevertNames()286 void NameTable::Builder::RevertNames() {
287   name_entry_map_.clear();
288   set_model_changed(false);
289 }
290 
BuilderCount()291 int32_t NameTable::Builder::BuilderCount() {
292   GetNameBuilders();  // Ensure name_entry_map_ is built.
293   return (int32_t)name_entry_map_.size();
294 }
295 
Has(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)296 bool NameTable::Builder::Has(int32_t platform_id,
297                              int32_t encoding_id,
298                              int32_t language_id,
299                              int32_t name_id) {
300   NameEntryId probe(platform_id, encoding_id, language_id, name_id);
301   GetNameBuilders();  // Ensure name_entry_map_ is built.
302   return (name_entry_map_.find(probe) != name_entry_map_.end());
303 }
304 
305 CALLER_ATTACH NameTable::NameEntryBuilder*
NameBuilder(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)306     NameTable::Builder::NameBuilder(int32_t platform_id,
307                                     int32_t encoding_id,
308                                     int32_t language_id,
309                                     int32_t name_id) {
310   NameEntryId probe(platform_id, encoding_id, language_id, name_id);
311   NameEntryBuilderMap builders;
312   GetNameBuilders();  // Ensure name_entry_map_ is built.
313   if (name_entry_map_.find(probe) != name_entry_map_.end()) {
314     return name_entry_map_[probe];
315   }
316   NameEntryBuilderPtr builder = new NameEntryBuilder(probe);
317   name_entry_map_[probe] = builder;
318   return builder.Detach();
319 }
320 
Remove(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)321 bool NameTable::Builder::Remove(int32_t platform_id,
322                                 int32_t encoding_id,
323                                 int32_t language_id,
324                                 int32_t name_id) {
325   NameEntryId probe(platform_id, encoding_id, language_id, name_id);
326   GetNameBuilders();  // Ensure name_entry_map_ is built.
327   NameEntryBuilderMap::iterator position = name_entry_map_.find(probe);
328   if (position != name_entry_map_.end()) {
329     name_entry_map_.erase(position);
330     return true;
331   }
332   return false;
333 }
334 
335 CALLER_ATTACH FontDataTable*
SubBuildTable(ReadableFontData * data)336     NameTable::Builder::SubBuildTable(ReadableFontData* data) {
337   FontDataTablePtr table = new NameTable(header(), data);
338   return table.Detach();
339 }
340 
SubDataSet()341 void NameTable::Builder::SubDataSet() {
342   name_entry_map_.clear();
343   set_model_changed(false);
344 }
345 
SubDataSizeToSerialize()346 int32_t NameTable::Builder::SubDataSizeToSerialize() {
347   if (name_entry_map_.empty()) {
348     return 0;
349   }
350 
351   int32_t size = NameTable::Offset::kNameRecordStart +
352                  name_entry_map_.size() * NameTable::Offset::kNameRecordSize;
353   for (NameEntryBuilderMap::iterator b = name_entry_map_.begin(),
354                                      end = name_entry_map_.end();
355                                      b != end; ++b) {
356     NameEntryBuilderPtr p = b->second;
357     NameEntry* entry = p->name_entry();
358     size += entry->NameBytesLength();
359   }
360   return size;
361 }
362 
SubReadyToSerialize()363 bool NameTable::Builder::SubReadyToSerialize() {
364   return !name_entry_map_.empty();
365 }
366 
SubSerialize(WritableFontData * new_data)367 int32_t NameTable::Builder::SubSerialize(WritableFontData* new_data) {
368   int32_t string_table_start_offset =
369       NameTable::Offset::kNameRecordStart +
370       name_entry_map_.size() * NameTable::Offset::kNameRecordSize;
371 
372   // Header
373   new_data->WriteUShort(NameTable::Offset::kFormat, 0);
374   new_data->WriteUShort(NameTable::Offset::kCount, name_entry_map_.size());
375   new_data->WriteUShort(NameTable::Offset::kStringOffset,
376                         string_table_start_offset);
377   int32_t name_record_offset = NameTable::Offset::kNameRecordStart;
378   int32_t string_offset = 0;
379   // Note: we offered operator< in NameEntryId, which will be used by std::less,
380   //       and therefore our map will act like TreeMap in Java to provide
381   //       sorted key set.
382   for (NameEntryBuilderMap::iterator b = name_entry_map_.begin(),
383                                      end = name_entry_map_.end();
384                                      b != end; ++b) {
385     new_data->WriteUShort(
386         name_record_offset + NameTable::Offset::kNameRecordPlatformId,
387         b->first.platform_id());
388     new_data->WriteUShort(
389         name_record_offset + NameTable::Offset::kNameRecordEncodingId,
390         b->first.encoding_id());
391     new_data->WriteUShort(
392         name_record_offset + NameTable::Offset::kNameRecordLanguageId,
393         b->first.language_id());
394     new_data->WriteUShort(
395         name_record_offset + NameTable::Offset::kNameRecordNameId,
396         b->first.name_id());
397     NameEntry* builder_entry = b->second->name_entry();
398     new_data->WriteUShort(
399         name_record_offset + NameTable::Offset::kNameRecordStringLength,
400         builder_entry->NameBytesLength());
401     new_data->WriteUShort(
402         name_record_offset + NameTable::Offset::kNameRecordStringOffset,
403         string_offset);
404     name_record_offset += NameTable::Offset::kNameRecordSize;
405     string_offset += new_data->WriteBytes(
406         string_offset + string_table_start_offset,
407         builder_entry->NameAsBytes());
408   }
409 
410   return string_offset + string_table_start_offset;
411 }
412 
Initialize(ReadableFontData * data)413 void NameTable::Builder::Initialize(ReadableFontData* data) {
414   if (data) {
415     NameTablePtr table = new NameTable(header(), data);
416     Ptr<NameEntryIterator> name_iter;
417     name_iter.Attach(table->Iterator());
418     while (name_iter->HasNext()) {
419       NameEntryPtr name_entry;
420       name_entry.Attach(name_iter->Next());
421       NameEntryBuilderPtr name_entry_builder = new NameEntryBuilder(name_entry);
422       NameEntry* builder_entry = name_entry_builder->name_entry();
423       NameEntryId probe = builder_entry->name_entry_id();
424       name_entry_map_[probe] = name_entry_builder;
425     }
426   }
427 }
428 
GetNameBuilders()429 NameTable::NameEntryBuilderMap* NameTable::Builder::GetNameBuilders() {
430   if (name_entry_map_.empty()) {
431     Initialize(InternalReadData());
432   }
433   set_model_changed();
434   return &name_entry_map_;
435 }
436 
437 /******************************************************************************
438  * NameTable class
439  ******************************************************************************/
~NameTable()440 NameTable::~NameTable() {}
441 
Format()442 int32_t NameTable::Format() {
443   return data_->ReadUShort(Offset::kFormat);
444 }
445 
NameCount()446 int32_t NameTable::NameCount() {
447   return data_->ReadUShort(Offset::kCount);
448 }
449 
PlatformId(int32_t index)450 int32_t NameTable::PlatformId(int32_t index) {
451   return data_->ReadUShort(Offset::kNameRecordPlatformId +
452                            OffsetForNameRecord(index));
453 }
454 
EncodingId(int32_t index)455 int32_t NameTable::EncodingId(int32_t index) {
456   return data_->ReadUShort(Offset::kNameRecordEncodingId +
457                            OffsetForNameRecord(index));
458 }
459 
LanguageId(int32_t index)460 int32_t NameTable::LanguageId(int32_t index) {
461   return data_->ReadUShort(Offset::kNameRecordLanguageId +
462                            OffsetForNameRecord(index));
463 }
464 
NameId(int32_t index)465 int32_t NameTable::NameId(int32_t index) {
466   return data_->ReadUShort(Offset::kNameRecordNameId +
467                            OffsetForNameRecord(index));
468 }
469 
NameAsBytes(int32_t index,std::vector<uint8_t> * b)470 void NameTable::NameAsBytes(int32_t index, std::vector<uint8_t>* b) {
471   assert(b);
472   b->clear();
473 
474   int32_t length = NameLength(index);
475   if (length <= 0)
476     return;
477 
478   int32_t offset = NameOffset(index);
479   if (offset < 0)
480     return;
481 
482   b->resize(length);
483   data_->ReadBytes(offset, &((*b)[0]), 0, length);
484 }
485 
NameAsBytes(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id,std::vector<uint8_t> * b)486 void NameTable::NameAsBytes(int32_t platform_id,
487                             int32_t encoding_id,
488                             int32_t language_id,
489                             int32_t name_id,
490                             std::vector<uint8_t>* b) {
491   assert(b);
492   NameEntryPtr entry;
493   entry.Attach(GetNameEntry(platform_id, encoding_id, language_id, name_id));
494   if (entry) {
495     std::vector<uint8_t>* name = entry->NameAsBytes();
496     std::copy(name->begin(), name->end(), b->begin());
497   }
498 }
499 
Name(int32_t index)500 UChar* NameTable::Name(int32_t index) {
501   std::vector<uint8_t> b;
502   NameAsBytes(index, &b);
503   return ConvertFromNameBytes(&b, PlatformId(index), EncodingId(index));
504 }
505 
Name(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)506 UChar* NameTable::Name(int32_t platform_id,
507                        int32_t encoding_id,
508                        int32_t language_id,
509                        int32_t name_id) {
510   NameEntryPtr entry;
511   entry.Attach(GetNameEntry(platform_id, encoding_id, language_id, name_id));
512   if (entry) {
513     return entry->Name();
514   }
515   return NULL;
516 }
517 
GetNameEntry(int32_t index)518 CALLER_ATTACH NameTable::NameEntry* NameTable::GetNameEntry(int32_t index) {
519   std::vector<uint8_t> b;
520   NameAsBytes(index, &b);
521   NameEntryPtr instance = new NameEntry(PlatformId(index),
522                                         EncodingId(index),
523                                         LanguageId(index),
524                                         NameId(index), b);
525   return instance.Detach();
526 }
527 
GetNameEntry(int32_t platform_id,int32_t encoding_id,int32_t language_id,int32_t name_id)528 CALLER_ATTACH NameTable::NameEntry* NameTable::GetNameEntry(int32_t platform_id,
529                                                             int32_t encoding_id,
530                                                             int32_t language_id,
531                                                             int32_t name_id) {
532   NameTable::NameEntryFilterInPlace
533       filter(platform_id, encoding_id, language_id, name_id);
534   Ptr<NameTable::NameEntryIterator> name_entry_iter;
535   name_entry_iter.Attach(Iterator(&filter));
536   NameEntryPtr result;
537   if (name_entry_iter->HasNext()) {
538     result = name_entry_iter->Next();
539   }
540   return result;
541 }
542 
Iterator()543 CALLER_ATTACH NameTable::NameEntryIterator* NameTable::Iterator() {
544   Ptr<NameEntryIterator> output = new NameTable::NameEntryIterator(this);
545   return output.Detach();
546 }
547 
548 CALLER_ATTACH
Iterator(NameEntryFilter * filter)549 NameTable::NameEntryIterator* NameTable::Iterator(NameEntryFilter* filter) {
550   Ptr<NameEntryIterator> output =
551       new NameTable::NameEntryIterator(this, filter);
552   return output.Detach();
553 }
554 
NameTable(Header * header,ReadableFontData * data)555 NameTable::NameTable(Header* header, ReadableFontData* data)
556     : SubTableContainerTable(header, data) {}
557 
StringOffset()558 int32_t NameTable::StringOffset() {
559   return data_->ReadUShort(Offset::kStringOffset);
560 }
561 
OffsetForNameRecord(int32_t index)562 int32_t NameTable::OffsetForNameRecord(int32_t index) {
563   return Offset::kNameRecordStart + index * Offset::kNameRecordSize;
564 }
565 
NameLength(int32_t index)566 int32_t NameTable::NameLength(int32_t index) {
567   return data_->ReadUShort(Offset::kNameRecordStringLength +
568                            OffsetForNameRecord(index));
569 }
570 
NameOffset(int32_t index)571 int32_t NameTable::NameOffset(int32_t index) {
572   return data_->ReadUShort(Offset::kNameRecordStringOffset +
573                            OffsetForNameRecord(index)) + StringOffset();
574 }
575 
GetEncodingName(int32_t platform_id,int32_t encoding_id)576 const char* NameTable::GetEncodingName(int32_t platform_id,
577                                        int32_t encoding_id) {
578   switch (platform_id) {
579     case PlatformId::kUnicode:
580       return "UTF-16BE";
581     case PlatformId::kMacintosh:
582       switch (encoding_id) {
583         case MacintoshEncodingId::kRoman:
584           return "MacRoman";
585         case MacintoshEncodingId::kJapanese:
586           return "Shift-JIS";
587         case MacintoshEncodingId::kChineseTraditional:
588           return "Big5";
589         case MacintoshEncodingId::kKorean:
590           return "EUC-KR";
591         case MacintoshEncodingId::kArabic:
592           return "MacArabic";
593         case MacintoshEncodingId::kHebrew:
594           return "MacHebrew";
595         case MacintoshEncodingId::kGreek:
596           return "MacGreek";
597         case MacintoshEncodingId::kRussian:
598           return "MacCyrillic";
599         case MacintoshEncodingId::kRSymbol:
600           return "MacSymbol";
601         case MacintoshEncodingId::kThai:
602           return "MacThai";
603         case MacintoshEncodingId::kChineseSimplified:
604           return "EUC-CN";
605         default:  // Note: unknown/unconfirmed cases are not ported.
606           break;
607       }
608       break;
609     case PlatformId::kISO:
610       break;
611     case PlatformId::kWindows:
612       switch (encoding_id) {
613         case WindowsEncodingId::kSymbol:
614         case WindowsEncodingId::kUnicodeUCS2:
615           return "UTF-16BE";
616         case WindowsEncodingId::kShiftJIS:
617           return "windows-933";
618         case WindowsEncodingId::kPRC:
619           return "windows-936";
620         case WindowsEncodingId::kBig5:
621           return "windows-950";
622         case WindowsEncodingId::kWansung:
623           return "windows-949";
624         case WindowsEncodingId::kJohab:
625           return "ms1361";
626         case WindowsEncodingId::kUnicodeUCS4:
627           return "UCS-4";
628       }
629       break;
630     case PlatformId::kCustom:
631       break;
632     default:
633       break;
634   }
635   return NULL;
636 }
637 
GetCharset(int32_t platform_id,int32_t encoding_id)638 UConverter* NameTable::GetCharset(int32_t platform_id, int32_t encoding_id) {
639   UErrorCode error_code = U_ZERO_ERROR;
640   UConverter* conv = ucnv_open(GetEncodingName(platform_id, encoding_id),
641                                &error_code);
642   if (U_SUCCESS(error_code)) {
643     return conv;
644   }
645 
646   if (conv) {
647     ucnv_close(conv);
648   }
649   return NULL;
650 }
651 
ConvertToNameBytes(const UChar * name,int32_t platform_id,int32_t encoding_id,std::vector<uint8_t> * b)652 void NameTable::ConvertToNameBytes(const UChar* name,
653                                    int32_t platform_id,
654                                    int32_t encoding_id,
655                                    std::vector<uint8_t>* b) {
656   assert(b);
657   assert(name);
658   b->clear();
659   UConverter* cs = GetCharset(platform_id, encoding_id);
660   if (cs == NULL) {
661     return;
662   }
663 
664   // Preflight to get buffer size.
665   UErrorCode error_code = U_ZERO_ERROR;
666   int32_t length = ucnv_fromUChars(cs, NULL, 0, name, -1, &error_code);
667   b->resize(length + 4);  // The longest termination "\0" is 4 bytes.
668   memset(&((*b)[0]), 0, length + 4);
669   error_code = U_ZERO_ERROR;
670   ucnv_fromUChars(cs,
671                   reinterpret_cast<char*>(&((*b)[0])),
672                   length + 4,
673                   name,
674                   -1,
675                   &error_code);
676   if (!U_SUCCESS(error_code)) {
677     b->clear();
678   }
679   ucnv_close(cs);
680 }
681 
ConvertFromNameBytes(std::vector<uint8_t> * name_bytes,int32_t platform_id,int32_t encoding_id)682 UChar* NameTable::ConvertFromNameBytes(std::vector<uint8_t>* name_bytes,
683                                        int32_t platform_id,
684                                        int32_t encoding_id) {
685   if (name_bytes == NULL || name_bytes->size() == 0) {
686     return NULL;
687   }
688   UConverter* cs = GetCharset(platform_id, encoding_id);
689   UErrorCode error_code = U_ZERO_ERROR;
690   if (cs == NULL) {
691     char buffer[11] = {0};
692 #if defined (WIN32)
693     _itoa_s(platform_id, buffer, 16);
694 #else
695     snprintf(buffer, sizeof(buffer), "%x", platform_id);
696 #endif
697     UChar* result = new UChar[12];
698     memset(result, 0, sizeof(UChar) * 12);
699     cs = ucnv_open("utf-8", &error_code);
700     if (U_SUCCESS(error_code)) {
701       ucnv_toUChars(cs, result, 12, buffer, 11, &error_code);
702       ucnv_close(cs);
703       if (U_SUCCESS(error_code)) {
704         return result;
705       }
706     }
707     delete[] result;
708     return NULL;
709   }
710 
711   // No preflight needed here, we will be bigger.
712   UChar* output_buffer = new UChar[name_bytes->size() + 1];
713   memset(output_buffer, 0, sizeof(UChar) * (name_bytes->size() + 1));
714   int32_t length = ucnv_toUChars(cs,
715                                  output_buffer,
716                                  name_bytes->size(),
717                                  reinterpret_cast<char*>(&((*name_bytes)[0])),
718                                  name_bytes->size(),
719                                  &error_code);
720   ucnv_close(cs);
721   if (length > 0) {
722     return output_buffer;
723   }
724 
725   delete[] output_buffer;
726   return NULL;
727 }
728 
729 }  // namespace sfntly
730