Bug 182975

Summary: Fix std::make_unique / new[] using system malloc
Product: WebKit Reporter: Yusuke Suzuki <ysuzuki>
Component: New BugsAssignee: Yusuke Suzuki <ysuzuki>
Status: RESOLVED FIXED    
Severity: Normal CC: achristensen, commit-queue, darin, ews-watchlist, jfbastien, mark.lam, saam, sam, webkit-bug-importer
Priority: P2 Keywords: InRadar
Version: WebKit Nightly Build   
Hardware: Unspecified   
OS: Unspecified   
Attachments:
Description Flags
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch
none
Patch jfbastien: review+, commit-queue: commit-queue-

Yusuke Suzuki
Reported 2018-02-20 09:30:43 PST
Fix std::make_unique / new[] using system malloc
Attachments
Patch (19.79 KB, patch)
2018-02-20 09:31 PST, Yusuke Suzuki
no flags
Patch (59.19 KB, patch)
2018-02-20 10:23 PST, Yusuke Suzuki
no flags
Patch (65.08 KB, patch)
2018-02-21 01:06 PST, Yusuke Suzuki
no flags
Patch (74.01 KB, patch)
2018-02-21 02:18 PST, Yusuke Suzuki
no flags
Patch (74.32 KB, patch)
2018-02-21 02:33 PST, Yusuke Suzuki
no flags
Patch (74.57 KB, patch)
2018-02-21 03:08 PST, Yusuke Suzuki
no flags
Patch (75.49 KB, patch)
2018-02-21 08:20 PST, Yusuke Suzuki
no flags
Patch (89.18 KB, patch)
2018-02-21 09:08 PST, Yusuke Suzuki
no flags
Patch (89.16 KB, patch)
2018-02-22 19:09 PST, Yusuke Suzuki
no flags
Patch (90.90 KB, patch)
2018-02-23 10:18 PST, Yusuke Suzuki
no flags
Patch (107.48 KB, patch)
2018-02-24 08:23 PST, Yusuke Suzuki
no flags
Patch (109.14 KB, patch)
2018-02-24 08:42 PST, Yusuke Suzuki
no flags
Patch (109.14 KB, patch)
2018-02-24 10:21 PST, Yusuke Suzuki
no flags
Patch (109.33 KB, patch)
2018-03-03 01:07 PST, Yusuke Suzuki
no flags
Patch (109.30 KB, patch)
2018-03-03 07:14 PST, Yusuke Suzuki
jfbastien: review+
commit-queue: commit-queue-
Yusuke Suzuki
Comment 1 2018-02-20 09:31:00 PST
Yusuke Suzuki
Comment 2 2018-02-20 10:23:50 PST
Yusuke Suzuki
Comment 3 2018-02-21 01:06:48 PST
Yusuke Suzuki
Comment 4 2018-02-21 02:18:27 PST
Yusuke Suzuki
Comment 5 2018-02-21 02:33:52 PST
Yusuke Suzuki
Comment 6 2018-02-21 03:08:31 PST
Yusuke Suzuki
Comment 7 2018-02-21 08:20:40 PST
Yusuke Suzuki
Comment 8 2018-02-21 09:08:34 PST
Saam Barati
Comment 9 2018-02-22 18:41:59 PST
Comment on attachment 334385 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334385&action=review > Source/WTF/wtf/UniqueArray.h:44 > + // If it is acceptable, we can just use Vector<T> instead. So this UniqueArray<T> only > + // accepts the type T which does not have non trivial destructor. This allows us to > + // skip calling destructors for N elements. And this allows UniqueArray<T> not to > + // store its N size. Is it worth having a variant that handles non-trivial destructors? Presumably we want those going through bmalloc as well. "type T which does not have non trivial destructor" => "type T which has a trivial destructor"
Yusuke Suzuki
Comment 10 2018-02-22 18:55:09 PST
Comment on attachment 334385 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334385&action=review Thank you >> Source/WTF/wtf/UniqueArray.h:44 >> + // store its N size. > > Is it worth having a variant that handles non-trivial destructors? Presumably we want those going through bmalloc as well. > > "type T which does not have non trivial destructor" => "type T which has a trivial destructor" We can implement it, but in practice, I think it is not so useful. When the class has non-trivial destructors, in WebKit, typically, it is user-defined class. So annotating the class with WTF_MAKE_FAST_ALLOCATED can achieve what we want. What do you think of? I'll fix this typo. > Source/WTF/wtf/UniqueArray.h:49 > + // more larger storage than the `sizeof(U) * size` storage since it want to store Fix "more larger" => "larger"
Yusuke Suzuki
Comment 11 2018-02-22 19:09:19 PST
Created attachment 334499 [details] Patch Update
Yusuke Suzuki
Comment 12 2018-02-23 10:18:58 PST
Yusuke Suzuki
Comment 13 2018-02-23 10:20:53 PST
Comment on attachment 334538 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334538&action=review > Source/WTF/wtf/UniqueArray.h:97 > +} I've added UniqueArray<T> implementation for types which have non-trivial-destructors too.
Yusuke Suzuki
Comment 14 2018-02-24 08:23:49 PST
Yusuke Suzuki
Comment 15 2018-02-24 08:42:06 PST
EWS Watchlist
Comment 16 2018-02-24 10:06:43 PST
Comment on attachment 334563 [details] Patch Attachment 334563 [details] did not pass jsc-ews (mac): Output: http://webkit-queues.webkit.org/results/6653356 New failing tests: stress/ftl-put-by-id-setter-exception-interesting-live-state.js.ftl-eager
Yusuke Suzuki
Comment 17 2018-02-24 10:21:01 PST
Saam Barati
Comment 18 2018-02-26 12:08:40 PST
Comment on attachment 334385 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334385&action=review >>> Source/WTF/wtf/UniqueArray.h:44 >>> + // store its N size. >> >> Is it worth having a variant that handles non-trivial destructors? Presumably we want those going through bmalloc as well. >> >> "type T which does not have non trivial destructor" => "type T which has a trivial destructor" > > We can implement it, but in practice, I think it is not so useful. > When the class has non-trivial destructors, in WebKit, typically, it is user-defined class. > So annotating the class with WTF_MAKE_FAST_ALLOCATED can achieve what we want. > What do you think of? > > I'll fix this typo. I don't follow why that would actually make that happen. Let's say I have a type T that is marked WTF_MAKE_FAST_ALLOCATED, and it has a non trivial destructor. Won't the compiler just end up emitting placement new? E.g, wouldn't std::unique_ptr<T[]>(size) just do something like: T* p = malloc(sizeof(T)*size); for ( i = 0; i < size; ++i) new (&p[i]) T;
Yusuke Suzuki
Comment 19 2018-02-27 00:46:28 PST
Comment on attachment 334385 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334385&action=review >>>> Source/WTF/wtf/UniqueArray.h:44 >>>> + // store its N size. >>> >>> Is it worth having a variant that handles non-trivial destructors? Presumably we want those going through bmalloc as well. >>> >>> "type T which does not have non trivial destructor" => "type T which has a trivial destructor" >> >> We can implement it, but in practice, I think it is not so useful. >> When the class has non-trivial destructors, in WebKit, typically, it is user-defined class. >> So annotating the class with WTF_MAKE_FAST_ALLOCATED can achieve what we want. >> What do you think of? >> >> I'll fix this typo. > > I don't follow why that would actually make that happen. Let's say I have a type T that is marked WTF_MAKE_FAST_ALLOCATED, and it has a non trivial destructor. Won't the compiler just end up emitting placement new? > > E.g, wouldn't std::unique_ptr<T[]>(size) just do something like: > T* p = malloc(sizeof(T)*size); > for ( i = 0; i < size; ++i) > new (&p[i]) T; No. When WTF_MAKE_FAST_ALLOCATED is annotated, the `new[]` operator (not `new` operator. WTF_MAKE_FAST_ALLOCATED defines `new` and `new[]`) is invoked, which is defined by WTF_MAKE_FAST_ALLOCATED. Since it allocates the memory from FastMalloc, we do not use system malloc for that. So, `make_unique<T[]>(size)` works with FastMalloc if T is annotated with WTF_MAKE_FAST_ALLOCATED. BTW, just for the usability, I added makeUniqueArray<T> for T which has a non trivial destructor in the latest patch.
JF Bastien
Comment 20 2018-03-02 10:33:32 PST
Comment on attachment 334564 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334564&action=review > Source/JavaScriptCore/runtime/StructureIDTable.cpp:36 > + , m_table(makeUniqueArray<StructureOrOffset>(s_initialSize)) Seems like a vector would be better here since we realloc below. > Source/WTF/wtf/UniqueArray.h:46 > + using U = typename std::remove_extent<T>::type; Why remove_extent? We don't use UniqueArray<T[]> right? Should it be an error to do so? > Source/WTF/wtf/UniqueArray.h:96 > + return UniqueArrayMaker<std::is_trivially_destructible<typename std::remove_extent<T>::type>::value, T>::make(size); Same here. You also remove_extent and it seems like it should just be an error to get it. So you need std::is_same<T, remove_extent<T>> > Source/WebCore/html/canvas/WebGLRenderingContextBase.cpp:5933 > + i = std::min(m_buffers.size() - 1, i); Can m_buffers.size() be 0? > Source/WebCore/platform/graphics/FormatConverter.h:49 > + m_unpackedIntermediateSrcData = makeUniqueArray<uint8_t>(m_width * MaxNumberOfComponents * MaxBytesPerComponent); Can this overflow? > Source/WebCore/platform/graphics/cg/GraphicsContext3DCG.cpp:481 > + m_formalizedRGBA8Data = makeUniqueArray<uint8_t>(m_imageWidth * m_imageHeight * 4); Can this overflow? > Source/WebKit/Platform/IPC/unix/ConnectionUnix.cpp:149 > + memcpy(attachmentInfo.data(), messageData, sizeof(AttachmentInfo) * attachmentCount); Can this overflow? > Tools/TestWebKitAPI/Tests/WTF/UniqueArray.cpp:33 > +static unsigned numberOfConstrucions { 0 }; typo "constructions" > Tools/TestWebKitAPI/Tests/WTF/UniqueArray.cpp:44 > + } This isn't trivial?
Yusuke Suzuki
Comment 21 2018-03-02 10:57:23 PST
Comment on attachment 334564 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334564&action=review Thanks. >> Source/JavaScriptCore/runtime/StructureIDTable.cpp:36 >> + , m_table(makeUniqueArray<StructureOrOffset>(s_initialSize)) > > Seems like a vector would be better here since we realloc below. This is intentional. In `StructureIDTable::resize` function, we first allocate the table pointer, fill the values, and perform `WTF::storeStoreFence`. If the table pointer is swapped before copying the values, we encounter the issue. (And this order depends on Vector's implementation. And we do not want to limit the Vector's implementation due to StructureIDTable implementaiton.). >> Source/WTF/wtf/UniqueArray.h:46 >> + using U = typename std::remove_extent<T>::type; > > Why remove_extent? We don't use UniqueArray<T[]> right? Should it be an error to do so? OK, I'll insert `static_assert` to ensure `T[]` is not coming here. >> Source/WTF/wtf/UniqueArray.h:96 >> + return UniqueArrayMaker<std::is_trivially_destructible<typename std::remove_extent<T>::type>::value, T>::make(size); > > Same here. You also remove_extent and it seems like it should just be an error to get it. So you need std::is_same<T, remove_extent<T>> Ditto. >> Source/WebCore/html/canvas/WebGLRenderingContextBase.cpp:5933 >> + i = std::min(m_buffers.size() - 1, i); > > Can m_buffers.size() be 0? It would be 0. However, in that case, we never call this `imageBuffer` function. >> Source/WebCore/platform/graphics/FormatConverter.h:49 >> + m_unpackedIntermediateSrcData = makeUniqueArray<uint8_t>(m_width * MaxNumberOfComponents * MaxBytesPerComponent); > > Can this overflow? It would be overflow. Using Checked<> and causing abort would be safer. >> Source/WebCore/platform/graphics/cg/GraphicsContext3DCG.cpp:481 >> + m_formalizedRGBA8Data = makeUniqueArray<uint8_t>(m_imageWidth * m_imageHeight * 4); > > Can this overflow? Ditto. >> Source/WebKit/Platform/IPC/unix/ConnectionUnix.cpp:149 >> + memcpy(attachmentInfo.data(), messageData, sizeof(AttachmentInfo) * attachmentCount); > > Can this overflow? No. In that case, Vector already causes abort. >> Tools/TestWebKitAPI/Tests/WTF/UniqueArray.cpp:33 >> +static unsigned numberOfConstrucions { 0 }; > > typo "constructions" Fixed. >> Tools/TestWebKitAPI/Tests/WTF/UniqueArray.cpp:44 >> + } > > This isn't trivial? Oops, I need to swap `TrivialDestructor` and `NonTrivialDestructor` here.
Yusuke Suzuki
Comment 22 2018-03-03 01:07:44 PST
Yusuke Suzuki
Comment 23 2018-03-03 07:14:25 PST
JF Bastien
Comment 24 2018-03-05 09:37:52 PST
Comment on attachment 334955 [details] Patch r=me
Yusuke Suzuki
Comment 25 2018-03-05 09:49:19 PST
Comment on attachment 334955 [details] Patch Thanks!
WebKit Commit Bot
Comment 26 2018-03-05 09:53:12 PST
Comment on attachment 334955 [details] Patch Rejecting attachment 334955 [details] from commit-queue. Failed to run "['/Volumes/Data/EWS/WebKit/Tools/Scripts/webkit-patch', '--status-host=webkit-queues.webkit.org', '--bot-id=webkit-cq-02', 'apply-attachment', '--no-update', '--non-interactive', 334955, '--port=mac']" exit_code: 2 cwd: /Volumes/Data/EWS/WebKit Last 500 characters of output: ile Source/WebKitLegacy/win/WebPreferences.cpp patching file Source/WebKitLegacy/win/WebView.cpp patching file Tools/ChangeLog Hunk #1 succeeded at 1 with fuzz 3. patching file Tools/TestWebKitAPI/CMakeLists.txt patching file Tools/TestWebKitAPI/TestWebKitAPI.xcodeproj/project.pbxproj patching file Tools/TestWebKitAPI/Tests/WTF/UniqueArray.cpp Failed to run "[u'/Volumes/Data/EWS/WebKit/Tools/Scripts/svn-apply', '--force', '--reviewer', u'JF Bastien']" exit_code: 1 cwd: /Volumes/Data/EWS/WebKit Full output: http://webkit-queues.webkit.org/results/6814363
Yusuke Suzuki
Comment 27 2018-03-05 10:23:33 PST
Comment on attachment 334955 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=334955&action=review > Source/WTF/wtf/Vector.h:85 > new (NotNull, cur) T; Oops, this can be used for POD, and at that time, value is not initialized since it is not `T()`. In `initialize()` function with `bool needsInitialization = true`, we should call `T()` instead of `T` since it *wants* initialization. This is discovered by UniqueArray test. Fixed.
Yusuke Suzuki
Comment 28 2018-03-05 23:25:22 PST
Radar WebKit Bug Importer
Comment 29 2018-03-05 23:26:21 PST
Yusuke Suzuki
Comment 30 2018-03-06 09:38:09 PST
Note You need to log in before you can comment on or make changes to this bug.