0abdbe5a7d
Former-commit-id: 7467d4b717762eeaf652d77f1486dd11ffb1ff1f
456 lines
11 KiB
C#
456 lines
11 KiB
C#
//
|
|
// System.String.cs
|
|
//
|
|
// Authors:
|
|
// Patrik Torstensson
|
|
// Jeffrey Stedfast (fejj@ximian.com)
|
|
// Dan Lewis (dihlewis@yahoo.co.uk)
|
|
// Sebastien Pouliot <sebastien@ximian.com>
|
|
// Marek Safar (marek.safar@seznam.cz)
|
|
// Andreas Nahr (Classdevelopment@A-SoftTech.com)
|
|
//
|
|
// (C) 2001 Ximian, Inc. http://www.ximian.com
|
|
// Copyright (C) 2004-2005 Novell (http://www.novell.com)
|
|
// Copyright (c) 2012 Xamarin, Inc (http://www.xamarin.com)
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
//
|
|
|
|
#if BIT64
|
|
using nuint = System.UInt64;
|
|
#else
|
|
using nuint = System.UInt32;
|
|
#endif
|
|
|
|
using System.Runtime.CompilerServices;
|
|
using System.Text;
|
|
using System.Globalization;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Runtime.InteropServices;
|
|
using System.Diagnostics;
|
|
|
|
namespace System
|
|
{
|
|
partial class String
|
|
{
|
|
[NonSerialized]
|
|
int _stringLength;
|
|
[NonSerialized]
|
|
char _firstChar;
|
|
|
|
public static readonly String Empty;
|
|
|
|
public int Length => _stringLength;
|
|
|
|
internal unsafe int IndexOfUnchecked (string value, int startIndex, int count)
|
|
{
|
|
int valueLen = value.Length;
|
|
if (count < valueLen)
|
|
return -1;
|
|
|
|
if (valueLen == 0)
|
|
return startIndex;
|
|
|
|
fixed (char* thisptr = &_firstChar, valueptr = value) {
|
|
char* ap = thisptr + startIndex;
|
|
char* thisEnd = ap + count - valueLen + 1;
|
|
while (ap != thisEnd) {
|
|
if (*ap == *valueptr) {
|
|
for (int i = 1; i < valueLen; i++) {
|
|
if (ap[i] != valueptr[i])
|
|
goto NextVal;
|
|
}
|
|
return (int)(ap - thisptr);
|
|
}
|
|
NextVal:
|
|
ap++;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
[CLSCompliant(false)]
|
|
public static String Concat(Object arg0, Object arg1, Object arg2, Object arg3, __arglist)
|
|
{
|
|
// Added to maintain backward compatibility, see https://github.com/mono/mono/issues/9996
|
|
throw new PlatformNotSupportedException();
|
|
}
|
|
|
|
internal unsafe int IndexOfUncheckedIgnoreCase (string value, int startIndex, int count)
|
|
{
|
|
int valueLen = value.Length;
|
|
if (count < valueLen)
|
|
return -1;
|
|
|
|
if (valueLen == 0)
|
|
return startIndex;
|
|
|
|
var ti = CultureInfo.InvariantCulture.TextInfo;
|
|
|
|
fixed (char* thisptr = &_firstChar, valueptr = value) {
|
|
char* ap = thisptr + startIndex;
|
|
char* thisEnd = ap + count - valueLen + 1;
|
|
char valueUpper = ti.ToUpper (*valueptr);
|
|
while (ap != thisEnd) {
|
|
if (ti.ToUpper (*ap) == valueUpper) {
|
|
for (int i = 1; i < valueLen; i++) {
|
|
if (ti.ToUpper (ap[i]) != ti.ToUpper (valueptr [i]))
|
|
goto NextVal;
|
|
}
|
|
return (int)(ap - thisptr);
|
|
}
|
|
NextVal:
|
|
ap++;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
internal unsafe int LastIndexOfUnchecked (string value, int startIndex, int count)
|
|
{
|
|
int valueLen = value.Length;
|
|
if (count < valueLen)
|
|
return -1;
|
|
|
|
if (valueLen == 0)
|
|
return startIndex;
|
|
|
|
fixed (char* thisptr = &_firstChar, valueptr = value) {
|
|
char* ap = thisptr + startIndex;
|
|
|
|
char* thisEnd = ap - count + valueLen - 1;
|
|
char* valueEnd = valueptr + valueLen - 1;
|
|
|
|
while (ap != thisEnd) {
|
|
if (*ap == *valueEnd) {
|
|
char* apEnd = ap;
|
|
while (valueptr != valueEnd) {
|
|
valueEnd--;
|
|
ap--;
|
|
if (*ap != *valueEnd) {
|
|
valueEnd = valueptr + valueLen - 1;
|
|
ap = apEnd;
|
|
goto NextVal;
|
|
}
|
|
}
|
|
|
|
return (int)(ap - thisptr);
|
|
}
|
|
NextVal:
|
|
ap--;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
internal unsafe int LastIndexOfUncheckedIgnoreCase (string value, int startIndex, int count)
|
|
{
|
|
int valueLen = value.Length;
|
|
if (count < valueLen)
|
|
return -1;
|
|
|
|
if (valueLen == 0)
|
|
return startIndex;
|
|
|
|
var ti = CultureInfo.InvariantCulture.TextInfo;
|
|
|
|
fixed (char* thisptr = &_firstChar, valueptr = value) {
|
|
char* ap = thisptr + startIndex;
|
|
|
|
char* thisEnd = ap - count + valueLen - 1;
|
|
char* valueEnd = valueptr + valueLen - 1;
|
|
|
|
var valueEndUpper = ti.ToUpper (*valueEnd);
|
|
|
|
while (ap != thisEnd) {
|
|
if (ti.ToUpper (*ap) == valueEndUpper) {
|
|
char* apEnd = ap;
|
|
while (valueptr != valueEnd) {
|
|
valueEnd--;
|
|
ap--;
|
|
if (ti.ToUpper (*ap) != ti.ToUpper (*valueEnd)) {
|
|
valueEnd = valueptr + valueLen - 1;
|
|
ap = apEnd;
|
|
goto NextVal;
|
|
}
|
|
}
|
|
|
|
return (int)(ap - thisptr);
|
|
}
|
|
NextVal:
|
|
ap--;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
internal bool StartsWithOrdinalUnchecked (String value)
|
|
{
|
|
if (this.Length < value.Length || _firstChar != value._firstChar)
|
|
return false;
|
|
|
|
return value.Length == 1 ?
|
|
true :
|
|
SpanHelpers.SequenceEqual (
|
|
ref Unsafe.As<char, byte> (ref this.GetRawStringData ()),
|
|
ref Unsafe.As<char, byte> (ref value.GetRawStringData ()),
|
|
((nuint)value.Length) * 2);
|
|
}
|
|
|
|
[MethodImplAttribute (MethodImplOptions.InternalCall)]
|
|
internal extern static String FastAllocateString (int length);
|
|
|
|
[MethodImplAttribute (MethodImplOptions.InternalCall)]
|
|
private extern static string InternalIsInterned (string str);
|
|
|
|
[MethodImplAttribute (MethodImplOptions.InternalCall)]
|
|
private extern static string InternalIntern (string str);
|
|
|
|
static unsafe int FastCompareStringHelper (uint* strAChars, int countA, uint* strBChars, int countB)
|
|
{
|
|
// CoreRT implementation has alignment issues
|
|
char* ap = (char*) strAChars;
|
|
char* bp = (char*) strBChars;
|
|
char* end = ap + Math.Min (countA, countB);
|
|
while (ap < end) {
|
|
if (*ap != *bp)
|
|
return (int)*ap - (int)*bp;
|
|
ap++;
|
|
bp++;
|
|
}
|
|
return countA - countB;
|
|
}
|
|
|
|
internal static unsafe void CharCopy (char *dest, char *src, int count) {
|
|
// Same rules as for memcpy, but with the premise that
|
|
// chars can only be aligned to even addresses if their
|
|
// enclosing types are correctly aligned
|
|
if ((((int)(byte*)dest | (int)(byte*)src) & 3) != 0) {
|
|
if (((int)(byte*)dest & 2) != 0 && ((int)(byte*)src & 2) != 0 && count > 0) {
|
|
((short*)dest) [0] = ((short*)src) [0];
|
|
dest++;
|
|
src++;
|
|
count--;
|
|
}
|
|
if ((((int)(byte*)dest | (int)(byte*)src) & 2) != 0) {
|
|
Buffer.memcpy2 ((byte*)dest, (byte*)src, count * 2);
|
|
return;
|
|
}
|
|
}
|
|
Buffer.memcpy4 ((byte*)dest, (byte*)src, count * 2);
|
|
}
|
|
|
|
#region Runtime method-to-ir dependencies
|
|
|
|
/* helpers used by the runtime as well as above or eslewhere in corlib */
|
|
static unsafe void memset (byte *dest, int val, int len)
|
|
{
|
|
if (len < 8) {
|
|
while (len != 0) {
|
|
*dest = (byte)val;
|
|
++dest;
|
|
--len;
|
|
}
|
|
return;
|
|
}
|
|
if (val != 0) {
|
|
val = val | (val << 8);
|
|
val = val | (val << 16);
|
|
}
|
|
// align to 4
|
|
int rest = (int)dest & 3;
|
|
if (rest != 0) {
|
|
rest = 4 - rest;
|
|
len -= rest;
|
|
do {
|
|
*dest = (byte)val;
|
|
++dest;
|
|
--rest;
|
|
} while (rest != 0);
|
|
}
|
|
while (len >= 16) {
|
|
((int*)dest) [0] = val;
|
|
((int*)dest) [1] = val;
|
|
((int*)dest) [2] = val;
|
|
((int*)dest) [3] = val;
|
|
dest += 16;
|
|
len -= 16;
|
|
}
|
|
while (len >= 4) {
|
|
((int*)dest) [0] = val;
|
|
dest += 4;
|
|
len -= 4;
|
|
}
|
|
// tail bytes
|
|
while (len > 0) {
|
|
*dest = (byte)val;
|
|
dest++;
|
|
len--;
|
|
}
|
|
}
|
|
|
|
static unsafe void memcpy (byte *dest, byte *src, int size)
|
|
{
|
|
Buffer.Memcpy (dest, src, size);
|
|
}
|
|
|
|
/* Used by the runtime */
|
|
internal static unsafe void bzero (byte *dest, int len) {
|
|
memset (dest, 0, len);
|
|
}
|
|
|
|
internal static unsafe void bzero_aligned_1 (byte *dest, int len) {
|
|
((byte*)dest) [0] = 0;
|
|
}
|
|
|
|
internal static unsafe void bzero_aligned_2 (byte *dest, int len) {
|
|
((short*)dest) [0] = 0;
|
|
}
|
|
|
|
internal static unsafe void bzero_aligned_4 (byte *dest, int len) {
|
|
((int*)dest) [0] = 0;
|
|
}
|
|
|
|
internal static unsafe void bzero_aligned_8 (byte *dest, int len) {
|
|
((long*)dest) [0] = 0;
|
|
}
|
|
|
|
internal static unsafe void memcpy_aligned_1 (byte *dest, byte *src, int size) {
|
|
((byte*)dest) [0] = ((byte*)src) [0];
|
|
}
|
|
|
|
internal static unsafe void memcpy_aligned_2 (byte *dest, byte *src, int size) {
|
|
((short*)dest) [0] = ((short*)src) [0];
|
|
}
|
|
|
|
internal static unsafe void memcpy_aligned_4 (byte *dest, byte *src, int size) {
|
|
((int*)dest) [0] = ((int*)src) [0];
|
|
}
|
|
|
|
internal static unsafe void memcpy_aligned_8 (byte *dest, byte *src, int size) {
|
|
((long*)dest) [0] = ((long*)src) [0];
|
|
}
|
|
|
|
#endregion
|
|
|
|
// Certain constructors are redirected to CreateString methods with
|
|
// matching argument list. The this pointer should not be used.
|
|
|
|
unsafe String CreateString (sbyte* value)
|
|
{
|
|
return Ctor (value);
|
|
}
|
|
|
|
unsafe String CreateString (sbyte* value, int startIndex, int length)
|
|
{
|
|
return Ctor (value, startIndex, length);
|
|
}
|
|
|
|
unsafe string CreateString (char* value)
|
|
{
|
|
return Ctor (value);
|
|
}
|
|
|
|
unsafe string CreateString (char* value, int startIndex, int length)
|
|
{
|
|
return Ctor (value, startIndex, length);
|
|
}
|
|
|
|
string CreateString (char [] val, int startIndex, int length)
|
|
{
|
|
return Ctor (val, startIndex, length);
|
|
}
|
|
|
|
string CreateString (char [] val)
|
|
{
|
|
return Ctor (val);
|
|
}
|
|
|
|
string CreateString (char c, int count)
|
|
{
|
|
return Ctor (c, count);
|
|
}
|
|
|
|
unsafe String CreateString (sbyte* value, int startIndex, int length, Encoding enc)
|
|
{
|
|
return Ctor (value, startIndex, length, enc);
|
|
}
|
|
|
|
String CreateString (ReadOnlySpan<char> value)
|
|
{
|
|
return Ctor (value);
|
|
}
|
|
|
|
[IndexerName ("Chars")]
|
|
public char this [int index] {
|
|
[IntrinsicAttribute]
|
|
get {
|
|
if ((uint)index >= _stringLength)
|
|
ThrowHelper.ThrowIndexOutOfRangeException ();
|
|
|
|
return Unsafe.Add (ref _firstChar, index);
|
|
}
|
|
}
|
|
|
|
public static String Intern (String str)
|
|
{
|
|
if (str == null) {
|
|
throw new ArgumentNullException ("str");
|
|
}
|
|
|
|
return InternalIntern (str);
|
|
}
|
|
|
|
public static String IsInterned (String str)
|
|
{
|
|
if (str == null)
|
|
throw new ArgumentNullException ("str");
|
|
|
|
return InternalIsInterned (str);
|
|
}
|
|
|
|
int LegacyStringGetHashCode ()
|
|
{
|
|
int hash1 = 5381;
|
|
int hash2 = hash1;
|
|
|
|
unsafe {
|
|
fixed (char *src = this) {
|
|
int c;
|
|
char *s = src;
|
|
while ((c = s[0]) != 0) {
|
|
hash1 = ((hash1 << 5) + hash1) ^ c;
|
|
c = s [1];
|
|
if (c == 0)
|
|
break;
|
|
hash2 = ((hash2 << 5) + hash2) ^ c;
|
|
s += 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hash1 + (hash2 * 1566083941);
|
|
}
|
|
}
|
|
} |