cp936ext.h 3.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. /*
  2. * Copyright (C) 1999-2001, 2016 Free Software Foundation, Inc.
  3. * This file is part of the GNU LIBICONV Library.
  4. *
  5. * The GNU LIBICONV Library is free software; you can redistribute it
  6. * and/or modify it under the terms of the GNU Lesser General Public
  7. * License as published by the Free Software Foundation; either version 2.1
  8. * of the License, or (at your option) any later version.
  9. *
  10. * The GNU LIBICONV Library is distributed in the hope that it will be
  11. * useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  13. * Lesser General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU Lesser General Public
  16. * License along with the GNU LIBICONV Library; see the file COPYING.LIB.
  17. * If not, see <https://www.gnu.org/licenses/>.
  18. */
  19. /*
  20. * CP936 extensions
  21. */
  22. static const unsigned short cp936ext_2uni_pagea6[181-159] = {
  23. /* 0xa6 */
  24. 0xfe35,
  25. 0xfe36, 0xfe39, 0xfe3a, 0xfe3f, 0xfe40, 0xfe3d, 0xfe3e, 0xfe41,
  26. 0xfe42, 0xfe43, 0xfe44, 0xfffd, 0xfffd, 0xfe3b, 0xfe3c, 0xfe37,
  27. 0xfe38, 0xfe31, 0xfffd, 0xfe33, 0xfe34,
  28. };
  29. static const unsigned short cp936ext_2uni_pagea8[128-122] = {
  30. /* 0xa8 */
  31. 0x0251, 0xfffd, 0x0144, 0x0148, 0xfffd, 0x0261,
  32. };
  33. static int
  34. cp936ext_mbtowc (conv_t conv, ucs4_t *pwc, const unsigned char *s, size_t n)
  35. {
  36. unsigned char c1 = s[0];
  37. if ((c1 == 0xa6) || (c1 == 0xa8)) {
  38. if (n >= 2) {
  39. unsigned char c2 = s[1];
  40. if ((c2 >= 0x40 && c2 < 0x7f) || (c2 >= 0x80 && c2 < 0xff)) {
  41. unsigned int i = 190 * (c1 - 0x81) + (c2 - (c2 >= 0x80 ? 0x41 : 0x40));
  42. unsigned short wc = 0xfffd;
  43. if (i < 7410) {
  44. if (i >= 7189 && i < 7211)
  45. wc = cp936ext_2uni_pagea6[i-7189];
  46. } else {
  47. if (i >= 7532 && i < 7538)
  48. wc = cp936ext_2uni_pagea8[i-7532];
  49. }
  50. if (wc != 0xfffd) {
  51. *pwc = (ucs4_t) wc;
  52. return 2;
  53. }
  54. }
  55. return RET_ILSEQ;
  56. }
  57. return RET_TOOFEW(0);
  58. }
  59. return RET_ILSEQ;
  60. }
  61. static const unsigned short cp936ext_page01[16] = {
  62. 0x0000, 0x0000, 0x0000, 0x0000, 0xa8bd, 0x0000, 0x0000, 0x0000, /*0x40-0x47*/
  63. 0xa8be, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, /*0x48-0x4f*/
  64. };
  65. static const unsigned short cp936ext_page02[24] = {
  66. 0x0000, 0xa8bb, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, /*0x50-0x57*/
  67. 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, /*0x58-0x5f*/
  68. 0x0000, 0xa8c0, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, /*0x60-0x67*/
  69. };
  70. static const unsigned short cp936ext_pagefe[24] = {
  71. 0x0000, 0xa6f2, 0x0000, 0xa6f4, 0xa6f5, 0xa6e0, 0xa6e1, 0xa6f0, /*0x30-0x37*/
  72. 0xa6f1, 0xa6e2, 0xa6e3, 0xa6ee, 0xa6ef, 0xa6e6, 0xa6e7, 0xa6e4, /*0x38-0x3f*/
  73. 0xa6e5, 0xa6e8, 0xa6e9, 0xa6ea, 0xa6eb, 0x0000, 0x0000, 0x0000, /*0x40-0x47*/
  74. };
  75. static int
  76. cp936ext_wctomb (conv_t conv, unsigned char *r, ucs4_t wc, size_t n)
  77. {
  78. if (n >= 2) {
  79. unsigned short c = 0;
  80. if (wc >= 0x0140 && wc < 0x0150)
  81. c = cp936ext_page01[wc-0x0140];
  82. else if (wc >= 0x0250 && wc < 0x0268)
  83. c = cp936ext_page02[wc-0x0250];
  84. else if (wc >= 0xfe30 && wc < 0xfe48)
  85. c = cp936ext_pagefe[wc-0xfe30];
  86. if (c != 0) {
  87. r[0] = (c >> 8); r[1] = (c & 0xff);
  88. return 2;
  89. }
  90. return RET_ILUNI;
  91. }
  92. return RET_TOOSMALL;
  93. }