mirror of
				https://github.com/acidanthera/audk.git
				synced 2025-10-26 08:43:46 +01:00 
			
		
		
		
	git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@3 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			271 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			271 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*++
 | |
| 
 | |
| Copyright (c) 2006, Intel Corporation                                                         
 | |
| All rights reserved. This program and the accompanying materials                          
 | |
| are licensed and made available under the terms and conditions of the BSD License         
 | |
| which accompanies this distribution.  The full text of the license may be found at        
 | |
| http://opensource.org/licenses/bsd-license.php                                            
 | |
|                                                                                           
 | |
| THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,                     
 | |
| WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.             
 | |
| 
 | |
| Module Name:
 | |
| 
 | |
|     vtutf8.c
 | |
|     
 | |
| Abstract: 
 | |
|     
 | |
| 
 | |
| Revision History
 | |
| --*/
 | |
| 
 | |
| 
 | |
| #include "Terminal.h"
 | |
| 
 | |
| VOID
 | |
| VTUTF8RawDataToUnicode (
 | |
|   IN  TERMINAL_DEV    *TerminalDevice
 | |
|   )
 | |
| {
 | |
|   UTF8_CHAR Utf8Char;
 | |
|   UINT8     ValidBytes;
 | |
|   UINT16    UnicodeChar;
 | |
| 
 | |
|   ValidBytes = 0;
 | |
|   //
 | |
|   // pop the raw data out from the raw fifo,
 | |
|   // and translate it into unicode, then push
 | |
|   // the unicode into unicode fifo, until the raw fifo is empty.
 | |
|   //
 | |
|   while (!IsRawFiFoEmpty (TerminalDevice)) {
 | |
| 
 | |
|     GetOneValidUtf8Char (TerminalDevice, &Utf8Char, &ValidBytes);
 | |
| 
 | |
|     if (ValidBytes < 1 || ValidBytes > 3) {
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     Utf8ToUnicode (Utf8Char, ValidBytes, (CHAR16 *) &UnicodeChar);
 | |
| 
 | |
|     UnicodeFiFoInsertOneKey (TerminalDevice, UnicodeChar);
 | |
|   }
 | |
| }
 | |
| 
 | |
| VOID
 | |
| GetOneValidUtf8Char (
 | |
|   IN  TERMINAL_DEV      *Utf8Device,
 | |
|   OUT UTF8_CHAR         *Utf8Char,
 | |
|   OUT UINT8             *ValidBytes
 | |
|   )
 | |
| {
 | |
|   UINT8   Temp;
 | |
|   UINT8   Index;
 | |
|   BOOLEAN FetchFlag;
 | |
| 
 | |
|   Temp      = 0;
 | |
|   Index     = 0;
 | |
|   FetchFlag = TRUE;
 | |
| 
 | |
|   //
 | |
|   // if no valid Utf8 char is found in the RawFiFo,
 | |
|   // then *ValidBytes will be zero.
 | |
|   //
 | |
|   *ValidBytes = 0;
 | |
| 
 | |
|   while (!IsRawFiFoEmpty (Utf8Device)) {
 | |
| 
 | |
|     RawFiFoRemoveOneKey (Utf8Device, &Temp);
 | |
| 
 | |
|     switch (*ValidBytes) {
 | |
| 
 | |
|     case 0:
 | |
|       if ((Temp & 0x80) == 0) {
 | |
|         //
 | |
|         // one-byte utf8 char
 | |
|         //
 | |
|         *ValidBytes       = 1;
 | |
| 
 | |
|         Utf8Char->Utf8_1  = Temp;
 | |
| 
 | |
|         FetchFlag         = FALSE;
 | |
| 
 | |
|       } else if ((Temp & 0xe0) == 0xc0) {
 | |
|         //
 | |
|         // two-byte utf8 char
 | |
|         //
 | |
|         *ValidBytes         = 2;
 | |
| 
 | |
|         Utf8Char->Utf8_2[1] = Temp;
 | |
| 
 | |
|       } else if ((Temp & 0xf0) == 0xe0) {
 | |
|         //
 | |
|         // three-byte utf8 char
 | |
|         //
 | |
|         *ValidBytes         = 3;
 | |
| 
 | |
|         Utf8Char->Utf8_3[2] = Temp;
 | |
| 
 | |
|         Index++;
 | |
| 
 | |
|       } else {
 | |
|         //
 | |
|         // reset *ValidBytes to zero, let valid utf8 char search restart
 | |
|         //
 | |
|         *ValidBytes = 0;
 | |
|       }
 | |
| 
 | |
|       break;
 | |
| 
 | |
|     case 2:
 | |
|       if ((Temp & 0xc0) == 0x80) {
 | |
| 
 | |
|         Utf8Char->Utf8_2[0] = Temp;
 | |
| 
 | |
|         FetchFlag           = FALSE;
 | |
| 
 | |
|       } else {
 | |
| 
 | |
|         *ValidBytes = 0;
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case 3:
 | |
|       if ((Temp & 0xc0) == 0x80) {
 | |
| 
 | |
|         Utf8Char->Utf8_3[2 - Index] = Temp;
 | |
|         Index++;
 | |
|         if (Index == 3) {
 | |
|           FetchFlag = FALSE;
 | |
|         }
 | |
|       } else {
 | |
| 
 | |
|         *ValidBytes = 0;
 | |
|         Index       = 0;
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (!FetchFlag) {
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| VOID
 | |
| Utf8ToUnicode (
 | |
|   IN  UTF8_CHAR       Utf8Char,
 | |
|   IN  UINT8           ValidBytes,
 | |
|   OUT CHAR16          *UnicodeChar
 | |
|   )
 | |
| {
 | |
|   UINT8 UnicodeByte0;
 | |
|   UINT8 UnicodeByte1;
 | |
|   UINT8 Byte0;
 | |
|   UINT8 Byte1;
 | |
|   UINT8 Byte2;
 | |
| 
 | |
|   *UnicodeChar = 0;
 | |
| 
 | |
|   //
 | |
|   // translate utf8 code to unicode, in terminal standard,
 | |
|   // up to 3 bytes utf8 code is supported.
 | |
|   //
 | |
|   switch (ValidBytes) {
 | |
|   case 1:
 | |
|     //
 | |
|     // one-byte utf8 code
 | |
|     //
 | |
|     *UnicodeChar = (UINT16) Utf8Char.Utf8_1;
 | |
|     break;
 | |
| 
 | |
|   case 2:
 | |
|     //
 | |
|     // two-byte utf8 code
 | |
|     //
 | |
|     Byte0         = Utf8Char.Utf8_2[0];
 | |
|     Byte1         = Utf8Char.Utf8_2[1];
 | |
| 
 | |
|     UnicodeByte0  = (UINT8) ((Byte1 << 6) | (Byte0 & 0x3f));
 | |
|     UnicodeByte1  = (UINT8) ((Byte1 >> 2) & 0x07);
 | |
|     *UnicodeChar  = (UINT16) (UnicodeByte0 | (UnicodeByte1 << 8));
 | |
|     break;
 | |
| 
 | |
|   case 3:
 | |
|     //
 | |
|     // three-byte utf8 code
 | |
|     //
 | |
|     Byte0         = Utf8Char.Utf8_3[0];
 | |
|     Byte1         = Utf8Char.Utf8_3[1];
 | |
|     Byte2         = Utf8Char.Utf8_3[2];
 | |
| 
 | |
|     UnicodeByte0  = (UINT8) ((Byte1 << 6) | (Byte0 & 0x3f));
 | |
|     UnicodeByte1  = (UINT8) ((Byte2 << 4) | ((Byte1 >> 2) & 0x0f));
 | |
|     *UnicodeChar  = (UINT16) (UnicodeByte0 | (UnicodeByte1 << 8));
 | |
| 
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| VOID
 | |
| UnicodeToUtf8 (
 | |
|   IN  CHAR16      Unicode,
 | |
|   OUT UTF8_CHAR   *Utf8Char,
 | |
|   OUT UINT8       *ValidBytes
 | |
|   )
 | |
| {
 | |
|   UINT8 UnicodeByte0;
 | |
|   UINT8 UnicodeByte1;
 | |
|   //
 | |
|   // translate unicode to utf8 code
 | |
|   //
 | |
|   UnicodeByte0  = (UINT8) Unicode;
 | |
|   UnicodeByte1  = (UINT8) (Unicode >> 8);
 | |
| 
 | |
|   if (Unicode < 0x0080) {
 | |
| 
 | |
|     Utf8Char->Utf8_1  = (UINT8) (UnicodeByte0 & 0x7f);
 | |
|     *ValidBytes       = 1;
 | |
| 
 | |
|   } else if (Unicode < 0x0800) {
 | |
|     //
 | |
|     // byte sequence: high -> low
 | |
|     //                Utf8_2[0], Utf8_2[1]
 | |
|     //
 | |
|     Utf8Char->Utf8_2[1] = (UINT8) ((UnicodeByte0 & 0x3f) + 0x80);
 | |
|     Utf8Char->Utf8_2[0] = (UINT8) ((((UnicodeByte1 << 2) + (UnicodeByte0 >> 6)) & 0x1f) + 0xc0);
 | |
| 
 | |
|     *ValidBytes         = 2;
 | |
| 
 | |
|   } else {
 | |
|     //
 | |
|     // byte sequence: high -> low
 | |
|     //                Utf8_3[0], Utf8_3[1], Utf8_3[2]
 | |
|     //
 | |
|     Utf8Char->Utf8_3[2] = (UINT8) ((UnicodeByte0 & 0x3f) + 0x80);
 | |
|     Utf8Char->Utf8_3[1] = (UINT8) ((((UnicodeByte1 << 2) + (UnicodeByte0 >> 6)) & 0x3f) + 0x80);
 | |
|     Utf8Char->Utf8_3[0] = (UINT8) (((UnicodeByte1 >> 4) & 0x0f) + 0xe0);
 | |
| 
 | |
|     *ValidBytes         = 3;
 | |
|   }
 | |
| }
 | |
| 
 | |
| EFI_STATUS
 | |
| VTUTF8TestString (
 | |
|   IN  TERMINAL_DEV    *TerminalDevice,
 | |
|   IN  CHAR16          *WString
 | |
|   )
 | |
| {
 | |
|   //
 | |
|   // to utf8, all kind of characters are supported.
 | |
|   //
 | |
|   return EFI_SUCCESS;
 | |
| }
 |