2 * Copyright (C) 2006, 2009, 2012 Apple Inc. All rights reserved.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public License
15 * along with this library; see the file COPYING.LIB. If not, write to
16 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
17 * Boston, MA 02110-1301, USA.
22 #include "wtf/text/StringImpl.h"
26 #include "wtf/MainThread.h"
27 #include "wtf/PassRefPtr.h"
28 #include "wtf/RetainPtr.h"
29 #include "wtf/Threading.h"
30 #include <CoreFoundation/CoreFoundation.h>
34 namespace StringWrapperCFAllocator
{
36 static StringImpl
* currentString
;
38 static const void* retain(const void* info
)
43 NO_RETURN_DUE_TO_ASSERT
44 static void release(const void*)
49 static CFStringRef
copyDescription(const void*)
51 return CFSTR("WTF::String-based allocator");
54 static void* allocate(CFIndex size
, CFOptionFlags
, void*)
56 StringImpl
* underlyingString
= 0;
58 underlyingString
= currentString
;
59 if (underlyingString
) {
61 underlyingString
->ref(); // Balanced by call to deref in deallocate below.
64 StringImpl
** header
= static_cast<StringImpl
**>(fastMalloc(sizeof(StringImpl
*) + size
));
65 *header
= underlyingString
;
69 static void* reallocate(void* pointer
, CFIndex newSize
, CFOptionFlags
, void*)
71 size_t newAllocationSize
= sizeof(StringImpl
*) + newSize
;
72 StringImpl
** header
= static_cast<StringImpl
**>(pointer
) - 1;
74 header
= static_cast<StringImpl
**>(fastRealloc(header
, newAllocationSize
));
78 static void deallocateOnMainThread(void* headerPointer
)
80 StringImpl
** header
= static_cast<StringImpl
**>(headerPointer
);
81 StringImpl
* underlyingString
= *header
;
82 ASSERT(underlyingString
);
83 underlyingString
->deref(); // Balanced by call to ref in allocate above.
87 static void deallocate(void* pointer
, void*)
89 StringImpl
** header
= static_cast<StringImpl
**>(pointer
) - 1;
90 StringImpl
* underlyingString
= *header
;
91 if (!underlyingString
) {
94 if (!isMainThread()) {
95 internal::callOnMainThread(&deallocateOnMainThread
, header
);
97 underlyingString
->deref(); // Balanced by call to ref in allocate above.
103 static CFIndex
preferredSize(CFIndex size
, CFOptionFlags
, void*)
105 // FIXME: If FastMalloc provided a "good size" callback, we'd want to use it here.
106 // Note that this optimization would help performance for strings created with the
107 // allocator that are mutable, and those typically are only created by callers who
108 // make a new string using the old string's allocator, such as some of the call
113 static CFAllocatorRef
create()
115 CFAllocatorContext context
= { 0, 0, retain
, release
, copyDescription
, allocate
, reallocate
, deallocate
, preferredSize
};
116 return CFAllocatorCreate(0, &context
);
119 static CFAllocatorRef
allocator()
121 static CFAllocatorRef allocator
= create();
127 RetainPtr
<CFStringRef
> StringImpl::createCFString()
129 // Since garbage collection isn't compatible with custom allocators, we
130 // can't use the NoCopy variants of CFStringCreate*() when GC is enabled.
131 if (!m_length
|| !isMainThread()) {
133 return adoptCF(CFStringCreateWithBytes(0, reinterpret_cast<const UInt8
*>(characters8()), m_length
, kCFStringEncodingISOLatin1
, false));
134 return adoptCF(CFStringCreateWithCharacters(0, reinterpret_cast<const UniChar
*>(characters16()), m_length
));
136 CFAllocatorRef allocator
= StringWrapperCFAllocator::allocator();
138 // Put pointer to the StringImpl in a global so the allocator can store it with the CFString.
139 ASSERT(!StringWrapperCFAllocator::currentString
);
140 StringWrapperCFAllocator::currentString
= this;
144 string
= CFStringCreateWithBytesNoCopy(allocator
, reinterpret_cast<const UInt8
*>(characters8()), m_length
, kCFStringEncodingISOLatin1
, false, kCFAllocatorNull
);
146 string
= CFStringCreateWithCharactersNoCopy(allocator
, reinterpret_cast<const UniChar
*>(characters16()), m_length
, kCFAllocatorNull
);
147 // CoreFoundation might not have to allocate anything, we clear currentString in case we did not execute allocate().
148 StringWrapperCFAllocator::currentString
= 0;
150 return adoptCF(string
);
153 // On StringImpl creation we could check if the allocator is the StringWrapperCFAllocator.
154 // If it is, then we could find the original StringImpl and just return that. But to
155 // do that we'd have to compute the offset from CFStringRef to the allocated block;
156 // the CFStringRef is *not* at the start of an allocated block. Testing shows 1000x
157 // more calls to createCFString than calls to the create functions with the appropriate
158 // allocator, so it's probably not urgent optimize that case.