| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  | // Copyright (c) 2019 GitHub, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shell/common/gin_helper/destroyable.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  | #include "base/no_destructor.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  | #include "gin/converter.h"
 | 
					
						
							| 
									
										
										
										
											2019-12-05 18:46:34 +09:00
										 |  |  | #include "shell/common/gin_helper/wrappable_base.h"
 | 
					
						
							| 
									
										
										
										
											2024-07-29 12:42:57 -05:00
										 |  |  | #include "v8/include/v8-function.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace gin_helper { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  | v8::Global<v8::FunctionTemplate>* GetDestroyFunc() { | 
					
						
							|  |  |  |   static base::NoDestructor<v8::Global<v8::FunctionTemplate>> destroy_func; | 
					
						
							|  |  |  |   return destroy_func.get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | v8::Global<v8::FunctionTemplate>* GetIsDestroyedFunc() { | 
					
						
							|  |  |  |   static base::NoDestructor<v8::Global<v8::FunctionTemplate>> is_destroyed_func; | 
					
						
							|  |  |  |   return is_destroyed_func.get(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | void DestroyFunc(const v8::FunctionCallbackInfo<v8::Value>& info) { | 
					
						
							| 
									
										
										
										
											2024-07-02 09:51:33 +02:00
										 |  |  |   v8::Local<v8::Object> holder = info.This(); | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  |   if (Destroyable::IsDestroyed(holder)) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-05 18:46:34 +09:00
										 |  |  |   // TODO(zcbenz): gin_helper::Wrappable will be removed.
 | 
					
						
							|  |  |  |   delete static_cast<gin_helper::WrappableBase*>( | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  |       holder->GetAlignedPointerFromInternalField(0)); | 
					
						
							|  |  |  |   holder->SetAlignedPointerInInternalField(0, nullptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void IsDestroyedFunc(const v8::FunctionCallbackInfo<v8::Value>& info) { | 
					
						
							|  |  |  |   info.GetReturnValue().Set(gin::ConvertToV8( | 
					
						
							| 
									
										
										
										
											2024-07-02 09:51:33 +02:00
										 |  |  |       info.GetIsolate(), Destroyable::IsDestroyed(info.This()))); | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // static
 | 
					
						
							|  |  |  | bool Destroyable::IsDestroyed(v8::Local<v8::Object> object) { | 
					
						
							|  |  |  |   // An object is considered destroyed if it has no internal pointer or its
 | 
					
						
							|  |  |  |   // internal has been destroyed.
 | 
					
						
							|  |  |  |   return object->InternalFieldCount() == 0 || | 
					
						
							|  |  |  |          object->GetAlignedPointerFromInternalField(0) == nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // static
 | 
					
						
							|  |  |  | void Destroyable::MakeDestroyable(v8::Isolate* isolate, | 
					
						
							|  |  |  |                                   v8::Local<v8::FunctionTemplate> prototype) { | 
					
						
							|  |  |  |   // Cache the FunctionTemplate of "destroy" and "isDestroyed".
 | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  |   if (GetDestroyFunc()->IsEmpty()) { | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  |     auto templ = v8::FunctionTemplate::New(isolate, DestroyFunc); | 
					
						
							|  |  |  |     templ->RemovePrototype(); | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  |     GetDestroyFunc()->Reset(isolate, templ); | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  |     templ = v8::FunctionTemplate::New(isolate, IsDestroyedFunc); | 
					
						
							|  |  |  |     templ->RemovePrototype(); | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  |     GetIsDestroyedFunc()->Reset(isolate, templ); | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto proto_templ = prototype->PrototypeTemplate(); | 
					
						
							|  |  |  |   proto_templ->Set( | 
					
						
							|  |  |  |       gin::StringToSymbol(isolate, "destroy"), | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  |       v8::Local<v8::FunctionTemplate>::New(isolate, *GetDestroyFunc())); | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  |   proto_templ->Set( | 
					
						
							|  |  |  |       gin::StringToSymbol(isolate, "isDestroyed"), | 
					
						
							| 
									
										
										
										
											2020-03-20 14:15:55 -07:00
										 |  |  |       v8::Local<v8::FunctionTemplate>::New(isolate, *GetIsDestroyedFunc())); | 
					
						
							| 
									
										
										
										
											2019-09-04 11:14:16 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace gin_helper
 |