2021-03-16 04:49:09 +03:00
/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
* Some of the source code in this file came from fs / cifs / cifs_unicode . c
* cifs_unicode : Unicode kernel case support
*
* Function :
* Convert a unicode character to upper or lower case using
* compressed tables .
*
* Copyright ( c ) International Business Machines Corp . , 2000 , 2009
*
*
* Notes :
* These APIs are based on the C library functions . The semantics
* should match the C functions but with expanded size operands .
*
* The upper / lower functions are based on a table created by mkupr .
* This is a compressed table of upper and lower case conversion .
*
*/
# ifndef _CIFS_UNICODE_H
# define _CIFS_UNICODE_H
# include <asm/byteorder.h>
# include <linux/types.h>
# include <linux/nls.h>
# define UNIUPR_NOLOWER /* Example to not expand lower case tables */
/*
* Windows maps these to the user defined 16 bit Unicode range since they are
* reserved symbols ( along with \ and / ) , otherwise illegal to store
* in filenames in NTFS
*/
2021-03-30 08:25:35 +03:00
# define UNI_ASTERISK ((__u16)('*' + 0xF000))
# define UNI_QUESTION ((__u16)('?' + 0xF000))
# define UNI_COLON ((__u16)(':' + 0xF000))
# define UNI_GRTRTHAN ((__u16)('>' + 0xF000))
# define UNI_LESSTHAN ((__u16)('<' + 0xF000))
# define UNI_PIPE ((__u16)('|' + 0xF000))
# define UNI_SLASH ((__u16)('\\' + 0xF000))
2021-03-16 04:49:09 +03:00
/* Just define what we want from uniupr.h. We don't want to define the tables
* in each source file .
*/
# ifndef UNICASERANGE_DEFINED
struct UniCaseRange {
wchar_t start ;
wchar_t end ;
signed char * table ;
} ;
# endif /* UNICASERANGE_DEFINED */
# ifndef UNIUPR_NOUPPER
extern signed char SmbUniUpperTable [ 512 ] ;
extern const struct UniCaseRange SmbUniUpperRange [ ] ;
# endif /* UNIUPR_NOUPPER */
# ifndef UNIUPR_NOLOWER
extern signed char CifsUniLowerTable [ 512 ] ;
extern const struct UniCaseRange CifsUniLowerRange [ ] ;
# endif /* UNIUPR_NOLOWER */
# ifdef __KERNEL__
int smb_strtoUTF16 ( __le16 * to , const char * from , int len ,
2021-05-26 11:57:12 +03:00
const struct nls_table * codepage ) ;
2021-03-30 08:25:35 +03:00
char * smb_strndup_from_utf16 ( const char * src , const int maxlen ,
2021-05-26 11:57:12 +03:00
const bool is_unicode ,
const struct nls_table * codepage ) ;
2021-03-30 08:25:35 +03:00
int smbConvertToUTF16 ( __le16 * target , const char * source , int srclen ,
2021-05-26 11:57:12 +03:00
const struct nls_table * cp , int mapchars ) ;
2021-03-30 08:25:35 +03:00
char * ksmbd_extract_sharename ( char * treename ) ;
2021-03-16 04:49:09 +03:00
# endif
/*
* UniStrcat : Concatenate the second string to the first
*
* Returns :
* Address of the first string
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrcat ( wchar_t * ucs1 , const wchar_t * ucs2 )
2021-03-16 04:49:09 +03:00
{
wchar_t * anchor = ucs1 ; /* save a pointer to start of ucs1 */
while ( * ucs1 + + )
/*NULL*/ ; /* To end of first string */
ucs1 - - ; /* Return to the null */
while ( ( * ucs1 + + = * ucs2 + + ) )
/*NULL*/ ; /* copy string 2 over */
return anchor ;
}
/*
* UniStrchr : Find a character in a string
*
* Returns :
* Address of first occurrence of character in string
* or NULL if the character is not in the string
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrchr ( const wchar_t * ucs , wchar_t uc )
2021-03-16 04:49:09 +03:00
{
while ( ( * ucs ! = uc ) & & * ucs )
ucs + + ;
if ( * ucs = = uc )
2021-03-30 08:25:35 +03:00
return ( wchar_t * ) ucs ;
2021-03-16 04:49:09 +03:00
return NULL ;
}
/*
* UniStrcmp : Compare two strings
*
* Returns :
* < 0 : First string is less than second
* = 0 : Strings are equal
* > 0 : First string is greater than second
*/
2021-03-30 08:25:35 +03:00
static inline int UniStrcmp ( const wchar_t * ucs1 , const wchar_t * ucs2 )
2021-03-16 04:49:09 +03:00
{
while ( ( * ucs1 = = * ucs2 ) & & * ucs1 ) {
ucs1 + + ;
ucs2 + + ;
}
2021-03-30 08:25:35 +03:00
return ( int ) * ucs1 - ( int ) * ucs2 ;
2021-03-16 04:49:09 +03:00
}
/*
* UniStrcpy : Copy a string
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrcpy ( wchar_t * ucs1 , const wchar_t * ucs2 )
2021-03-16 04:49:09 +03:00
{
wchar_t * anchor = ucs1 ; /* save the start of result string */
while ( ( * ucs1 + + = * ucs2 + + ) )
/*NULL*/ ;
return anchor ;
}
/*
* UniStrlen : Return the length of a string ( in 16 bit Unicode chars not bytes )
*/
2021-03-30 08:25:35 +03:00
static inline size_t UniStrlen ( const wchar_t * ucs1 )
2021-03-16 04:49:09 +03:00
{
int i = 0 ;
while ( * ucs1 + + )
i + + ;
return i ;
}
/*
* UniStrnlen : Return the length ( in 16 bit Unicode chars not bytes ) of a
* string ( length limited )
*/
2021-03-30 08:25:35 +03:00
static inline size_t UniStrnlen ( const wchar_t * ucs1 , int maxlen )
2021-03-16 04:49:09 +03:00
{
int i = 0 ;
while ( * ucs1 + + ) {
i + + ;
if ( i > = maxlen )
break ;
}
return i ;
}
/*
* UniStrncat : Concatenate length limited string
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrncat ( wchar_t * ucs1 , const wchar_t * ucs2 , size_t n )
2021-03-16 04:49:09 +03:00
{
wchar_t * anchor = ucs1 ; /* save pointer to string 1 */
while ( * ucs1 + + )
/*NULL*/ ;
ucs1 - - ; /* point to null terminator of s1 */
while ( n - - & & ( * ucs1 = * ucs2 ) ) { /* copy s2 after s1 */
ucs1 + + ;
ucs2 + + ;
}
* ucs1 = 0 ; /* Null terminate the result */
return anchor ;
}
/*
* UniStrncmp : Compare length limited string
*/
2021-03-30 08:25:35 +03:00
static inline int UniStrncmp ( const wchar_t * ucs1 , const wchar_t * ucs2 , size_t n )
2021-03-16 04:49:09 +03:00
{
if ( ! n )
return 0 ; /* Null strings are equal */
while ( ( * ucs1 = = * ucs2 ) & & * ucs1 & & - - n ) {
ucs1 + + ;
ucs2 + + ;
}
2021-03-30 08:25:35 +03:00
return ( int ) * ucs1 - ( int ) * ucs2 ;
2021-03-16 04:49:09 +03:00
}
/*
* UniStrncmp_le : Compare length limited string - native to little - endian
*/
2021-05-26 11:57:12 +03:00
static inline int
2021-03-16 04:49:09 +03:00
UniStrncmp_le ( const wchar_t * ucs1 , const wchar_t * ucs2 , size_t n )
{
if ( ! n )
return 0 ; /* Null strings are equal */
while ( ( * ucs1 = = __le16_to_cpu ( * ucs2 ) ) & & * ucs1 & & - - n ) {
ucs1 + + ;
ucs2 + + ;
}
2021-03-30 08:25:35 +03:00
return ( int ) * ucs1 - ( int ) __le16_to_cpu ( * ucs2 ) ;
2021-03-16 04:49:09 +03:00
}
/*
* UniStrncpy : Copy length limited string with pad
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrncpy ( wchar_t * ucs1 , const wchar_t * ucs2 , size_t n )
2021-03-16 04:49:09 +03:00
{
wchar_t * anchor = ucs1 ;
while ( n - - & & * ucs2 ) /* Copy the strings */
* ucs1 + + = * ucs2 + + ;
n + + ;
while ( n - - ) /* Pad with nulls */
* ucs1 + + = 0 ;
return anchor ;
}
/*
* UniStrncpy_le : Copy length limited string with pad to little - endian
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrncpy_le ( wchar_t * ucs1 , const wchar_t * ucs2 , size_t n )
2021-03-16 04:49:09 +03:00
{
wchar_t * anchor = ucs1 ;
while ( n - - & & * ucs2 ) /* Copy the strings */
* ucs1 + + = __le16_to_cpu ( * ucs2 + + ) ;
n + + ;
while ( n - - ) /* Pad with nulls */
* ucs1 + + = 0 ;
return anchor ;
}
/*
* UniStrstr : Find a string in a string
*
* Returns :
* Address of first match found
* NULL if no matching string is found
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrstr ( const wchar_t * ucs1 , const wchar_t * ucs2 )
2021-03-16 04:49:09 +03:00
{
const wchar_t * anchor1 = ucs1 ;
const wchar_t * anchor2 = ucs2 ;
while ( * ucs1 ) {
if ( * ucs1 = = * ucs2 ) {
/* Partial match found */
ucs1 + + ;
ucs2 + + ;
} else {
if ( ! * ucs2 ) /* Match found */
2021-03-30 08:25:35 +03:00
return ( wchar_t * ) anchor1 ;
2021-03-16 04:49:09 +03:00
ucs1 = + + anchor1 ; /* No match */
ucs2 = anchor2 ;
}
}
if ( ! * ucs2 ) /* Both end together */
2021-03-30 08:25:35 +03:00
return ( wchar_t * ) anchor1 ; /* Match found */
2021-03-16 04:49:09 +03:00
return NULL ; /* No match */
}
# ifndef UNIUPR_NOUPPER
/*
* UniToupper : Convert a unicode character to upper case
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t UniToupper ( register wchar_t uc )
2021-03-16 04:49:09 +03:00
{
register const struct UniCaseRange * rp ;
if ( uc < sizeof ( SmbUniUpperTable ) ) {
/* Latin characters */
return uc + SmbUniUpperTable [ uc ] ; /* Use base tables */
}
rp = SmbUniUpperRange ; /* Use range tables */
while ( rp - > start ) {
if ( uc < rp - > start ) /* Before start of range */
return uc ; /* Uppercase = input */
if ( uc < = rp - > end ) /* In range */
return uc + rp - > table [ uc - rp - > start ] ;
rp + + ; /* Try next range */
}
return uc ; /* Past last range */
}
/*
* UniStrupr : Upper case a unicode string
*/
2021-03-30 08:25:35 +03:00
static inline __le16 * UniStrupr ( register __le16 * upin )
2021-03-16 04:49:09 +03:00
{
register __le16 * up ;
up = upin ;
while ( * up ) { /* For all characters */
* up = cpu_to_le16 ( UniToupper ( le16_to_cpu ( * up ) ) ) ;
up + + ;
}
return upin ; /* Return input pointer */
}
# endif /* UNIUPR_NOUPPER */
# ifndef UNIUPR_NOLOWER
/*
* UniTolower : Convert a unicode character to lower case
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t UniTolower ( register wchar_t uc )
2021-03-16 04:49:09 +03:00
{
register const struct UniCaseRange * rp ;
if ( uc < sizeof ( CifsUniLowerTable ) ) {
/* Latin characters */
return uc + CifsUniLowerTable [ uc ] ; /* Use base tables */
}
rp = CifsUniLowerRange ; /* Use range tables */
while ( rp - > start ) {
if ( uc < rp - > start ) /* Before start of range */
return uc ; /* Uppercase = input */
if ( uc < = rp - > end ) /* In range */
return uc + rp - > table [ uc - rp - > start ] ;
rp + + ; /* Try next range */
}
return uc ; /* Past last range */
}
/*
* UniStrlwr : Lower case a unicode string
*/
2021-03-30 08:25:35 +03:00
static inline wchar_t * UniStrlwr ( register wchar_t * upin )
2021-03-16 04:49:09 +03:00
{
register wchar_t * up ;
up = upin ;
while ( * up ) { /* For all characters */
* up = UniTolower ( * up ) ;
up + + ;
}
return upin ; /* Return input pointer */
}
# endif
# endif /* _CIFS_UNICODE_H */