2015-05-05 10:50:19 +08:00
|
|
|
/*
|
|
|
|
* Created by Rolando Abarca on 3/14/12.
|
|
|
|
* Copyright (c) 2012 Zynga Inc. All rights reserved.
|
2018-01-29 16:25:32 +08:00
|
|
|
* Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
* Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2015-05-05 10:50:19 +08:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2016-03-20 21:53:44 +08:00
|
|
|
#include "scripting/js-bindings/manual/ScriptingCore.h"
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
// Removed in Firefox v27, use 'js/OldDebugAPI.h' instead
|
|
|
|
//#include "jsdbgapi.h"
|
|
|
|
#include "js/OldDebugAPI.h"
|
|
|
|
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-03-20 21:53:44 +08:00
|
|
|
#include "storage/local-storage/LocalStorage.h"
|
|
|
|
#include "scripting/js-bindings/manual/cocos2d_specifics.hpp"
|
|
|
|
#include "scripting/js-bindings/auto/jsb_cocos2dx_auto.hpp"
|
|
|
|
#include "scripting/js-bindings/manual/js_bindings_config.h"
|
2015-08-27 17:55:38 +08:00
|
|
|
|
2016-04-18 15:09:21 +08:00
|
|
|
#include "cocos2d.h" // we used cocos2dVersion() ...
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// for debug socket
|
metal support for cocos2d-x (#19305)
* remove deprecated files
* remove some deprecated codes
* remove more deprecated codes
* remove ui deprecated codes
* remove more deprecated codes
* remove deprecated codes in ccmenuitem
* remove more deprecated codes in ui
* remove more deprecated codes in ui
* remove more deprecated codes in ui
* remove more deprecated codes
* remove more deprecated codes
* remove more deprecated codes
* remove vr related codes and ignore some modules
* remove allocator
* remove some config
* 【Feature】add back-end project file
* [Feature] add back-end file
* add pipeline descriptor and shader cache
* [Feature] support sprite for backend
* [Feature] remove unneeded code
* [Feature] according to es2.0 spec, you must use clamp-to-edge as texture wrap mode, and no mipmapping for non-power-of-two texture
* [Feature] set texture wrap mode to clamp-to-edge, and no mipmapping for non-power-of-two texture
* [Feature] remove macro define to .cpp file
* [Feature] add log info
* [Feature] add PipelineDescriptor for TriangleCommand
* [Feature] add PipelineDescriptor object as member of TriangleCommand
* [Feature] add getPipelineDescriptor method
* add renderbackend
* complete pipeline descriptor
* [Feature] add viewport in RenderCommand
* set viewport when rendrering
* [Feature] occur error when using RendererBackend, to be fixed.
* a workaround to fix black screen on macOS 10.14 (#19090)
* add rendererbackend init function
* fix typo
* [Feature] modify testFile
* [BugFix] modify shader path
* [Feature] set default viewport
* fix projection
* [Feature] modify log info
* [BugFix] change viewport data type to int
* [BugFix] add BindGroup to PipelienDescriptor
* [BugFix] change a_position to vec3 in sprite.vert
* [BugFix] set vertexLayout according to V3F_C4B_T2F structure
* [Feature] revert a_position to vec4
* [Feature] renderer should not use gl codes directly
* [Feature] it's better not use default value parameter
* fix depth test setting
* rendererbackend -> renderer
* clear color and depth at begin
* add metal backend
* metal support normalized attribute
* simplify codes
* update external
* add render pass desctriptor in pipeline descriptor
* fix warnings
* fix crash and memeory leak
* refactor Texture2D
* put pipeline descriptor into render command
* simplify codes
* [Feature] update Sprite
* fix crash when closing app
* [Feature] update SpriteBatchNode and TextureAtlas
* support render texture(not finish)
* [Feature] remove unused code
* make tests work on mac
* fix download-deps path error
* make tests work on iOS
* [Feature] support ttf under normal label effect
* refactor triangle command processing
* let renderer handle more common commands
* refactor backend
* make render texture work
* [Feature] refactor backend for GL
* [Feature]Renaming to make it easy to understand
* [Feature] change warp mode to CLAMP_TO_EDGE
* fix ghost
* simplify visit render queue logic
* support progress timer without rial mode
* support partcile system
* Feature/update label (#149)
* [BugFix] fix compile error
* [Feature] support outline effect in ios
* [Feature] add shader file
* [BugFix] fix begin and end RenderPass
* [Feature] update CustomCommand
* [Feature] revert project.pbxproj
* [Feature] simplify codes
* [BugFix] pack AI88 to RGBA8888 only when outline enable
* [Feature] support shadow effect in Label
* [Feature] support BMFont
* [Feature] support glow effect
* [Feature] simplify shader files
* LabelAtlas work
* handle blend function correctly
* support tile map
* don't share buffer in metal
* alloc buffer size as needed
* support more tilemap
* Merge branch 'minggo/metal-support' into feature/updateLabel
* minggo/metal-support:
support tile map
handle blend function correctly
LabelAtlas work
Feature/update label (#149)
support partcile system
# Conflicts:
# cocos/2d/CCLabel.cpp
# cocos/2d/CCSprite.cpp
# cocos/2d/CCSpriteBatchNode.cpp
# cocos/renderer/CCQuadCommand.cpp
# cocos/renderer/CCQuadCommand.h
* render texture work without saving file
* use global viewport
* grid3d works
* remove grabber
* tiled3d works
* [BugFix] fix label bug
* [Feature] add updateSubData for buffer
* [Feature] remove setVertexCount
* support depth test
* add callback command
* [Feature] add UITest
* [Feature] update UITest
* [Feature] remove unneeded codes
* fix custom command issue
* fix layer color blend issue
* [BugFix] fix iOS compile error
* [Feature] remove unneeded codes
* [Feature] fix updateVertexBuffer
* layerradial works
* add draw test back
* fix batch issue
* fix compiling error
* [BugFix] support ETC1
* [BugFix] get the correct pipelineDescriptor
* [BugFix] skip draw when backendTexture nullptr
* clipping node support
* [Feature] add shader files
* fix stencil issue in metal
* [Feature] update UILayoutTest
* [BugFix] skip drawing when vertexCount is zero
* refactor renderer
* add set global z order for stencil manager commands
* fix warnings caused by type
* remove viewport in render command
* [Feature] fix warnings caused by type
* [BugFix] clear vertexCount and indexCount for CustomComand when needed
* [Feature] update clear for CustomCommand
* ios use metal
* fix viewport issue
* fix LayerColorGradient crash
* [cmake] transport to android and windows (#160)
* save point 1
* compile on windows
* run on android
* revert useless change
* android set CC_ENABLE_CACHE_TEXTURE_DATA to 1
* add initGlew
* fix android crash
* add TODO new-renderer
* review update
* revert onGLFWWindowPosCallback
* fix android compiling error
* Impl progress radial (#162)
* progresstimer add radial impl
* default drawType to element
* dec invoke times of createVertexBuffer (#163)
* support depth/stencil format for gl backend
* simplify progress timer codes
* support motionstreak, effect is wrong
* fix motionstreak issue
* [Feature] update Scissor Test (#161)
* [Feature] update Scissor Test
* [Feature] update ScissorTest
* [Feature] rename function
* [Feature] get constant reference if needed
* [Feature] show render status (#164)
* improve performance
* fix depth state
* fill error that triangle vertex/index number bigger than buffer
* fix compiline error in release mode
* fix buffer conflict between CPU and GPU on iOS/macOS
* Renderer refactor (#165)
* use one vertes/index buffer with opengl
* fix error on windows
* custom command support index format config
* CCLayer: compact vertex data structure
* update comment
* fix doc
* support fast tilemap
* pass index format instead
* fix some wrong effect
* fix render texture error
* fix texture per-element size
* fix texture format error
* BlendFunc type refactor, GLenum -> backend::BlendFactor (#167)
* BlendFunc use backend::BlendFactor as inner field
* update comments
* use int to replace GLenum
* update xcode project fiel
* rename to GLBlendConst
* add ccConstants.h
* update xcode project file
* update copyright
* remove primitive command
* remove CCPrimitive.cpp/.h
* remove deprecated files
* remove unneeded files
* remove multiple view support
* remove multiple view support
* remove the usage of frame buffer in camera
* director don't use frame buffer
* remove FrameBuffer
* remove BatchCommand
* add some api reference
* add physics2d back
* fix crash when close app on mac
* improve render texture
* fix rendertexture issue
* fix rendertexture issue
* simplify codes
* CMake support for mac & ios (#169)
* update cmake
* fix compile error
* update 3rd libs version
* remove CCThread.h/.cpp
* remove ccthread
* use audio engine to implement simple audio engine
* remove unneeded codes
* remove deprecated codes
* remove winrt macro
* remove CC_USE_WIC
* set partcile blend function in more elegant way
* remove unneeded codes
* remove unneeded codes
* cmake works on windows
* update project setting
* improve performance
* GLFloat -> float
* sync v3 cmake improvements into metal-support (#172)
* pick: modern cmake, compile definitions improvement (#19139)
* modern cmake, use target_compile_definitions partly
* simplify macro define, remove USE_*
* modern cmake, macro define
* add physics 2d macro define into ccConfig.h
* remove USE_CHIPMUNK macro in build.gradle
* remove CocosSelectModule.cmake
* shrink useless define
* simplify compile options config, re-add if necessary
* update external for tmp CI test
* un-quote target_compile_options value
* add "-g" parameter only when debug mode
* keep single build type when generator Xcode & VS projecy
* update external for tmp CI tes
* add static_cast<char>(-1), fix -Wc++11-narrowing
* simplify win32 compile define
* not modify code, only improve compile options
# Conflicts:
# .gitignore
# cmake/Modules/CocosConfigDepend.cmake
# cocos/CMakeLists.txt
# external/config.json
# tests/cpp-tests/CMakeLists.txt
* modern cmake, improve cmake_compiler_flags (#19145)
* cmake_compiler_flags
* Fix typo
* Fix typo2
* Remove chanages from Android.mk
* correct lua template cmake build (#19149)
* don't add -Wno-deprecated into jsb target
* correct lua template cmake build
* fix win32 lua template compile error
* prevent cmake in-source-build friendly (#19151)
* pick: Copy resources to "Resources/" on win32 like in linux configuration
* add "/Z7" for cpp-tests on windows
* [cmake] fix iOS xcode property setting failed (#19208)
* fix iOS xcode property setting failed
* use search_depend_libs_recursive at dlls collect
* fix typo
* [cmake] add find_host_library into iOS toolchain file (#19230)
* pick: [lua android] use luajit & template cmake update (#19239)
* increase cmake stability , remove tests/CMakeLists.txt (#19261)
* cmake win32 Precompiled header (#19273)
* Precompiled header
* Fix
* Precompiled header for cocos
* Precompiled header jscocos2d
* Fix for COCOS2D_DEBUG is always 1 on Android (#19291)
Related #19289
* little build fix, tests cpp-tests works on mac
* sync v3 build related codes into metal-support (#173)
* strict initialization for std::array
* remove proj.win32 project configs
* modern cmake, cmake_cleanup_remove_unused_variables (#19146)
* Switch travis CI to xenial (#19207)
* Switch travis CI to xenial
* Remove language: android
* Set language: cpp
* Fix java problem
* Update sdkmanager
* Fix sdkmanger
* next sdkmanager fix
* Remove xenial from android
* revert to sdk-tools-{system}-3859397
* Remove linux cmake install
* Update before-install.sh
* Update .travis.yml
* Simplify install-deps-linux.sh, tested on Ubuntu 16.04 (#19212)
* Simplify install-deps-linux.sh
* Cleanup
* pick: install ninja
* update cocos2d-console submodule
* for metal-support alpha release, we only test cpp
* add HelloCpp into project(Cocos2d-x) for tmp test
* update extenal metal-support-4
* update uniform setting
* [Feature] update BindGroup
* [Feature] empty-test
* [Feature] cpp-test
* [Feature] fix GL compiler error
* [Feature] fix GL crash
* [Feature] empty-test
* [Feature] cpp-tests
* [feature] improve frameRate
* [feature] fix opengl compile error
* [feature] fix opengl compile error
* [BugFix] fix compute maxLocation error
* [Feature] update setting unifrom
* [Feature] fix namespace
* [Feature] remove unneeded code
* [Bugfix] fix project file
* [Feature] update review
* [texture2d] impl texture format support (#175)
* texture update
* update
* update texture
* commit
* compile on windows
* ddd
* rename
* rename methods
* no crash
* save gl
* save
* save
* rename
* move out pixel format convert functions
* metal crash
* update
* update android
* support gles compressed texture format
* support more compress format
* add more conversion methods
* ss
* save
* update conversion methods
* add PVRTC format support
* reformat
* add marco linux
* fix GL marcro
* pvrtc supported only by ios 8.0+
* remove unused cmake
* revert change
* refactor Texture2D::initWithData
* fix conversion log
* refactor Texture2D::initWithData
* remove some OpenGL constants for PVRTC
* add todo
* fix typo
* AutoTest works on mac/iOS by disable part cases, sync v3 bug fix (#174)
* review cpp-tests, and fix part issues on start auto test
* sync png format fix: Node:Particle3D abnormal texture effects #19204
* fix cpp-tests SpritePolygon crash, wrong png format (#19170)
* fix wrong png convert format from sRGB to Gray
* erase plist index if all frames was erased
* test_A8.png have I8 format, fix it
* [CCSpriteCache] allow re-add plist & add testcase (#19175)
* allow re-add plist & add testcase
* remove comments/rename method/update testcase
* fix isSpriteFramesWithFileLoaded & add testcase
* remove used variable
* remove unused variable
* fix double free issues when js/lua-tests exit on iOS (#19236)
* disable part cases, AutoTest works without crash on mac
* update cocos2dx files json, to test cocos new next
* fix spritecache plist parsing issue (#19269)
* [linux] Fix FileUtils::getContents with folder (#19157)
* fix FileUtils::getContents on linux/mac
* use stat.st_mode
* simplify
* [CCFileUtils] win32 getFileSize (#19176)
* win32 getFileSize
* fix stat
* [cpp test-Android]20:FileUtils/2 change title (#19197)
* sync #19200
* sync #19231
* [android lua] improve performance of lua loader (#19234)
* [lua] improve performance of lua loader
* remove cache fix
* Revert "fix spritecache plist parsing issue (#19269)"
This reverts commit f3a85ece4307a7b90816c34489d1ed2c8fd11baf.
* remove win32 project files ref in template.json
* add metal framework lnk ref into cpp template
* test on iOS, and disable part cases
* alBufferData instead of alBufferDataStatic for small audio file on Apple (#19227)
* changes AudioCache to use alBufferData instead of alBufferDataStatic
(also makes test 19 faster to trigger openal bugs faster)
The original problem: CrashIfClientProvidedBogusAudioBufferList
https://github.com/cocos2d/cocos2d-x/issues/18948
is not happening anymore, but there's still a not very frequent issue
that makes OpenAL crash with a call stack like this.
AudioCache::readDataTask > alBufferData > CleanUpDeadBufferList
It happes more frequently when the device is "cold", which means after
half an hour of not using the device (locked).
I could not find the actual source code for iOS OpenAL, so I used the
macOS versions:
https://opensource.apple.com/source/OpenAL/OpenAL-48.7/Source/OpenAL/oalImp.cpp.auto.html
They seem to use CAGuard.h to make sure the dead buffer list
has no threading issues. I'm worried because the CAGuard code I found
has macos and win32 define but no iOS, so I'm not sure. I guess the
iOS version is different and has the guard.
I could not find a place in the code that's unprotected by the locks
except the InitializeBufferMap() which should not be called more than
once from cocos, and there's a workaround in AudioEngine-impl for it.
I reduced the occurence of the CleanUpDeadBufferList crash by moving
the guard in ~AudioCache to cover the alDeleteBuffers call.
* remove hack method "setTimeout" on audio
* AutoTest works on iOS
* support set ios deployment target for root project
* enable all texture2d cases, since Jiang have fixed
* add CCTextureUtils to xcode project file (#176)
* add leak cases for SpriteFrameCache (#177)
* re-add SpriteFrameCache cases
* update template file json
* Update SpriteFrameCacheTest.cpp
* fix compiling error
2019-01-18 15:08:25 +08:00
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32)
|
2015-05-05 10:50:19 +08:00
|
|
|
#include <io.h>
|
|
|
|
#include <WS2tcpip.h>
|
|
|
|
#else
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <thread>
|
|
|
|
#include <iostream>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
|
|
|
|
#ifdef ANDROID
|
|
|
|
#include <android/log.h>
|
|
|
|
#include <jni/JniHelper.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ANDROID
|
|
|
|
#define LOG_TAG "ScriptingCore.cpp"
|
|
|
|
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define LOGD(...) js_log(__VA_ARGS__)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if COCOS2D_DEBUG
|
|
|
|
#define TRACE_DEBUGGER_SERVER(...) CCLOG(__VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define TRACE_DEBUGGER_SERVER(...)
|
2015-12-17 21:53:40 +08:00
|
|
|
#endif // #if COCOS2D_DEBUG
|
2015-05-05 10:50:19 +08:00
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
#error "The functionality of JS controls C++ object's lifecycle isn't stable enough, please don't enable it now."
|
|
|
|
#endif
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
#define BYTE_CODE_FILE_EXT ".jsc"
|
|
|
|
|
|
|
|
using namespace cocos2d;
|
|
|
|
|
|
|
|
static std::string inData;
|
|
|
|
static std::string outData;
|
|
|
|
static std::vector<std::string> g_queue;
|
|
|
|
static std::mutex g_qMutex;
|
|
|
|
static std::mutex g_rwMutex;
|
|
|
|
static int clientSocket = -1;
|
|
|
|
static uint32_t s_nestedLoopLevel = 0;
|
|
|
|
|
|
|
|
// server entry point for the bg thread
|
|
|
|
static void serverEntryPoint(unsigned int port);
|
|
|
|
|
|
|
|
std::unordered_map<std::string, js_type_class_t*> _js_global_type_map;
|
2015-12-17 09:37:44 +08:00
|
|
|
static std::unordered_map<void*, js_proxy_t*> _native_js_global_map;
|
|
|
|
static std::unordered_map<JSObject*, js_proxy_t*> _js_native_global_map;
|
2016-01-18 16:40:00 +08:00
|
|
|
static std::unordered_map<JSObject*, JSObject*> _js_hook_owner_map;
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
static char *_js_log_buf = NULL;
|
|
|
|
|
|
|
|
static std::vector<sc_register_sth> registrationList;
|
|
|
|
|
|
|
|
// name ~> JSScript map
|
2016-06-22 16:18:20 +08:00
|
|
|
static std::unordered_map<std::string, JS::PersistentRootedScript*> filename_script;
|
2015-05-05 10:50:19 +08:00
|
|
|
// port ~> socket map
|
|
|
|
static std::unordered_map<int,int> ports_sockets;
|
|
|
|
|
|
|
|
static void cc_closesocket(int fd)
|
|
|
|
{
|
metal support for cocos2d-x (#19305)
* remove deprecated files
* remove some deprecated codes
* remove more deprecated codes
* remove ui deprecated codes
* remove more deprecated codes
* remove deprecated codes in ccmenuitem
* remove more deprecated codes in ui
* remove more deprecated codes in ui
* remove more deprecated codes in ui
* remove more deprecated codes
* remove more deprecated codes
* remove more deprecated codes
* remove vr related codes and ignore some modules
* remove allocator
* remove some config
* 【Feature】add back-end project file
* [Feature] add back-end file
* add pipeline descriptor and shader cache
* [Feature] support sprite for backend
* [Feature] remove unneeded code
* [Feature] according to es2.0 spec, you must use clamp-to-edge as texture wrap mode, and no mipmapping for non-power-of-two texture
* [Feature] set texture wrap mode to clamp-to-edge, and no mipmapping for non-power-of-two texture
* [Feature] remove macro define to .cpp file
* [Feature] add log info
* [Feature] add PipelineDescriptor for TriangleCommand
* [Feature] add PipelineDescriptor object as member of TriangleCommand
* [Feature] add getPipelineDescriptor method
* add renderbackend
* complete pipeline descriptor
* [Feature] add viewport in RenderCommand
* set viewport when rendrering
* [Feature] occur error when using RendererBackend, to be fixed.
* a workaround to fix black screen on macOS 10.14 (#19090)
* add rendererbackend init function
* fix typo
* [Feature] modify testFile
* [BugFix] modify shader path
* [Feature] set default viewport
* fix projection
* [Feature] modify log info
* [BugFix] change viewport data type to int
* [BugFix] add BindGroup to PipelienDescriptor
* [BugFix] change a_position to vec3 in sprite.vert
* [BugFix] set vertexLayout according to V3F_C4B_T2F structure
* [Feature] revert a_position to vec4
* [Feature] renderer should not use gl codes directly
* [Feature] it's better not use default value parameter
* fix depth test setting
* rendererbackend -> renderer
* clear color and depth at begin
* add metal backend
* metal support normalized attribute
* simplify codes
* update external
* add render pass desctriptor in pipeline descriptor
* fix warnings
* fix crash and memeory leak
* refactor Texture2D
* put pipeline descriptor into render command
* simplify codes
* [Feature] update Sprite
* fix crash when closing app
* [Feature] update SpriteBatchNode and TextureAtlas
* support render texture(not finish)
* [Feature] remove unused code
* make tests work on mac
* fix download-deps path error
* make tests work on iOS
* [Feature] support ttf under normal label effect
* refactor triangle command processing
* let renderer handle more common commands
* refactor backend
* make render texture work
* [Feature] refactor backend for GL
* [Feature]Renaming to make it easy to understand
* [Feature] change warp mode to CLAMP_TO_EDGE
* fix ghost
* simplify visit render queue logic
* support progress timer without rial mode
* support partcile system
* Feature/update label (#149)
* [BugFix] fix compile error
* [Feature] support outline effect in ios
* [Feature] add shader file
* [BugFix] fix begin and end RenderPass
* [Feature] update CustomCommand
* [Feature] revert project.pbxproj
* [Feature] simplify codes
* [BugFix] pack AI88 to RGBA8888 only when outline enable
* [Feature] support shadow effect in Label
* [Feature] support BMFont
* [Feature] support glow effect
* [Feature] simplify shader files
* LabelAtlas work
* handle blend function correctly
* support tile map
* don't share buffer in metal
* alloc buffer size as needed
* support more tilemap
* Merge branch 'minggo/metal-support' into feature/updateLabel
* minggo/metal-support:
support tile map
handle blend function correctly
LabelAtlas work
Feature/update label (#149)
support partcile system
# Conflicts:
# cocos/2d/CCLabel.cpp
# cocos/2d/CCSprite.cpp
# cocos/2d/CCSpriteBatchNode.cpp
# cocos/renderer/CCQuadCommand.cpp
# cocos/renderer/CCQuadCommand.h
* render texture work without saving file
* use global viewport
* grid3d works
* remove grabber
* tiled3d works
* [BugFix] fix label bug
* [Feature] add updateSubData for buffer
* [Feature] remove setVertexCount
* support depth test
* add callback command
* [Feature] add UITest
* [Feature] update UITest
* [Feature] remove unneeded codes
* fix custom command issue
* fix layer color blend issue
* [BugFix] fix iOS compile error
* [Feature] remove unneeded codes
* [Feature] fix updateVertexBuffer
* layerradial works
* add draw test back
* fix batch issue
* fix compiling error
* [BugFix] support ETC1
* [BugFix] get the correct pipelineDescriptor
* [BugFix] skip draw when backendTexture nullptr
* clipping node support
* [Feature] add shader files
* fix stencil issue in metal
* [Feature] update UILayoutTest
* [BugFix] skip drawing when vertexCount is zero
* refactor renderer
* add set global z order for stencil manager commands
* fix warnings caused by type
* remove viewport in render command
* [Feature] fix warnings caused by type
* [BugFix] clear vertexCount and indexCount for CustomComand when needed
* [Feature] update clear for CustomCommand
* ios use metal
* fix viewport issue
* fix LayerColorGradient crash
* [cmake] transport to android and windows (#160)
* save point 1
* compile on windows
* run on android
* revert useless change
* android set CC_ENABLE_CACHE_TEXTURE_DATA to 1
* add initGlew
* fix android crash
* add TODO new-renderer
* review update
* revert onGLFWWindowPosCallback
* fix android compiling error
* Impl progress radial (#162)
* progresstimer add radial impl
* default drawType to element
* dec invoke times of createVertexBuffer (#163)
* support depth/stencil format for gl backend
* simplify progress timer codes
* support motionstreak, effect is wrong
* fix motionstreak issue
* [Feature] update Scissor Test (#161)
* [Feature] update Scissor Test
* [Feature] update ScissorTest
* [Feature] rename function
* [Feature] get constant reference if needed
* [Feature] show render status (#164)
* improve performance
* fix depth state
* fill error that triangle vertex/index number bigger than buffer
* fix compiline error in release mode
* fix buffer conflict between CPU and GPU on iOS/macOS
* Renderer refactor (#165)
* use one vertes/index buffer with opengl
* fix error on windows
* custom command support index format config
* CCLayer: compact vertex data structure
* update comment
* fix doc
* support fast tilemap
* pass index format instead
* fix some wrong effect
* fix render texture error
* fix texture per-element size
* fix texture format error
* BlendFunc type refactor, GLenum -> backend::BlendFactor (#167)
* BlendFunc use backend::BlendFactor as inner field
* update comments
* use int to replace GLenum
* update xcode project fiel
* rename to GLBlendConst
* add ccConstants.h
* update xcode project file
* update copyright
* remove primitive command
* remove CCPrimitive.cpp/.h
* remove deprecated files
* remove unneeded files
* remove multiple view support
* remove multiple view support
* remove the usage of frame buffer in camera
* director don't use frame buffer
* remove FrameBuffer
* remove BatchCommand
* add some api reference
* add physics2d back
* fix crash when close app on mac
* improve render texture
* fix rendertexture issue
* fix rendertexture issue
* simplify codes
* CMake support for mac & ios (#169)
* update cmake
* fix compile error
* update 3rd libs version
* remove CCThread.h/.cpp
* remove ccthread
* use audio engine to implement simple audio engine
* remove unneeded codes
* remove deprecated codes
* remove winrt macro
* remove CC_USE_WIC
* set partcile blend function in more elegant way
* remove unneeded codes
* remove unneeded codes
* cmake works on windows
* update project setting
* improve performance
* GLFloat -> float
* sync v3 cmake improvements into metal-support (#172)
* pick: modern cmake, compile definitions improvement (#19139)
* modern cmake, use target_compile_definitions partly
* simplify macro define, remove USE_*
* modern cmake, macro define
* add physics 2d macro define into ccConfig.h
* remove USE_CHIPMUNK macro in build.gradle
* remove CocosSelectModule.cmake
* shrink useless define
* simplify compile options config, re-add if necessary
* update external for tmp CI test
* un-quote target_compile_options value
* add "-g" parameter only when debug mode
* keep single build type when generator Xcode & VS projecy
* update external for tmp CI tes
* add static_cast<char>(-1), fix -Wc++11-narrowing
* simplify win32 compile define
* not modify code, only improve compile options
# Conflicts:
# .gitignore
# cmake/Modules/CocosConfigDepend.cmake
# cocos/CMakeLists.txt
# external/config.json
# tests/cpp-tests/CMakeLists.txt
* modern cmake, improve cmake_compiler_flags (#19145)
* cmake_compiler_flags
* Fix typo
* Fix typo2
* Remove chanages from Android.mk
* correct lua template cmake build (#19149)
* don't add -Wno-deprecated into jsb target
* correct lua template cmake build
* fix win32 lua template compile error
* prevent cmake in-source-build friendly (#19151)
* pick: Copy resources to "Resources/" on win32 like in linux configuration
* add "/Z7" for cpp-tests on windows
* [cmake] fix iOS xcode property setting failed (#19208)
* fix iOS xcode property setting failed
* use search_depend_libs_recursive at dlls collect
* fix typo
* [cmake] add find_host_library into iOS toolchain file (#19230)
* pick: [lua android] use luajit & template cmake update (#19239)
* increase cmake stability , remove tests/CMakeLists.txt (#19261)
* cmake win32 Precompiled header (#19273)
* Precompiled header
* Fix
* Precompiled header for cocos
* Precompiled header jscocos2d
* Fix for COCOS2D_DEBUG is always 1 on Android (#19291)
Related #19289
* little build fix, tests cpp-tests works on mac
* sync v3 build related codes into metal-support (#173)
* strict initialization for std::array
* remove proj.win32 project configs
* modern cmake, cmake_cleanup_remove_unused_variables (#19146)
* Switch travis CI to xenial (#19207)
* Switch travis CI to xenial
* Remove language: android
* Set language: cpp
* Fix java problem
* Update sdkmanager
* Fix sdkmanger
* next sdkmanager fix
* Remove xenial from android
* revert to sdk-tools-{system}-3859397
* Remove linux cmake install
* Update before-install.sh
* Update .travis.yml
* Simplify install-deps-linux.sh, tested on Ubuntu 16.04 (#19212)
* Simplify install-deps-linux.sh
* Cleanup
* pick: install ninja
* update cocos2d-console submodule
* for metal-support alpha release, we only test cpp
* add HelloCpp into project(Cocos2d-x) for tmp test
* update extenal metal-support-4
* update uniform setting
* [Feature] update BindGroup
* [Feature] empty-test
* [Feature] cpp-test
* [Feature] fix GL compiler error
* [Feature] fix GL crash
* [Feature] empty-test
* [Feature] cpp-tests
* [feature] improve frameRate
* [feature] fix opengl compile error
* [feature] fix opengl compile error
* [BugFix] fix compute maxLocation error
* [Feature] update setting unifrom
* [Feature] fix namespace
* [Feature] remove unneeded code
* [Bugfix] fix project file
* [Feature] update review
* [texture2d] impl texture format support (#175)
* texture update
* update
* update texture
* commit
* compile on windows
* ddd
* rename
* rename methods
* no crash
* save gl
* save
* save
* rename
* move out pixel format convert functions
* metal crash
* update
* update android
* support gles compressed texture format
* support more compress format
* add more conversion methods
* ss
* save
* update conversion methods
* add PVRTC format support
* reformat
* add marco linux
* fix GL marcro
* pvrtc supported only by ios 8.0+
* remove unused cmake
* revert change
* refactor Texture2D::initWithData
* fix conversion log
* refactor Texture2D::initWithData
* remove some OpenGL constants for PVRTC
* add todo
* fix typo
* AutoTest works on mac/iOS by disable part cases, sync v3 bug fix (#174)
* review cpp-tests, and fix part issues on start auto test
* sync png format fix: Node:Particle3D abnormal texture effects #19204
* fix cpp-tests SpritePolygon crash, wrong png format (#19170)
* fix wrong png convert format from sRGB to Gray
* erase plist index if all frames was erased
* test_A8.png have I8 format, fix it
* [CCSpriteCache] allow re-add plist & add testcase (#19175)
* allow re-add plist & add testcase
* remove comments/rename method/update testcase
* fix isSpriteFramesWithFileLoaded & add testcase
* remove used variable
* remove unused variable
* fix double free issues when js/lua-tests exit on iOS (#19236)
* disable part cases, AutoTest works without crash on mac
* update cocos2dx files json, to test cocos new next
* fix spritecache plist parsing issue (#19269)
* [linux] Fix FileUtils::getContents with folder (#19157)
* fix FileUtils::getContents on linux/mac
* use stat.st_mode
* simplify
* [CCFileUtils] win32 getFileSize (#19176)
* win32 getFileSize
* fix stat
* [cpp test-Android]20:FileUtils/2 change title (#19197)
* sync #19200
* sync #19231
* [android lua] improve performance of lua loader (#19234)
* [lua] improve performance of lua loader
* remove cache fix
* Revert "fix spritecache plist parsing issue (#19269)"
This reverts commit f3a85ece4307a7b90816c34489d1ed2c8fd11baf.
* remove win32 project files ref in template.json
* add metal framework lnk ref into cpp template
* test on iOS, and disable part cases
* alBufferData instead of alBufferDataStatic for small audio file on Apple (#19227)
* changes AudioCache to use alBufferData instead of alBufferDataStatic
(also makes test 19 faster to trigger openal bugs faster)
The original problem: CrashIfClientProvidedBogusAudioBufferList
https://github.com/cocos2d/cocos2d-x/issues/18948
is not happening anymore, but there's still a not very frequent issue
that makes OpenAL crash with a call stack like this.
AudioCache::readDataTask > alBufferData > CleanUpDeadBufferList
It happes more frequently when the device is "cold", which means after
half an hour of not using the device (locked).
I could not find the actual source code for iOS OpenAL, so I used the
macOS versions:
https://opensource.apple.com/source/OpenAL/OpenAL-48.7/Source/OpenAL/oalImp.cpp.auto.html
They seem to use CAGuard.h to make sure the dead buffer list
has no threading issues. I'm worried because the CAGuard code I found
has macos and win32 define but no iOS, so I'm not sure. I guess the
iOS version is different and has the guard.
I could not find a place in the code that's unprotected by the locks
except the InitializeBufferMap() which should not be called more than
once from cocos, and there's a workaround in AudioEngine-impl for it.
I reduced the occurence of the CleanUpDeadBufferList crash by moving
the guard in ~AudioCache to cover the alDeleteBuffers call.
* remove hack method "setTimeout" on audio
* AutoTest works on iOS
* support set ios deployment target for root project
* enable all texture2d cases, since Jiang have fixed
* add CCTextureUtils to xcode project file (#176)
* add leak cases for SpriteFrameCache (#177)
* re-add SpriteFrameCache cases
* update template file json
* Update SpriteFrameCacheTest.cpp
* fix compiling error
2019-01-18 15:08:25 +08:00
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32)
|
2015-05-05 10:50:19 +08:00
|
|
|
closesocket(fd);
|
|
|
|
#else
|
|
|
|
close(fd);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ReportException(JSContext *cx)
|
|
|
|
{
|
|
|
|
if (JS_IsExceptionPending(cx)) {
|
|
|
|
if (!JS_ReportPendingException(cx)) {
|
|
|
|
JS_ClearPendingException(cx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
static void onGarbageCollect(JSRuntime* rt, JSGCStatus status, void* data)
|
|
|
|
{
|
|
|
|
/* We finalize any pending toggle refs before doing any garbage collection,
|
|
|
|
* so that we can collect the JS wrapper objects, and in order to minimize
|
|
|
|
* the chances of objects having a pending toggle up queued when they are
|
|
|
|
* garbage collected. */
|
|
|
|
if (status == JSGC_BEGIN)
|
|
|
|
{
|
|
|
|
CCLOGINFO("onGarbageCollect begin, native->js map: %d, js->native map: %d", (int)_native_js_global_map.size(), (int)_js_native_global_map.size());
|
|
|
|
}
|
|
|
|
else if (status == JSGC_END)
|
|
|
|
{
|
|
|
|
CCLOGINFO("onGarbageCollect end, native->js map: %d, js->native map: %d", (int)_native_js_global_map.size(), (int)_js_native_global_map.size());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
static void executeJSFunctionFromReservedSpot(JSContext *cx, JS::HandleObject obj,
|
|
|
|
const JS::HandleValueArray& dataVal, JS::MutableHandleValue retval) {
|
|
|
|
|
|
|
|
JS::RootedValue func(cx, JS_GetReservedSlot(obj, 0));
|
|
|
|
|
|
|
|
if (func.isNullOrUndefined()) { return; }
|
|
|
|
JS::RootedValue thisObj(cx, JS_GetReservedSlot(obj, 1));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if (thisObj.isNullOrUndefined()) {
|
|
|
|
JS_CallFunctionValue(cx, obj, func, dataVal, retval);
|
|
|
|
} else {
|
|
|
|
assert(!thisObj.isPrimitive());
|
2015-11-27 01:26:54 +08:00
|
|
|
JS::RootedObject jsthis(cx, thisObj.toObjectOrNull());
|
|
|
|
JS_CallFunctionValue(cx, jsthis, func, dataVal, retval);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string getTouchesFuncName(EventTouch::EventCode eventCode)
|
|
|
|
{
|
|
|
|
std::string funcName;
|
|
|
|
switch(eventCode)
|
|
|
|
{
|
|
|
|
case EventTouch::EventCode::BEGAN:
|
|
|
|
funcName = "onTouchesBegan";
|
|
|
|
break;
|
|
|
|
case EventTouch::EventCode::ENDED:
|
|
|
|
funcName = "onTouchesEnded";
|
|
|
|
break;
|
|
|
|
case EventTouch::EventCode::MOVED:
|
|
|
|
funcName = "onTouchesMoved";
|
|
|
|
break;
|
|
|
|
case EventTouch::EventCode::CANCELLED:
|
|
|
|
funcName = "onTouchesCancelled";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CCASSERT(false, "Invalid event code!");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return funcName;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string getTouchFuncName(EventTouch::EventCode eventCode)
|
|
|
|
{
|
|
|
|
std::string funcName;
|
|
|
|
switch(eventCode) {
|
|
|
|
case EventTouch::EventCode::BEGAN:
|
|
|
|
funcName = "onTouchBegan";
|
|
|
|
break;
|
|
|
|
case EventTouch::EventCode::ENDED:
|
|
|
|
funcName = "onTouchEnded";
|
|
|
|
break;
|
|
|
|
case EventTouch::EventCode::MOVED:
|
|
|
|
funcName = "onTouchMoved";
|
|
|
|
break;
|
|
|
|
case EventTouch::EventCode::CANCELLED:
|
|
|
|
funcName = "onTouchCancelled";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CCASSERT(false, "Invalid event code!");
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return funcName;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string getMouseFuncName(EventMouse::MouseEventType eventType)
|
|
|
|
{
|
|
|
|
std::string funcName;
|
|
|
|
switch(eventType) {
|
|
|
|
case EventMouse::MouseEventType::MOUSE_DOWN:
|
|
|
|
funcName = "onMouseDown";
|
|
|
|
break;
|
|
|
|
case EventMouse::MouseEventType::MOUSE_UP:
|
|
|
|
funcName = "onMouseUp";
|
|
|
|
break;
|
|
|
|
case EventMouse::MouseEventType::MOUSE_MOVE:
|
|
|
|
funcName = "onMouseMove";
|
|
|
|
break;
|
|
|
|
case EventMouse::MouseEventType::MOUSE_SCROLL:
|
|
|
|
funcName = "onMouseScroll";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CCASSERT(false, "Invalid event code!");
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return funcName;
|
|
|
|
}
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
void removeJSObject(JSContext* cx, cocos2d::Ref* nativeObj)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
auto proxy = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (proxy)
|
2015-12-16 04:27:53 +08:00
|
|
|
{
|
2016-04-07 01:26:46 +08:00
|
|
|
#if ! CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RemoveObjectRoot(cx, &proxy->obj);
|
|
|
|
#endif
|
2015-12-17 09:37:44 +08:00
|
|
|
// remove the proxy here, since this was a "stack" object, not heap
|
|
|
|
// when js_finalize will be called, it will fail, but
|
|
|
|
// the correct solution is to have a new finalize for event
|
|
|
|
jsb_remove_proxy(proxy);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2015-12-16 04:27:53 +08:00
|
|
|
else CCLOG("removeJSObject: BUG: cannot find native object = %p", nativeObj);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::executeJSFunctionWithThisObj(JS::HandleValue thisObj, JS::HandleValue callback)
|
|
|
|
{
|
|
|
|
JS::RootedValue retVal(_cx);
|
|
|
|
executeJSFunctionWithThisObj(thisObj, callback, JS::HandleValueArray::empty(), &retVal);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::executeJSFunctionWithThisObj(JS::HandleValue thisObj,
|
|
|
|
JS::HandleValue callback,
|
|
|
|
const JS::HandleValueArray& vp,
|
|
|
|
JS::MutableHandleValue retVal)
|
|
|
|
{
|
|
|
|
if (!callback.isNullOrUndefined() || !thisObj.isNullOrUndefined())
|
|
|
|
{
|
|
|
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// Very important: The last parameter 'retVal' passed to 'JS_CallFunctionValue' should not be a NULL pointer.
|
|
|
|
// If it's a NULL pointer, crash will be triggered in 'JS_CallFunctionValue'. To find out the reason of this crash is very difficult.
|
|
|
|
// So we have to check the availability of 'retVal'.
|
2017-03-16 09:42:16 +08:00
|
|
|
JS::RootedObject jsthis(_cx, thisObj.toObjectOrNull());
|
|
|
|
JS_CallFunctionValue(_cx, jsthis, callback, vp, retVal);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void js_log(const char *format, ...) {
|
|
|
|
|
|
|
|
if (_js_log_buf == NULL)
|
|
|
|
{
|
|
|
|
_js_log_buf = (char *)calloc(sizeof(char), MAX_LOG_LENGTH+1);
|
|
|
|
_js_log_buf[MAX_LOG_LENGTH] = '\0';
|
|
|
|
}
|
|
|
|
va_list vl;
|
|
|
|
va_start(vl, format);
|
|
|
|
int len = vsnprintf(_js_log_buf, MAX_LOG_LENGTH, format, vl);
|
|
|
|
va_end(vl);
|
|
|
|
if (len > 0)
|
|
|
|
{
|
|
|
|
CCLOG("JS: %s", _js_log_buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JSBCore_platform(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc!=0)
|
|
|
|
{
|
|
|
|
JS_ReportError(cx, "Invalid number of arguments in __getPlatform");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
Application::Platform platform;
|
|
|
|
|
|
|
|
// config.deviceType: Device Type
|
|
|
|
// 'mobile' for any kind of mobile devices, 'desktop' for PCs, 'browser' for Web Browsers
|
|
|
|
// #if (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32) || (CC_TARGET_PLATFORM == CC_PLATFORM_LINUX) || (CC_TARGET_PLATFORM == CC_PLATFORM_MAC)
|
|
|
|
// platform = JS_InternString(_cx, "desktop");
|
|
|
|
// #else
|
|
|
|
platform = Application::getInstance()->getTargetPlatform();
|
|
|
|
// #endif
|
|
|
|
|
|
|
|
args.rval().set(INT_TO_JSVAL((int)platform));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool JSBCore_version(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc!=0)
|
|
|
|
{
|
|
|
|
JS_ReportError(cx, "Invalid number of arguments in __getVersion");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
char version[256];
|
|
|
|
snprintf(version, sizeof(version)-1, "%s", cocos2dVersion());
|
|
|
|
JSString * js_version = JS_InternString(cx, version);
|
|
|
|
|
|
|
|
args.rval().set(STRING_TO_JSVAL(js_version));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool JSBCore_os(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc!=0)
|
|
|
|
{
|
|
|
|
JS_ReportError(cx, "Invalid number of arguments in __getOS");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
|
|
|
|
JSString * os;
|
|
|
|
|
|
|
|
// osx, ios, android, windows, linux, etc..
|
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
|
|
|
|
os = JS_InternString(cx, "iOS");
|
|
|
|
#elif (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
|
|
|
|
os = JS_InternString(cx, "Android");
|
|
|
|
#elif (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32)
|
|
|
|
os = JS_InternString(cx, "Windows");
|
|
|
|
#elif (CC_TARGET_PLATFORM == CC_PLATFORM_LINUX)
|
|
|
|
os = JS_InternString(cx, "Linux");
|
|
|
|
#elif (CC_TARGET_PLATFORM == CC_PLATFORM_MAC)
|
|
|
|
os = JS_InternString(cx, "OS X");
|
|
|
|
os = JS_InternString(cx, "Unknown");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
args.rval().set(STRING_TO_JSVAL(os));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool JSB_cleanScript(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
if (argc != 1)
|
|
|
|
{
|
|
|
|
JS_ReportError(cx, "Invalid number of arguments in JSB_cleanScript");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
JSString *jsPath = args.get(0).toString();
|
|
|
|
JSB_PRECONDITION2(jsPath, cx, false, "Error js file in clean script");
|
|
|
|
JSStringWrapper wrapper(jsPath);
|
|
|
|
ScriptingCore::getInstance()->cleanScript(wrapper.get());
|
|
|
|
|
|
|
|
args.rval().setUndefined();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
bool JSB_core_restartVM(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSB_PRECONDITION2(argc==0, cx, false, "Invalid number of arguments in executeScript");
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
ScriptingCore::getInstance()->reset();
|
|
|
|
args.rval().setUndefined();
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
2016-05-07 14:37:36 +08:00
|
|
|
bool JSB_closeWindow(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
Director::getInstance()->end();
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
void registerDefaultClasses(JSContext* cx, JS::HandleObject global) {
|
|
|
|
// first, try to get the ns
|
|
|
|
JS::RootedValue nsval(cx);
|
|
|
|
JS::RootedObject ns(cx);
|
|
|
|
JS_GetProperty(cx, global, "cc", &nsval);
|
|
|
|
// Not exist, create it
|
|
|
|
if (nsval == JSVAL_VOID)
|
|
|
|
{
|
|
|
|
ns.set(JS_NewObject(cx, NULL, JS::NullPtr(), JS::NullPtr()));
|
|
|
|
nsval = OBJECT_TO_JSVAL(ns);
|
|
|
|
JS_SetProperty(cx, global, "cc", nsval);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ns.set(nsval.toObjectOrNull());
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Javascript controller (__jsc__)
|
|
|
|
//
|
|
|
|
JS::RootedObject proto(cx);
|
|
|
|
JS::RootedObject parent(cx);
|
|
|
|
JS::RootedObject jsc(cx, JS_NewObject(cx, NULL, proto, parent));
|
|
|
|
JS::RootedValue jscVal(cx);
|
|
|
|
jscVal = OBJECT_TO_JSVAL(jsc);
|
|
|
|
JS_SetProperty(cx, global, "__jsc__", jscVal);
|
|
|
|
|
|
|
|
JS_DefineFunction(cx, jsc, "garbageCollect", ScriptingCore::forceGC, 0, JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_ENUMERATE );
|
|
|
|
JS_DefineFunction(cx, jsc, "dumpRoot", ScriptingCore::dumpRoot, 0, JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_ENUMERATE );
|
|
|
|
JS_DefineFunction(cx, jsc, "executeScript", ScriptingCore::executeScript, 1, JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_ENUMERATE );
|
|
|
|
|
|
|
|
// register some global functions
|
2015-12-29 21:28:30 +08:00
|
|
|
JS_DefineFunction(cx, global, "require", ScriptingCore::executeScript, 1, JSPROP_PERMANENT);
|
2016-05-11 11:58:24 +08:00
|
|
|
JS_DefineFunction(cx, global, "log", ScriptingCore::log, 0, JSPROP_PERMANENT);
|
2015-05-05 10:50:19 +08:00
|
|
|
JS_DefineFunction(cx, global, "executeScript", ScriptingCore::executeScript, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, global, "forceGC", ScriptingCore::forceGC, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
JS_DefineFunction(cx, global, "__getPlatform", JSBCore_platform, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, global, "__getOS", JSBCore_os, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, global, "__getVersion", JSBCore_version, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, global, "__restartVM", JSB_core_restartVM, 0, JSPROP_READONLY | JSPROP_PERMANENT | JSPROP_ENUMERATE );
|
|
|
|
JS_DefineFunction(cx, global, "__cleanScript", JSB_cleanScript, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(cx, global, "__isObjectValid", ScriptingCore::isObjectValid, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
2016-05-07 14:37:36 +08:00
|
|
|
JS_DefineFunction(cx, global, "close", JSB_closeWindow, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
static void sc_finalize(JSFreeOp *freeOp, JSObject *obj) {
|
|
|
|
CCLOGINFO("jsbindings: finalizing JS object %p (global class)", obj);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//static JSClass global_class = {
|
|
|
|
// "global", JSCLASS_GLOBAL_FLAGS,
|
|
|
|
// JS_PropertyStub, JS_DeletePropertyStub, JS_PropertyStub, JS_StrictPropertyStub,
|
|
|
|
// JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, sc_finalize,
|
|
|
|
// JSCLASS_NO_OPTIONAL_MEMBERS
|
|
|
|
//};
|
|
|
|
|
|
|
|
static const JSClass global_class = {
|
|
|
|
"global", JSCLASS_GLOBAL_FLAGS,
|
|
|
|
JS_PropertyStub, JS_DeletePropertyStub, JS_PropertyStub, JS_StrictPropertyStub,
|
|
|
|
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, sc_finalize,
|
|
|
|
nullptr, nullptr, nullptr,
|
|
|
|
JS_GlobalObjectTraceHook
|
|
|
|
};
|
|
|
|
|
2015-09-08 11:04:51 +08:00
|
|
|
ScriptingCore* ScriptingCore::getInstance()
|
|
|
|
{
|
|
|
|
static ScriptingCore* instance = nullptr;
|
|
|
|
if (instance == nullptr)
|
2015-12-16 17:06:56 +08:00
|
|
|
instance = new (std::nothrow) ScriptingCore();
|
2015-09-08 11:04:51 +08:00
|
|
|
|
|
|
|
return instance;
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
ScriptingCore::ScriptingCore()
|
|
|
|
: _rt(nullptr)
|
|
|
|
, _cx(nullptr)
|
2016-02-02 23:38:37 +08:00
|
|
|
, _jsInited(false)
|
2016-05-07 14:37:36 +08:00
|
|
|
, _needCleanup(false)
|
2016-08-19 16:28:47 +08:00
|
|
|
, _global(nullptr)
|
|
|
|
, _debugGlobal(nullptr)
|
2018-01-10 14:39:45 +08:00
|
|
|
, _oldCompartment(nullptr)
|
2015-05-05 10:50:19 +08:00
|
|
|
, _callFromScript(false)
|
2016-12-14 09:27:47 +08:00
|
|
|
, _finalizing(nullptr)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
// set utf8 strings internally (we don't need utf16)
|
|
|
|
// XXX: Removed in SpiderMonkey 19.0
|
|
|
|
//JS_SetCStringsAreUTF8();
|
|
|
|
initRegister();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::initRegister()
|
|
|
|
{
|
|
|
|
this->addRegisterCallback(registerDefaultClasses);
|
2015-12-16 17:06:56 +08:00
|
|
|
this->_runLoop = new (std::nothrow) SimpleRunLoop();
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::string_report(JS::HandleValue val) {
|
|
|
|
if (val.isNull()) {
|
|
|
|
LOGD("val : (JSVAL_IS_NULL(val)");
|
|
|
|
// return 1;
|
|
|
|
} else if (val.isBoolean() && false == val.toBoolean()) {
|
|
|
|
LOGD("val : (return value is false");
|
|
|
|
// return 1;
|
|
|
|
} else if (val.isString()) {
|
|
|
|
JSContext* cx = this->getGlobalContext();
|
|
|
|
JS::RootedString str(cx, val.toString());
|
|
|
|
if (str.get()) {
|
|
|
|
LOGD("val : return string is NULL");
|
|
|
|
} else {
|
|
|
|
JSStringWrapper wrapper(str);
|
|
|
|
LOGD("val : return string =\n%s\n", wrapper.get());
|
|
|
|
}
|
|
|
|
} else if (val.isNumber()) {
|
|
|
|
double number = val.toNumber();
|
|
|
|
LOGD("val : return number =\n%f", number);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-14 13:55:17 +08:00
|
|
|
bool ScriptingCore::evalString(const char *string, JS::MutableHandleValue outVal, const char *filename, JSContext* cx, JS::HandleObject global)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::PersistentRootedScript script(cx);
|
|
|
|
|
|
|
|
JS::CompileOptions op(cx);
|
|
|
|
op.setUTF8(true);
|
|
|
|
|
|
|
|
std::string content = string;
|
|
|
|
|
|
|
|
bool ok = false;
|
|
|
|
bool evaluatedOK = false;
|
|
|
|
if (!content.empty())
|
|
|
|
{
|
|
|
|
ok = JS::Compile(cx, global, op, content.c_str(), content.size(), &(script) );
|
|
|
|
}
|
|
|
|
if (ok) {
|
|
|
|
evaluatedOK = JS_ExecuteScript(cx, global, script, outVal);
|
|
|
|
if (false == evaluatedOK) {
|
|
|
|
cocos2d::log("Evaluating %s failed (evaluatedOK == JS_FALSE)", content.c_str());
|
|
|
|
JS_ReportPendingException(cx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cocos2d::log("ScriptingCore:: evaluateScript fail: %s", content.c_str());
|
|
|
|
}
|
|
|
|
return evaluatedOK;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
2015-12-14 13:55:17 +08:00
|
|
|
bool ScriptingCore::evalString(const char *string, JS::MutableHandleValue outVal)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
|
|
|
return evalString(string, outVal, nullptr, _cx, global);
|
2015-12-14 13:55:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::evalString(const char *string)
|
|
|
|
{
|
|
|
|
JS::RootedValue retVal(_cx);
|
|
|
|
return evalString(string, &retVal);
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
void ScriptingCore::start()
|
|
|
|
{
|
2017-03-16 09:42:16 +08:00
|
|
|
_engineStartTime = std::chrono::steady_clock::now();
|
2015-05-05 10:50:19 +08:00
|
|
|
// for now just this
|
|
|
|
createGlobalContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::addRegisterCallback(sc_register_sth callback) {
|
|
|
|
registrationList.push_back(callback);
|
|
|
|
}
|
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
void ScriptingCore::removeAllRoots(JSContext *cx)
|
|
|
|
{
|
|
|
|
// Native -> JS. No need to free "second"
|
|
|
|
_native_js_global_map.clear();
|
|
|
|
|
|
|
|
// JS -> Native: free "second" and "unroot" it.
|
|
|
|
auto it_js = _js_native_global_map.begin();
|
|
|
|
while (it_js != _js_native_global_map.end())
|
|
|
|
{
|
|
|
|
JS::RemoveObjectRoot(cx, &it_js->second->obj);
|
|
|
|
free(it_js->second);
|
|
|
|
it_js = _js_native_global_map.erase(it_js);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2015-12-17 09:37:44 +08:00
|
|
|
_js_native_global_map.clear();
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Just a wrapper around JSPrincipals that allows static construction.
|
|
|
|
class CCJSPrincipals : public JSPrincipals
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit CCJSPrincipals(int rc = 0)
|
|
|
|
: JSPrincipals()
|
|
|
|
{
|
|
|
|
refcount = rc;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static CCJSPrincipals shellTrustedPrincipals(1);
|
|
|
|
|
|
|
|
static bool
|
|
|
|
CheckObjectAccess(JSContext *cx)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSSecurityCallbacks securityCallbacks = {
|
|
|
|
CheckObjectAccess,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
void ScriptingCore::createGlobalContext() {
|
|
|
|
if (_cx && _rt) {
|
|
|
|
ScriptingCore::removeAllRoots(_cx);
|
|
|
|
JS_DestroyContext(_cx);
|
|
|
|
JS_DestroyRuntime(_rt);
|
|
|
|
_cx = NULL;
|
|
|
|
_rt = NULL;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// Start the engine. Added in SpiderMonkey v25
|
2016-02-02 23:38:37 +08:00
|
|
|
if (!_jsInited && !JS_Init())
|
|
|
|
{
|
2015-05-05 10:50:19 +08:00
|
|
|
return;
|
2016-02-02 23:38:37 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_jsInited = true;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-12-21 13:39:34 +08:00
|
|
|
_rt = JS_NewRuntime(32L * 1024L * 1024L);
|
2017-03-16 09:42:16 +08:00
|
|
|
JS_SetGCParameter(_rt, JSGCParamKey::JSGC_MAX_BYTES, 0xffffffff);
|
|
|
|
JS_SetGCParameter(_rt, JSGCParamKey::JSGC_MODE, JSGC_MODE_COMPARTMENT);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
JS_SetTrustedPrincipals(_rt, &shellTrustedPrincipals);
|
|
|
|
JS_SetSecurityCallbacks(_rt, &securityCallbacks);
|
|
|
|
JS_SetNativeStackQuota(_rt, JSB_MAX_STACK_QUOTA);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-12-21 13:39:34 +08:00
|
|
|
_cx = JS_NewContext(_rt, 32 * 1024);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::RuntimeOptionsRef(_rt).setIon(true);
|
|
|
|
JS::RuntimeOptionsRef(_rt).setBaseline(true);
|
|
|
|
|
|
|
|
JS_SetErrorReporter(_cx, ScriptingCore::reportError);
|
|
|
|
#if defined(JS_GC_ZEAL) && defined(DEBUG)
|
2015-11-27 18:27:08 +08:00
|
|
|
JS_SetGCZeal(this->_cx, 2, JS_DEFAULT_ZEAL_FREQ);
|
2015-05-05 10:50:19 +08:00
|
|
|
#endif
|
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
JS_SetGCCallback(_rt, onGarbageCollect, nullptr);
|
|
|
|
|
|
|
|
_global = new (std::nothrow) JS::PersistentRootedObject(_rt, newGlobalObject(_cx, false));
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// Removed in Firefox v34
|
2016-08-19 16:28:47 +08:00
|
|
|
js::SetDefaultObjectForContext(_cx, global);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-07-10 09:49:39 +08:00
|
|
|
runScript("script/jsb_prepare.js");
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-10-12 01:59:44 +08:00
|
|
|
for (auto& callback : registrationList) {
|
2016-08-19 16:28:47 +08:00
|
|
|
callback(_cx, global);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2016-05-07 14:37:36 +08:00
|
|
|
|
|
|
|
_needCleanup = true;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static std::string RemoveFileExt(const std::string& filePath) {
|
|
|
|
size_t pos = filePath.rfind('.');
|
|
|
|
if (0 < pos) {
|
|
|
|
return filePath.substr(0, pos);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return filePath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::PersistentRootedScript* ScriptingCore::getScript(const std::string& path)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
// a) check jsc file first
|
|
|
|
std::string byteCodePath = RemoveFileExt(std::string(path)) + BYTE_CODE_FILE_EXT;
|
|
|
|
if (filename_script.find(byteCodePath) != filename_script.end())
|
|
|
|
return filename_script[byteCodePath];
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// b) no jsc file, check js file
|
|
|
|
std::string fullPath = cocos2d::FileUtils::getInstance()->fullPathForFilename(path);
|
|
|
|
if (filename_script.find(fullPath) != filename_script.end())
|
|
|
|
return filename_script[fullPath];
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-06-23 11:39:23 +08:00
|
|
|
return nullptr;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::PersistentRootedScript* ScriptingCore::compileScript(const std::string& path, JS::HandleObject global, JSContext* cx)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
if (path.empty()) {
|
2016-06-22 16:18:20 +08:00
|
|
|
return nullptr;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
JS::PersistentRootedScript* script = getScript(path);
|
|
|
|
if (script != nullptr) {
|
|
|
|
return script;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2016-08-19 16:28:47 +08:00
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
if (cx == nullptr) {
|
2015-05-05 10:50:19 +08:00
|
|
|
cx = _cx;
|
|
|
|
}
|
2016-08-19 16:28:47 +08:00
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
cocos2d::FileUtils *futil = cocos2d::FileUtils::getInstance();
|
2015-05-05 10:50:19 +08:00
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
script = new (std::nothrow) JS::PersistentRootedScript(cx);
|
|
|
|
if (script == nullptr) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::RootedObject obj(cx, global);
|
2016-06-22 16:18:20 +08:00
|
|
|
bool compileSucceed = false;
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// a) check jsc file first
|
|
|
|
std::string byteCodePath = RemoveFileExt(std::string(path)) + BYTE_CODE_FILE_EXT;
|
|
|
|
|
2016-07-03 23:42:10 +08:00
|
|
|
// Check whether '.jsc' files exist to avoid outputting log which says 'couldn't find .jsc file'.
|
2015-05-05 10:50:19 +08:00
|
|
|
if (futil->isFileExist(byteCodePath))
|
|
|
|
{
|
|
|
|
Data data = futil->getDataFromFile(byteCodePath);
|
|
|
|
if (!data.isNull())
|
|
|
|
{
|
2016-06-22 16:18:20 +08:00
|
|
|
*script = JS_DecodeScript(cx, data.getBytes(), static_cast<uint32_t>(data.getSize()), nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*script) {
|
|
|
|
compileSucceed = true;
|
|
|
|
filename_script[byteCodePath] = script;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// b) no jsc file, check js file
|
2016-06-22 16:18:20 +08:00
|
|
|
if (!(*script))
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
/* Clear any pending exception from previous failed decoding. */
|
|
|
|
ReportException(cx);
|
|
|
|
|
|
|
|
JS::CompileOptions op(cx);
|
|
|
|
op.setUTF8(true);
|
2017-03-16 09:42:16 +08:00
|
|
|
std::string fullPath = futil->fullPathForFilename(path);
|
2015-05-05 10:50:19 +08:00
|
|
|
op.setFileAndLine(fullPath.c_str(), 1);
|
|
|
|
|
|
|
|
bool ok = false;
|
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
|
|
|
|
std::string jsFileContent = futil->getStringFromFile(fullPath);
|
|
|
|
if (!jsFileContent.empty())
|
|
|
|
{
|
2016-06-22 16:18:20 +08:00
|
|
|
ok = JS::Compile(cx, obj, op, jsFileContent.c_str(), jsFileContent.size(), &(*script));
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
#else
|
2016-06-22 16:18:20 +08:00
|
|
|
ok = JS::Compile(cx, obj, op, fullPath.c_str(), &(*script));
|
2015-05-05 10:50:19 +08:00
|
|
|
#endif
|
|
|
|
if (ok) {
|
2016-06-22 16:18:20 +08:00
|
|
|
compileSucceed = true;
|
2015-05-05 10:50:19 +08:00
|
|
|
filename_script[fullPath] = script;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
filename_script[byteCodePath] = script;
|
|
|
|
}
|
2016-06-22 16:18:20 +08:00
|
|
|
|
|
|
|
if (compileSucceed) {
|
|
|
|
return script;
|
|
|
|
} else {
|
2016-08-19 16:28:47 +08:00
|
|
|
LOGD("ScriptingCore:: compileScript fail:%s", path.c_str());
|
2016-06-22 16:18:20 +08:00
|
|
|
CC_SAFE_DELETE(script);
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::cleanScript(const char *path)
|
|
|
|
{
|
|
|
|
std::string byteCodePath = RemoveFileExt(std::string(path)) + BYTE_CODE_FILE_EXT;
|
|
|
|
auto it = filename_script.find(byteCodePath);
|
|
|
|
if (it != filename_script.end())
|
|
|
|
{
|
2016-06-22 16:18:20 +08:00
|
|
|
delete it->second;
|
2015-05-05 10:50:19 +08:00
|
|
|
filename_script.erase(it);
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
std::string fullPath = cocos2d::FileUtils::getInstance()->fullPathForFilename(path);
|
|
|
|
it = filename_script.find(fullPath);
|
|
|
|
if (it != filename_script.end())
|
|
|
|
{
|
2016-06-22 16:18:20 +08:00
|
|
|
delete it->second;
|
2015-05-05 10:50:19 +08:00
|
|
|
filename_script.erase(it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
std::unordered_map<std::string, JS::PersistentRootedScript*>& ScriptingCore::getFileScript()
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
return filename_script;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::cleanAllScript()
|
|
|
|
{
|
|
|
|
filename_script.clear();
|
|
|
|
}
|
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
bool ScriptingCore::runScript(const std::string& path)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
|
|
|
return runScript(path, global, _cx);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
bool ScriptingCore::runScript(const std::string& path, JS::HandleObject global, JSContext* cx)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
if (cx == nullptr) {
|
2015-05-05 10:50:19 +08:00
|
|
|
cx = _cx;
|
|
|
|
}
|
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
auto script = compileScript(path, global, cx);
|
|
|
|
if (script == nullptr) {
|
|
|
|
return false;
|
|
|
|
}
|
2016-08-19 16:28:47 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
bool evaluatedOK = false;
|
|
|
|
if (script) {
|
|
|
|
JS::RootedValue rval(cx);
|
2016-06-22 16:18:20 +08:00
|
|
|
evaluatedOK = JS_ExecuteScript(cx, global, *script, &rval);
|
2015-05-05 10:50:19 +08:00
|
|
|
if (false == evaluatedOK) {
|
2016-08-19 16:28:47 +08:00
|
|
|
cocos2d::log("Evaluating %s failed (evaluatedOK == JS_FALSE)", path.c_str());
|
2015-05-05 10:50:19 +08:00
|
|
|
JS_ReportPendingException(cx);
|
|
|
|
}
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return evaluatedOK;
|
|
|
|
}
|
|
|
|
|
2015-09-19 00:18:28 +08:00
|
|
|
bool ScriptingCore::requireScript(const char *path, JS::MutableHandleValue jsvalRet)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
|
|
|
return requireScript(path, global, _cx, jsvalRet);
|
2015-09-19 00:18:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::requireScript(const char *path, JS::HandleObject global, JSContext* cx, JS::MutableHandleValue jsvalRet)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
if (cx == nullptr)
|
2015-09-19 00:18:28 +08:00
|
|
|
{
|
|
|
|
cx = _cx;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-06-22 16:18:20 +08:00
|
|
|
auto script = compileScript(path, global, cx);
|
|
|
|
|
2015-09-19 00:18:28 +08:00
|
|
|
bool evaluatedOK = false;
|
|
|
|
if (script)
|
|
|
|
{
|
2016-06-22 16:18:20 +08:00
|
|
|
evaluatedOK = JS_ExecuteScript(cx, global, (*script), jsvalRet);
|
2015-09-19 00:18:28 +08:00
|
|
|
if (false == evaluatedOK)
|
|
|
|
{
|
|
|
|
cocos2d::log("(evaluatedOK == JS_FALSE)");
|
|
|
|
JS_ReportPendingException(cx);
|
|
|
|
}
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-09-19 00:18:28 +08:00
|
|
|
return evaluatedOK;
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
void ScriptingCore::reset()
|
|
|
|
{
|
|
|
|
Director::getInstance()->restart();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::restartVM()
|
|
|
|
{
|
|
|
|
cleanup();
|
|
|
|
initRegister();
|
2015-10-16 14:00:06 +08:00
|
|
|
Application::getInstance()->applicationDidFinishLaunching();
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ScriptingCore::~ScriptingCore()
|
|
|
|
{
|
|
|
|
cleanup();
|
2016-02-02 23:38:37 +08:00
|
|
|
JS_ShutDown();
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::cleanup()
|
|
|
|
{
|
2016-05-07 14:37:36 +08:00
|
|
|
if (!_needCleanup) {
|
|
|
|
return;
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
localStorageFree();
|
|
|
|
removeAllRoots(_cx);
|
2016-01-28 01:03:50 +08:00
|
|
|
garbageCollect();
|
2017-03-16 09:42:16 +08:00
|
|
|
|
|
|
|
PoolManager::getInstance()->getCurrentPool()->clear();
|
2016-08-19 16:28:47 +08:00
|
|
|
|
|
|
|
if (_js_log_buf) {
|
|
|
|
free(_js_log_buf);
|
|
|
|
_js_log_buf = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& s : filename_script)
|
|
|
|
{
|
|
|
|
CC_SAFE_DELETE(s.second);
|
|
|
|
}
|
|
|
|
filename_script.clear();
|
|
|
|
registrationList.clear();
|
|
|
|
|
|
|
|
for (auto iter = _js_global_type_map.begin(); iter != _js_global_type_map.end(); ++iter)
|
|
|
|
{
|
|
|
|
delete iter->second->parentProto.ptr();
|
|
|
|
delete iter->second->proto.ptr();
|
|
|
|
}
|
|
|
|
|
|
|
|
CC_SAFE_DELETE(_global);
|
|
|
|
CC_SAFE_DELETE(_debugGlobal);
|
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
JS_LeaveCompartment(_cx, _oldCompartment);
|
|
|
|
_oldCompartment = nullptr;
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if (_cx)
|
|
|
|
{
|
|
|
|
JS_DestroyContext(_cx);
|
|
|
|
_cx = NULL;
|
|
|
|
}
|
2018-01-10 14:39:45 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if (_rt)
|
|
|
|
{
|
|
|
|
JS_DestroyRuntime(_rt);
|
|
|
|
_rt = NULL;
|
|
|
|
}
|
2016-08-19 16:28:47 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
for (auto iter = _js_global_type_map.begin(); iter != _js_global_type_map.end(); ++iter)
|
|
|
|
{
|
|
|
|
free(iter->second->jsclass);
|
|
|
|
free(iter->second);
|
|
|
|
}
|
|
|
|
_js_global_type_map.clear();
|
2016-05-07 14:37:36 +08:00
|
|
|
|
|
|
|
_needCleanup = false;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::reportError(JSContext *cx, const char *message, JSErrorReport *report)
|
|
|
|
{
|
|
|
|
js_log("%s:%u:%s\n",
|
|
|
|
report->filename ? report->filename : "<no filename=\"filename\">",
|
|
|
|
(unsigned int) report->lineno,
|
|
|
|
message);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
bool ScriptingCore::log(JSContext* cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
if (argc > 0) {
|
|
|
|
JSString *string = JS::ToString(cx, args.get(0));
|
|
|
|
if (string) {
|
|
|
|
JSStringWrapper wrapper(string);
|
|
|
|
js_log("%s", wrapper.get());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
args.rval().setUndefined();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
void ScriptingCore::retainScriptObject(cocos2d::Ref* owner, cocos2d::Ref* target)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsbObj(_cx);
|
|
|
|
get_or_create_js_obj(_cx, global, "jsb", &jsbObj);
|
|
|
|
JS::RootedValue jsbVal(_cx, OBJECT_TO_JSVAL(jsbObj));
|
|
|
|
if (jsbVal.isNullOrUndefined())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
js_proxy_t *pOwner = jsb_get_native_proxy(owner);
|
|
|
|
js_proxy_t *pTarget = jsb_get_native_proxy(target);
|
|
|
|
if (!pOwner || !pTarget)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
JS::RootedValue valOwner(_cx, OBJECT_TO_JSVAL(pOwner->obj));
|
|
|
|
JS::RootedValue valTarget(_cx, OBJECT_TO_JSVAL(pTarget->obj));
|
2016-01-24 19:40:44 +08:00
|
|
|
if (valTarget.isPrimitive())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval valArr[2];
|
|
|
|
valArr[0] = valOwner;
|
|
|
|
valArr[1] = valTarget;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
|
|
|
executeFunctionWithOwner(jsbVal, "registerNativeRef", args, &retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::rootScriptObject(cocos2d::Ref* target)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsbObj(_cx);
|
|
|
|
get_or_create_js_obj(_cx, global, "jsb", &jsbObj);
|
|
|
|
JS::RootedValue jsbVal(_cx, OBJECT_TO_JSVAL(jsbObj));
|
|
|
|
if (jsbVal.isNullOrUndefined())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
js_proxy_t *pTarget = jsb_get_native_proxy(target);
|
|
|
|
if (!pTarget)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
JS::RootedValue valTarget(_cx, OBJECT_TO_JSVAL(pTarget->obj));
|
2016-01-24 19:40:44 +08:00
|
|
|
if (valTarget.isPrimitive())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject root(_cx);
|
|
|
|
get_or_create_js_obj(_cx, jsbObj, "_root", &root);
|
|
|
|
JS::RootedValue valRoot(_cx, OBJECT_TO_JSVAL(root));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval valArr[2];
|
|
|
|
valArr[0] = valRoot;
|
|
|
|
valArr[1] = valTarget;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
|
|
|
executeFunctionWithOwner(jsbVal, "registerNativeRef", args, &retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::releaseScriptObject(cocos2d::Ref* owner, cocos2d::Ref* target)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsbObj(_cx);
|
|
|
|
get_or_create_js_obj(_cx, global, "jsb", &jsbObj);
|
|
|
|
JS::RootedValue jsbVal(_cx, OBJECT_TO_JSVAL(jsbObj));
|
|
|
|
if (jsbVal.isNullOrUndefined())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
js_proxy_t *pOwner = jsb_get_native_proxy(owner);
|
|
|
|
js_proxy_t *pTarget = jsb_get_native_proxy(target);
|
|
|
|
if (!pOwner || !pTarget)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
JS::RootedValue valOwner(_cx, OBJECT_TO_JSVAL(pOwner->obj));
|
|
|
|
JS::RootedValue valTarget(_cx, OBJECT_TO_JSVAL(pTarget->obj));
|
2016-01-24 19:40:44 +08:00
|
|
|
if (valTarget.isPrimitive())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval valArr[2];
|
|
|
|
valArr[0] = valOwner;
|
|
|
|
valArr[1] = valTarget;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
|
|
|
executeFunctionWithOwner(jsbVal, "unregisterNativeRef", args, &retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::unrootScriptObject(cocos2d::Ref* target)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsbObj(_cx);
|
|
|
|
get_or_create_js_obj(_cx, global, "jsb", &jsbObj);
|
|
|
|
JS::RootedValue jsbVal(_cx, OBJECT_TO_JSVAL(jsbObj));
|
|
|
|
if (jsbVal.isNullOrUndefined())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
js_proxy_t *pTarget = jsb_get_native_proxy(target);
|
|
|
|
if (!pTarget)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
JS::RootedValue valTarget(_cx, OBJECT_TO_JSVAL(pTarget->obj));
|
2016-01-24 19:40:44 +08:00
|
|
|
if (valTarget.isPrimitive())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject root(_cx);
|
|
|
|
get_or_create_js_obj(_cx, jsbObj, "_root", &root);
|
|
|
|
JS::RootedValue valRoot(_cx, OBJECT_TO_JSVAL(root));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval valArr[2];
|
|
|
|
valArr[0] = valRoot;
|
|
|
|
valArr[1] = valTarget;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(2, valArr);
|
|
|
|
executeFunctionWithOwner(jsbVal, "unregisterNativeRef", args, &retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::releaseAllChildrenRecursive(cocos2d::Node *node)
|
|
|
|
{
|
|
|
|
const Vector<Node*>& children = node->getChildren();
|
|
|
|
for (auto child : children)
|
|
|
|
{
|
|
|
|
releaseScriptObject(node, child);
|
|
|
|
releaseAllChildrenRecursive(child);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::releaseAllNativeRefs(cocos2d::Ref* owner)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject global(_cx, _global->get());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsbObj(_cx);
|
|
|
|
get_or_create_js_obj(_cx, global, "jsb", &jsbObj);
|
|
|
|
JS::RootedValue jsbVal(_cx, OBJECT_TO_JSVAL(jsbObj));
|
|
|
|
if (jsbVal.isNullOrUndefined())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
js_proxy_t *pOwner = jsb_get_native_proxy(owner);
|
|
|
|
if (!pOwner)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
JS::RootedValue valOwner(_cx, OBJECT_TO_JSVAL(pOwner->obj));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval valArr[1];
|
|
|
|
valArr[0] = valOwner;
|
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(1, valArr);
|
|
|
|
executeFunctionWithOwner(jsbVal, "unregisterAllNativeRefs", args, &retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ScriptingCore::removeScriptObjectByObject(Ref* pObj)
|
|
|
|
{
|
|
|
|
auto proxy = jsb_get_native_proxy(pObj);
|
2015-12-17 09:37:44 +08:00
|
|
|
if (proxy)
|
2015-11-27 01:26:54 +08:00
|
|
|
{
|
2015-12-16 04:27:53 +08:00
|
|
|
JSContext *cx = getGlobalContext();
|
2015-12-17 09:37:44 +08:00
|
|
|
JS::RemoveObjectRoot(cx, &proxy->obj);
|
|
|
|
jsb_remove_proxy(proxy);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::setReservedSpot(uint32_t i, JSObject *obj, jsval value) {
|
|
|
|
JS_SetReservedSlot(obj, i, value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::executeScript(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
2016-01-28 01:55:10 +08:00
|
|
|
ScriptingCore *engine = ScriptingCore::getInstance();
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
if (argc >= 1) {
|
2015-11-27 01:26:54 +08:00
|
|
|
JS::RootedValue jsstr(cx, args.get(0));
|
|
|
|
JSString* str = JS::ToString(cx, jsstr);
|
2015-05-05 10:50:19 +08:00
|
|
|
JSStringWrapper path(str);
|
|
|
|
bool res = false;
|
2016-01-15 18:02:11 +08:00
|
|
|
JS::RootedValue jsret(cx);
|
2015-05-05 10:50:19 +08:00
|
|
|
if (argc == 2 && args.get(1).isString()) {
|
|
|
|
JSString* globalName = args.get(1).toString();
|
|
|
|
JSStringWrapper name(globalName);
|
|
|
|
|
2016-01-28 01:55:10 +08:00
|
|
|
JS::RootedObject debugObj(cx, engine->getDebugGlobal());
|
2015-05-05 10:50:19 +08:00
|
|
|
if (debugObj) {
|
2016-01-28 01:55:10 +08:00
|
|
|
res = engine->requireScript(path.get(), debugObj, cx, &jsret);
|
2015-05-05 10:50:19 +08:00
|
|
|
} else {
|
|
|
|
JS_ReportError(cx, "Invalid global object: %s", name.get());
|
2016-01-15 18:02:11 +08:00
|
|
|
args.rval().setUndefined();
|
2015-05-05 10:50:19 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
JS::RootedObject glob(cx, JS::CurrentGlobalOrNull(cx));
|
2016-01-28 01:55:10 +08:00
|
|
|
res = engine->requireScript(path.get(), glob, cx, &jsret);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2016-01-15 18:02:11 +08:00
|
|
|
args.rval().set(jsret);
|
2015-05-05 10:50:19 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
args.rval().setUndefined();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::forceGC(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JSRuntime *rt = JS_GetRuntime(cx);
|
|
|
|
JS_GC(rt);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::dumpRoot(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
// JS_DumpNamedRoots is only available on DEBUG versions of SpiderMonkey.
|
|
|
|
// Mac and Simulator versions were compiled with DEBUG.
|
|
|
|
#if COCOS2D_DEBUG
|
|
|
|
// JSContext *_cx = ScriptingCore::getInstance()->getGlobalContext();
|
|
|
|
// JSRuntime *rt = JS_GetRuntime(_cx);
|
|
|
|
// JS_DumpNamedRoots(rt, dumpNamedRoot, NULL);
|
|
|
|
// JS_DumpHeap(rt, stdout, NULL, JSTRACE_OBJECT, NULL, 2, NULL);
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::pauseSchedulesAndActions(js_proxy_t* p)
|
|
|
|
{
|
|
|
|
JS::RootedObject obj(_cx, p->obj.get());
|
2016-08-19 16:28:47 +08:00
|
|
|
auto arr = JSScheduleWrapper::getTargetForJSObject(obj);
|
2015-05-05 10:50:19 +08:00
|
|
|
if (! arr) return;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
Node* node = (Node*)p->ptr;
|
2016-08-19 16:28:47 +08:00
|
|
|
for(ssize_t i = 0; i < arr->size(); ++i) {
|
|
|
|
if (arr->at(i)) {
|
|
|
|
node->getScheduler()->pauseTarget(arr->at(i));
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ScriptingCore::resumeSchedulesAndActions(js_proxy_t* p)
|
|
|
|
{
|
|
|
|
JS::RootedObject obj(_cx, p->obj.get());
|
2016-08-19 16:28:47 +08:00
|
|
|
auto arr = JSScheduleWrapper::getTargetForJSObject(obj);
|
2015-05-05 10:50:19 +08:00
|
|
|
if (!arr) return;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
Node* node = (Node*)p->ptr;
|
2016-08-19 16:28:47 +08:00
|
|
|
for(ssize_t i = 0; i < arr->size(); ++i) {
|
|
|
|
if (!arr->at(i)) continue;
|
|
|
|
node->getScheduler()->resumeTarget(arr->at(i));
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::cleanupSchedulesAndActions(js_proxy_t* p)
|
|
|
|
{
|
|
|
|
JS::RootedObject obj(_cx, p->obj.get());
|
2016-08-19 16:28:47 +08:00
|
|
|
auto targetArray = JSScheduleWrapper::getTargetForJSObject(obj);
|
|
|
|
if (targetArray)
|
|
|
|
{
|
2015-05-05 10:50:19 +08:00
|
|
|
Node* node = (Node*)p->ptr;
|
2016-08-19 16:28:47 +08:00
|
|
|
auto scheduler = node->getScheduler();
|
|
|
|
for (auto&& target : *targetArray)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
scheduler->unscheduleAllForTarget(target);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
JSScheduleWrapper::removeAllTargetsForJSObject(obj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::isFunctionOverridedInJS(JS::HandleObject obj, const std::string& name, JSNative native)
|
|
|
|
{
|
2016-02-02 23:38:37 +08:00
|
|
|
JS::RootedObject jsobj(_cx, obj);
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::RootedValue value(_cx);
|
2016-02-02 23:38:37 +08:00
|
|
|
bool ok = JS_GetProperty(_cx, jsobj, name.c_str(), &value);
|
2015-05-05 10:50:19 +08:00
|
|
|
if (ok && !value.isNullOrUndefined() && !JS_IsNativeFunction(value.toObjectOrNull(), native))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-27 17:55:38 +08:00
|
|
|
int ScriptingCore::handleActionEvent(void* data)
|
|
|
|
{
|
|
|
|
if (NULL == data)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-08-27 17:55:38 +08:00
|
|
|
ActionObjectScriptData* actionObjectScriptData = static_cast<ActionObjectScriptData*>(data);
|
|
|
|
if (NULL == actionObjectScriptData->nativeObject || NULL == actionObjectScriptData->eventType)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-08-27 17:55:38 +08:00
|
|
|
Action* actionObject = static_cast<Action*>(actionObjectScriptData->nativeObject);
|
|
|
|
int eventType = *((int*)(actionObjectScriptData->eventType));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-08-27 17:55:38 +08:00
|
|
|
js_proxy_t * p = jsb_get_native_proxy(actionObject);
|
|
|
|
if (!p) return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-08-27 17:55:38 +08:00
|
|
|
int ret = 0;
|
|
|
|
JS::RootedValue retval(_cx);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-08-27 17:55:38 +08:00
|
|
|
if (eventType == kActionUpdate)
|
|
|
|
{
|
2015-11-27 01:26:54 +08:00
|
|
|
JS::RootedObject jstarget(_cx, p->obj);
|
|
|
|
if (isFunctionOverridedInJS(jstarget, "update", js_cocos2dx_Action_update))
|
2015-08-27 17:55:38 +08:00
|
|
|
{
|
|
|
|
jsval dataVal = DOUBLE_TO_JSVAL(*((float *)actionObjectScriptData->param));
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "update", 1, &dataVal, &retval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
int ScriptingCore::handleNodeEvent(void* data)
|
|
|
|
{
|
|
|
|
if (NULL == data)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
BasicScriptData* basicScriptData = static_cast<BasicScriptData*>(data);
|
|
|
|
if (NULL == basicScriptData->nativeObject || NULL == basicScriptData->value)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
Node* node = static_cast<Node*>(basicScriptData->nativeObject);
|
|
|
|
int action = *((int*)(basicScriptData->value));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
js_proxy_t * p = jsb_get_native_proxy(node);
|
|
|
|
if (!p) return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
int ret = 0;
|
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval dataVal = INT_TO_JSVAL(1);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-11-27 01:26:54 +08:00
|
|
|
JS::RootedObject jstarget(_cx, p->obj);
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
if (action == kNodeOnEnter)
|
|
|
|
{
|
2015-11-27 01:26:54 +08:00
|
|
|
if (isFunctionOverridedInJS(jstarget, "onEnter", js_cocos2dx_Node_onEnter))
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "onEnter", 1, &dataVal, &retval);
|
|
|
|
}
|
|
|
|
resumeSchedulesAndActions(p);
|
|
|
|
}
|
|
|
|
else if (action == kNodeOnExit)
|
|
|
|
{
|
2015-11-27 01:26:54 +08:00
|
|
|
if (isFunctionOverridedInJS(jstarget, "onExit", js_cocos2dx_Node_onExit))
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "onExit", 1, &dataVal, &retval);
|
|
|
|
}
|
|
|
|
pauseSchedulesAndActions(p);
|
|
|
|
}
|
|
|
|
else if (action == kNodeOnEnterTransitionDidFinish)
|
|
|
|
{
|
2015-11-27 01:26:54 +08:00
|
|
|
if (isFunctionOverridedInJS(jstarget, "onEnterTransitionDidFinish", js_cocos2dx_Node_onEnterTransitionDidFinish))
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "onEnterTransitionDidFinish", 1, &dataVal, &retval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (action == kNodeOnExitTransitionDidStart)
|
|
|
|
{
|
2015-11-27 01:26:54 +08:00
|
|
|
if (isFunctionOverridedInJS(jstarget, "onExitTransitionDidStart", js_cocos2dx_Node_onExitTransitionDidStart))
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "onExitTransitionDidStart", 1, &dataVal, &retval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (action == kNodeOnCleanup) {
|
|
|
|
cleanupSchedulesAndActions(p);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-11-27 01:26:54 +08:00
|
|
|
if (isFunctionOverridedInJS(jstarget, "cleanup", js_cocos2dx_Node_cleanup))
|
2015-07-14 15:35:29 +08:00
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "cleanup", 1, &dataVal, &retval);
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ScriptingCore::handleComponentEvent(void* data)
|
|
|
|
{
|
|
|
|
if (NULL == data)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
BasicScriptData* basicScriptData = static_cast<BasicScriptData*>(data);
|
|
|
|
if (NULL == basicScriptData->nativeObject || NULL == basicScriptData->value)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
Component* node = static_cast<Component*>(basicScriptData->nativeObject);
|
|
|
|
int action = *((int*)(basicScriptData->value));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
js_proxy_t * p = jsb_get_native_proxy(node);
|
|
|
|
if (!p) return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
int ret = 0;
|
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval dataVal = INT_TO_JSVAL(1);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
JS::RootedValue nodeValue(_cx, OBJECT_TO_JSVAL(p->obj));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-06-04 15:59:05 +08:00
|
|
|
if (action == kComponentOnAdd)
|
|
|
|
{
|
2015-06-05 00:03:21 +08:00
|
|
|
ret = executeFunctionWithOwner(nodeValue, "onAdd", 1, &dataVal, &retval);
|
2015-06-04 15:59:05 +08:00
|
|
|
}
|
|
|
|
else if (action == kComponentOnRemove)
|
|
|
|
{
|
2015-06-05 00:03:21 +08:00
|
|
|
ret = executeFunctionWithOwner(nodeValue, "onRemove", 1, &dataVal, &retval);
|
2015-06-04 15:59:05 +08:00
|
|
|
}
|
|
|
|
else if (action == kComponentOnEnter)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2015-06-05 00:03:21 +08:00
|
|
|
ret = executeFunctionWithOwner(nodeValue, "onEnter", 1, &dataVal, &retval);
|
2016-08-19 16:28:47 +08:00
|
|
|
resumeSchedulesAndActions(p);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
else if (action == kComponentOnExit)
|
|
|
|
{
|
2015-06-05 00:03:21 +08:00
|
|
|
ret = executeFunctionWithOwner(nodeValue, "onExit", 1, &dataVal, &retval);
|
2016-08-19 16:28:47 +08:00
|
|
|
pauseSchedulesAndActions(p);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
else if (action == kComponentOnUpdate)
|
|
|
|
{
|
2015-06-05 00:03:21 +08:00
|
|
|
ret = executeFunctionWithOwner(nodeValue, "update", 1, &dataVal, &retval);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleTouchesEvent(void* nativeObj, cocos2d::EventTouch::EventCode eventCode, const std::vector<cocos2d::Touch*>& touches, cocos2d::Event* event)
|
|
|
|
{
|
|
|
|
JS::RootedValue ret(_cx);
|
|
|
|
return handleTouchesEvent(nativeObj, eventCode, touches, event, &ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleTouchesEvent(void* nativeObj, cocos2d::EventTouch::EventCode eventCode, const std::vector<cocos2d::Touch*>& touches, cocos2d::Event* event, JS::MutableHandleValue jsvalRet)
|
|
|
|
{
|
|
|
|
bool ret = false;
|
|
|
|
std::string funcName = getTouchesFuncName(eventCode);
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsretArr(_cx, JS_NewArrayObject(_cx, 0));
|
2015-05-05 10:50:19 +08:00
|
|
|
int count = 0;
|
2015-12-11 14:15:44 +08:00
|
|
|
|
|
|
|
js_type_class_t *typeClassEvent = nullptr;
|
|
|
|
js_type_class_t *typeClassTouch = nullptr;
|
|
|
|
|
|
|
|
if (touches.size()>0)
|
|
|
|
typeClassTouch = js_get_type_from_native<cocos2d::Touch>(touches[0]);
|
2015-12-17 21:53:40 +08:00
|
|
|
typeClassEvent = js_get_type_from_native<cocos2d::EventTouch>((cocos2d::EventTouch*)event);
|
2015-12-11 14:15:44 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
for (const auto& touch : touches)
|
|
|
|
{
|
2016-12-21 13:39:34 +08:00
|
|
|
JS::RootedValue jsret(_cx, OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, touch, typeClassTouch)));
|
2015-12-17 21:53:40 +08:00
|
|
|
if (!JS_SetElement(_cx, jsretArr, count, jsret))
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
js_proxy_t* p = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (p)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
jsval dataVal[2];
|
|
|
|
dataVal[0] = OBJECT_TO_JSVAL(jsretArr);
|
2016-12-21 13:39:34 +08:00
|
|
|
dataVal[1] = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, event, typeClassEvent));
|
2015-05-05 10:50:19 +08:00
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), funcName.c_str(), 2, dataVal, jsvalRet);
|
2017-03-16 09:42:16 +08:00
|
|
|
// event is created on the heap and its destructor won't be invoked, so we need to remove JS object manually
|
|
|
|
removeJSObject(_cx, event);
|
2015-12-11 14:15:44 +08:00
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleTouchEvent(void* nativeObj, cocos2d::EventTouch::EventCode eventCode, cocos2d::Touch* touch, cocos2d::Event* event)
|
|
|
|
{
|
|
|
|
JS::RootedValue ret(_cx);
|
|
|
|
return handleTouchEvent(nativeObj, eventCode, touch, event, &ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleTouchEvent(void* nativeObj, cocos2d::EventTouch::EventCode eventCode, cocos2d::Touch* touch, cocos2d::Event* event, JS::MutableHandleValue jsvalRet)
|
|
|
|
{
|
|
|
|
std::string funcName = getTouchFuncName(eventCode);
|
|
|
|
bool ret = false;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
js_proxy_t * p = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (p)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2015-12-11 14:15:44 +08:00
|
|
|
js_type_class_t *typeClassTouch = js_get_type_from_native<cocos2d::Touch>(touch);
|
2015-12-17 21:53:40 +08:00
|
|
|
js_type_class_t *typeClassEvent = js_get_type_from_native<cocos2d::EventTouch>((cocos2d::EventTouch*)event);
|
2015-12-11 14:15:44 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
jsval dataVal[2];
|
2016-12-21 13:39:34 +08:00
|
|
|
dataVal[0] = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, touch, typeClassTouch));
|
|
|
|
dataVal[1] = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, event, typeClassEvent));
|
2015-12-11 14:15:44 +08:00
|
|
|
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), funcName.c_str(), 2, dataVal, jsvalRet);
|
2017-03-16 09:42:16 +08:00
|
|
|
// event is created on the heap and its destructor won't be invoked, so we need to remove JS object manually
|
|
|
|
removeJSObject(_cx, event);
|
2015-12-11 14:15:44 +08:00
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleMouseEvent(void* nativeObj, cocos2d::EventMouse::MouseEventType eventType, cocos2d::Event* event)
|
|
|
|
{
|
|
|
|
JS::RootedValue ret(_cx);
|
|
|
|
return handleMouseEvent(nativeObj, eventType, event, &ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleMouseEvent(void* nativeObj, cocos2d::EventMouse::MouseEventType eventType, cocos2d::Event* event, JS::MutableHandleValue jsvalRet)
|
|
|
|
{
|
|
|
|
std::string funcName = getMouseFuncName(eventType);
|
|
|
|
bool ret = false;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
js_proxy_t * p = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (p)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2015-12-17 21:53:40 +08:00
|
|
|
js_type_class_t *typeClass = js_get_type_from_native<cocos2d::EventMouse>((cocos2d::EventMouse*)event);
|
2016-12-21 13:39:34 +08:00
|
|
|
jsval dataVal = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, event, typeClass));
|
2015-12-17 09:37:44 +08:00
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), funcName.c_str(), 1, &dataVal, jsvalRet);
|
2017-03-16 09:42:16 +08:00
|
|
|
// event is created on the heap and its destructor won't be invoked, so we need to remove JS object manually
|
2015-12-17 09:37:44 +08:00
|
|
|
removeJSObject(_cx, event);
|
|
|
|
}
|
|
|
|
else CCLOG("ScriptingCore::handleMouseEvent native proxy NOT found");
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
bool ScriptingCore::executeFunctionWithObjectData(void* nativeObj, const char *name, JSObject *obj)
|
|
|
|
{
|
2015-05-05 10:50:19 +08:00
|
|
|
js_proxy_t * p = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (!p) return false;
|
|
|
|
|
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
jsval dataVal = OBJECT_TO_JSVAL(obj);
|
|
|
|
|
|
|
|
executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), name, 1, &dataVal, &retval);
|
|
|
|
if (retval.isNull()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (retval.isBoolean()) {
|
|
|
|
return retval.toBoolean();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::executeFunctionWithOwner(jsval owner, const char *name, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(argc, vp);
|
|
|
|
JS::RootedValue rval(_cx);
|
|
|
|
return executeFunctionWithOwner(owner, name, args, &rval);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::executeFunctionWithOwner(jsval owner, const char *name, uint32_t argc, jsval *vp, JS::MutableHandleValue retVal)
|
|
|
|
{
|
|
|
|
//should not use CallArgs here, use HandleValueArray instead !!
|
|
|
|
//because the "jsval* vp" is not the standard format from JSNative, the array doesn't contain this and retval
|
|
|
|
//JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(argc, vp);
|
|
|
|
return executeFunctionWithOwner(owner, name, args, retVal);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::executeFunctionWithOwner(jsval owner, const char *name, const JS::HandleValueArray& args)
|
|
|
|
{
|
|
|
|
JS::RootedValue ret(_cx);
|
|
|
|
return executeFunctionWithOwner(owner, name, args, &ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::executeFunctionWithOwner(jsval owner, const char *name, const JS::HandleValueArray& args, JS::MutableHandleValue retVal)
|
|
|
|
{
|
|
|
|
bool bRet = false;
|
2017-03-16 09:42:16 +08:00
|
|
|
bool hasFunc;
|
2015-05-05 10:50:19 +08:00
|
|
|
JSContext* cx = this->_cx;
|
2017-03-16 09:42:16 +08:00
|
|
|
JS::RootedValue funcVal(cx);
|
2015-11-27 01:26:54 +08:00
|
|
|
JS::RootedValue ownerval(cx, owner);
|
|
|
|
JS::RootedObject obj(cx, ownerval.toObjectOrNull());
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
do
|
|
|
|
{
|
2017-03-16 09:42:16 +08:00
|
|
|
if (JS_HasProperty(cx, obj, name, &hasFunc) && hasFunc) {
|
|
|
|
if (!JS_GetProperty(cx, obj, name, &funcVal)) {
|
2015-05-05 10:50:19 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-03-16 09:42:16 +08:00
|
|
|
if (funcVal == JSVAL_VOID) {
|
2015-05-05 10:50:19 +08:00
|
|
|
break;
|
2016-04-18 15:09:21 +08:00
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
|
2017-03-16 09:42:16 +08:00
|
|
|
bRet = JS_CallFunctionValue(cx, obj, funcVal, args, retVal);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}while(0);
|
|
|
|
return bRet;
|
|
|
|
}
|
|
|
|
|
2016-11-04 11:58:48 +08:00
|
|
|
std::chrono::steady_clock::time_point ScriptingCore::getEngineStartTime() const
|
|
|
|
{
|
|
|
|
return _engineStartTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-24 09:14:36 +08:00
|
|
|
bool ScriptingCore::handleKeyboardEvent(void* nativeObj, cocos2d::EventKeyboard::KeyCode keyCode, bool isPressed, cocos2d::Event* event)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
js_proxy_t * p = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (nullptr == p)
|
|
|
|
return false;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
bool ret = false;
|
2015-12-11 14:15:44 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
js_type_class_t *typeClass = js_get_type_from_native<cocos2d::EventKeyboard>((cocos2d::EventKeyboard*)event);
|
2015-05-05 10:50:19 +08:00
|
|
|
jsval args[2] = {
|
|
|
|
int32_to_jsval(_cx, (int32_t)keyCode),
|
2016-12-21 13:39:34 +08:00
|
|
|
OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, event, typeClass))
|
2015-05-05 10:50:19 +08:00
|
|
|
};
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if (isPressed)
|
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "_onKeyPressed", 2, args);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "_onKeyReleased", 2, args);
|
|
|
|
}
|
|
|
|
|
|
|
|
removeJSObject(_cx, event);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::handleFocusEvent(void* nativeObj, cocos2d::ui::Widget* widgetLoseFocus, cocos2d::ui::Widget* widgetGetFocus)
|
|
|
|
{
|
|
|
|
js_proxy_t * p = jsb_get_native_proxy(nativeObj);
|
|
|
|
if (nullptr == p)
|
|
|
|
return false;
|
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
js_type_class_t *typeClass = js_get_type_from_native<cocos2d::ui::Widget>(widgetLoseFocus);
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
jsval args[2] = {
|
2015-12-17 21:53:40 +08:00
|
|
|
OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, widgetLoseFocus, typeClass, "cocos2d::ui::Widget")),
|
|
|
|
OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(_cx, widgetGetFocus, typeClass, "cocos2d::ui::Widget"))
|
2015-05-05 10:50:19 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
bool ret = executeFunctionWithOwner(OBJECT_TO_JSVAL(p->obj), "onFocusChanged", 2, args);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ScriptingCore::executeCustomTouchesEvent(EventTouch::EventCode eventType,
|
|
|
|
const std::vector<Touch*>& touches, JSObject *obj)
|
|
|
|
{
|
|
|
|
std::string funcName = getTouchesFuncName(eventType);
|
|
|
|
|
|
|
|
JS::RootedObject jsretArr(_cx, JS_NewArrayObject(this->_cx, 0));
|
|
|
|
int count = 0;
|
|
|
|
for (auto& touch : touches)
|
|
|
|
{
|
2015-12-11 14:15:44 +08:00
|
|
|
js_type_class_t *typeClass = js_get_type_from_native<cocos2d::Touch>(touch);
|
|
|
|
|
2016-12-21 13:39:34 +08:00
|
|
|
jsval jsret = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(this->_cx, touch, typeClass));
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::RootedValue jsval(_cx, jsret);
|
|
|
|
if (!JS_SetElement(this->_cx, jsretArr, count, jsval)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
jsval jsretArrVal = OBJECT_TO_JSVAL(jsretArr);
|
|
|
|
executeFunctionWithOwner(OBJECT_TO_JSVAL(obj), funcName.c_str(), 1, &jsretArrVal);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
int ScriptingCore::executeCustomTouchEvent(EventTouch::EventCode eventType, Touch *touch, JSObject *obj)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
JS::RootedValue retval(_cx);
|
|
|
|
std::string funcName = getTouchFuncName(eventType);
|
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
js_type_class_t *typeClass = js_get_type_from_native<cocos2d::Touch>(touch);
|
2016-12-21 13:39:34 +08:00
|
|
|
jsval jsTouch = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(this->_cx, touch, typeClass));
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
executeFunctionWithOwner(OBJECT_TO_JSVAL(obj), funcName.c_str(), 1, &jsTouch, &retval);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int ScriptingCore::executeCustomTouchEvent(EventTouch::EventCode eventType,
|
2015-12-11 14:15:44 +08:00
|
|
|
Touch *touch, JSObject *obj,
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::MutableHandleValue retval)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
std::string funcName = getTouchFuncName(eventType);
|
|
|
|
|
2015-12-11 14:15:44 +08:00
|
|
|
js_type_class_t *typeClass = js_get_type_from_native<cocos2d::Touch>(touch);
|
2016-12-21 13:39:34 +08:00
|
|
|
jsval jsTouch = OBJECT_TO_JSVAL(jsb_get_or_create_weak_jsobject(this->_cx, touch, typeClass));
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
executeFunctionWithOwner(OBJECT_TO_JSVAL(obj), funcName.c_str(), 1, &jsTouch, retval);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-03-16 09:42:16 +08:00
|
|
|
int ScriptingCore::executeGlobalFunction(const char* functionName)
|
|
|
|
{
|
|
|
|
std::string evalStr = functionName;
|
|
|
|
JS::RootedValue globalVal(_cx, OBJECT_TO_JSVAL(_global->get()));
|
|
|
|
return executeFunctionWithOwner(globalVal, functionName, 0, NULL);
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
int ScriptingCore::sendEvent(ScriptEvent* evt)
|
|
|
|
{
|
|
|
|
if (NULL == evt)
|
|
|
|
return 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// special type, can't use this code after JSAutoCompartment
|
|
|
|
if (evt->type == kRestartGame)
|
|
|
|
{
|
|
|
|
restartVM();
|
|
|
|
return 0;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
switch (evt->type)
|
|
|
|
{
|
|
|
|
case kNodeEvent:
|
|
|
|
{
|
|
|
|
return handleNodeEvent(evt->data);
|
|
|
|
}
|
|
|
|
break;
|
2015-08-27 17:55:38 +08:00
|
|
|
case kScriptActionEvent:
|
|
|
|
{
|
|
|
|
return handleActionEvent(evt->data);
|
|
|
|
}
|
|
|
|
break;
|
2015-05-05 10:50:19 +08:00
|
|
|
case kMenuClickedEvent:
|
|
|
|
break;
|
|
|
|
case kTouchEvent:
|
|
|
|
{
|
|
|
|
TouchScriptData* data = (TouchScriptData*)evt->data;
|
|
|
|
return handleTouchEvent(data->nativeObject, data->actionType, data->touch, data->event);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kTouchesEvent:
|
|
|
|
{
|
|
|
|
TouchesScriptData* data = (TouchesScriptData*)evt->data;
|
|
|
|
return handleTouchesEvent(data->nativeObject, data->actionType, data->touches, data->event);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kComponentEvent:
|
|
|
|
{
|
|
|
|
return handleComponentEvent(evt->data);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
CCASSERT(false, "Invalid script event.");
|
|
|
|
break;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::parseConfig(ConfigType type, const std::string &str)
|
|
|
|
{
|
|
|
|
jsval args[2];
|
|
|
|
args[0] = int32_to_jsval(_cx, static_cast<int>(type));
|
|
|
|
args[1] = std_string_to_jsval(_cx, str);
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedValue globalVal(_cx, OBJECT_TO_JSVAL(_global->get()));
|
|
|
|
return (true == executeFunctionWithOwner(globalVal, "__onParseConfig", 2, args));
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptingCore::isObjectValid(JSContext *cx, uint32_t argc, jsval *vp)
|
|
|
|
{
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
if (argc == 1) {
|
2018-01-10 14:39:45 +08:00
|
|
|
if (!args[0].isObject())
|
|
|
|
{
|
|
|
|
args.rval().setBoolean(false);
|
|
|
|
return true;
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::RootedObject tmpObj(cx, args.get(0).toObjectOrNull());
|
|
|
|
js_proxy_t *proxy = jsb_get_js_proxy(tmpObj);
|
|
|
|
if (proxy && proxy->ptr) {
|
|
|
|
args.rval().set(JSVAL_TRUE);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
args.rval().set(JSVAL_FALSE);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
JS_ReportError(cx, "Invalid number of arguments: %d. Expecting: 1", argc);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-03 06:45:13 +08:00
|
|
|
void ScriptingCore::rootObject(Ref* ref)
|
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
auto proxy = jsb_get_native_proxy(ref);
|
|
|
|
if (proxy) {
|
2015-12-04 06:49:23 +08:00
|
|
|
JSContext *cx = getGlobalContext();
|
2015-12-17 09:37:44 +08:00
|
|
|
JS::AddNamedObjectRoot(cx, &proxy->obj, typeid(*ref).name());
|
2015-12-16 04:27:53 +08:00
|
|
|
ref->_rooted = true;
|
2015-12-04 06:49:23 +08:00
|
|
|
}
|
2015-12-24 00:54:00 +08:00
|
|
|
else CCLOG("rootObject: BUG. native not found: %p (%s)", ref, typeid(*ref).name());
|
2015-12-03 06:45:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::unrootObject(Ref* ref)
|
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
auto proxy = jsb_get_native_proxy(ref);
|
|
|
|
if (proxy) {
|
2015-12-04 06:49:23 +08:00
|
|
|
JSContext *cx = getGlobalContext();
|
2015-12-17 09:37:44 +08:00
|
|
|
JS::RemoveObjectRoot(cx, &proxy->obj);
|
2015-12-16 04:27:53 +08:00
|
|
|
ref->_rooted = false;
|
2015-12-04 06:49:23 +08:00
|
|
|
}
|
2015-12-24 00:54:00 +08:00
|
|
|
else CCLOG("unrootObject: BUG. native not found: %p (%s)", ref, typeid(*ref).name());
|
|
|
|
}
|
|
|
|
|
2017-03-16 09:42:16 +08:00
|
|
|
void ScriptingCore::removeObjectProxy(Ref* obj)
|
|
|
|
{
|
|
|
|
auto proxy = jsb_get_native_proxy(obj);
|
|
|
|
if (proxy)
|
|
|
|
{
|
|
|
|
#if ! CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
JS::RemoveObjectRoot(_cx, &proxy->obj);
|
|
|
|
#endif
|
|
|
|
// remove the proxy here, since this was a "stack" object, not heap
|
|
|
|
// when js_finalize will be called, it will fail, but
|
|
|
|
// the correct solution is to have a new finalize for event
|
|
|
|
jsb_remove_proxy(proxy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-24 00:54:00 +08:00
|
|
|
void ScriptingCore::garbageCollect()
|
|
|
|
{
|
|
|
|
#if CC_TARGET_PLATFORM != CC_PLATFORM_WIN32
|
|
|
|
auto runtime = JS_GetRuntime(_cx);
|
|
|
|
// twice: yep, call it twice since this is a generational GC
|
|
|
|
// and we want to collect as much as possible when this is being called
|
|
|
|
// from replaceScene().
|
|
|
|
JS_GC(runtime);
|
|
|
|
JS_GC(runtime);
|
|
|
|
#endif
|
2015-12-03 06:45:13 +08:00
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
#pragma mark - Debug
|
|
|
|
|
|
|
|
void SimpleRunLoop::update(float dt)
|
|
|
|
{
|
2016-05-24 15:59:00 +08:00
|
|
|
std::string message;
|
|
|
|
size_t messageCount = 0;
|
|
|
|
while (true)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
g_qMutex.lock();
|
2016-05-24 15:59:00 +08:00
|
|
|
messageCount = g_queue.size();
|
|
|
|
if (messageCount > 0)
|
|
|
|
{
|
|
|
|
auto first = g_queue.begin();
|
|
|
|
message = *first;
|
|
|
|
g_queue.erase(first);
|
|
|
|
--messageCount;
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
g_qMutex.unlock();
|
|
|
|
|
2016-05-24 15:59:00 +08:00
|
|
|
if (!message.empty())
|
|
|
|
ScriptingCore::getInstance()->debugProcessInput(message);
|
2016-05-26 16:29:54 +08:00
|
|
|
|
|
|
|
if (messageCount == 0)
|
|
|
|
break;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::debugProcessInput(const std::string& str)
|
|
|
|
{
|
|
|
|
JSString* jsstr = JS_NewStringCopyZ(_cx, str.c_str());
|
|
|
|
jsval argv = STRING_TO_JSVAL(jsstr);
|
|
|
|
JS::RootedValue outval(_cx);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject debugGlobal(_cx, _debugGlobal->get());
|
2015-11-27 01:26:54 +08:00
|
|
|
JS_CallFunctionName(_cx, debugGlobal, "processInput", JS::HandleValueArray::fromMarkedLocation(1, &argv), &outval);
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool NS_ProcessNextEvent()
|
|
|
|
{
|
2016-05-24 15:59:00 +08:00
|
|
|
std::string message;
|
|
|
|
size_t messageCount = 0;
|
|
|
|
while (true)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
g_qMutex.lock();
|
2016-05-24 15:59:00 +08:00
|
|
|
messageCount = g_queue.size();
|
|
|
|
if (messageCount > 0)
|
|
|
|
{
|
|
|
|
auto first = g_queue.begin();
|
|
|
|
message = *first;
|
|
|
|
g_queue.erase(first);
|
|
|
|
--messageCount;
|
|
|
|
}
|
2015-05-05 10:50:19 +08:00
|
|
|
g_qMutex.unlock();
|
|
|
|
|
2016-05-24 15:59:00 +08:00
|
|
|
if (!message.empty())
|
|
|
|
ScriptingCore::getInstance()->debugProcessInput(message);
|
2016-05-26 16:29:54 +08:00
|
|
|
|
|
|
|
if (messageCount == 0)
|
|
|
|
break;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
// std::this_thread::yield();
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(10));
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JSBDebug_enterNestedEventLoop(JSContext* cx, unsigned argc, jsval* vp)
|
|
|
|
{
|
|
|
|
enum {
|
|
|
|
NS_OK = 0,
|
|
|
|
NS_ERROR_UNEXPECTED
|
|
|
|
};
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
#define NS_SUCCEEDED(v) ((v) == NS_OK)
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
int rv = NS_OK;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
uint32_t nestLevel = ++s_nestedLoopLevel;
|
|
|
|
|
|
|
|
while (NS_SUCCEEDED(rv) && s_nestedLoopLevel >= nestLevel) {
|
|
|
|
if (!NS_ProcessNextEvent())
|
|
|
|
rv = NS_ERROR_UNEXPECTED;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
CCASSERT(s_nestedLoopLevel <= nestLevel,
|
|
|
|
"nested event didn't unwind properly");
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
args.rval().set(UINT_TO_JSVAL(s_nestedLoopLevel));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JSBDebug_exitNestedEventLoop(JSContext* cx, unsigned argc, jsval* vp)
|
|
|
|
{
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
if (s_nestedLoopLevel > 0) {
|
|
|
|
--s_nestedLoopLevel;
|
|
|
|
} else {
|
|
|
|
args.rval().set(UINT_TO_JSVAL(0));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
args.rval().setUndefined();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JSBDebug_getEventLoopNestLevel(JSContext* cx, unsigned argc, jsval* vp)
|
|
|
|
{
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
args.rval().set(UINT_TO_JSVAL(s_nestedLoopLevel));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//#pragma mark - Debugger
|
|
|
|
|
|
|
|
static void _clientSocketWriteAndClearString(std::string& s)
|
|
|
|
{
|
|
|
|
::send(clientSocket, s.c_str(), s.length(), 0);
|
|
|
|
s.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void processInput(const std::string& data) {
|
|
|
|
std::lock_guard<std::mutex> lk(g_qMutex);
|
|
|
|
g_queue.push_back(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void clearBuffers() {
|
|
|
|
std::lock_guard<std::mutex> lk(g_rwMutex);
|
|
|
|
// only process input if there's something and we're not locked
|
2016-08-19 16:28:47 +08:00
|
|
|
if (!inData.empty()) {
|
2015-05-05 10:50:19 +08:00
|
|
|
processInput(inData);
|
|
|
|
inData.clear();
|
|
|
|
}
|
2016-08-19 16:28:47 +08:00
|
|
|
if (!outData.empty()) {
|
2015-05-05 10:50:19 +08:00
|
|
|
_clientSocketWriteAndClearString(outData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void serverEntryPoint(unsigned int port)
|
|
|
|
{
|
|
|
|
// start a server, accept the connection and keep reading data from it
|
|
|
|
struct addrinfo hints, *result = nullptr, *rp = nullptr;
|
|
|
|
int s = 0;
|
|
|
|
memset(&hints, 0, sizeof(struct addrinfo));
|
2016-05-16 15:15:29 +08:00
|
|
|
hints.ai_family = AF_UNSPEC;
|
2015-05-05 10:50:19 +08:00
|
|
|
hints.ai_socktype = SOCK_STREAM; // TCP stream sockets
|
|
|
|
hints.ai_flags = AI_PASSIVE; // fill in my IP for me
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
std::stringstream portstr;
|
|
|
|
portstr << port;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
int err = 0;
|
2016-04-18 15:09:21 +08:00
|
|
|
|
metal support for cocos2d-x (#19305)
* remove deprecated files
* remove some deprecated codes
* remove more deprecated codes
* remove ui deprecated codes
* remove more deprecated codes
* remove deprecated codes in ccmenuitem
* remove more deprecated codes in ui
* remove more deprecated codes in ui
* remove more deprecated codes in ui
* remove more deprecated codes
* remove more deprecated codes
* remove more deprecated codes
* remove vr related codes and ignore some modules
* remove allocator
* remove some config
* 【Feature】add back-end project file
* [Feature] add back-end file
* add pipeline descriptor and shader cache
* [Feature] support sprite for backend
* [Feature] remove unneeded code
* [Feature] according to es2.0 spec, you must use clamp-to-edge as texture wrap mode, and no mipmapping for non-power-of-two texture
* [Feature] set texture wrap mode to clamp-to-edge, and no mipmapping for non-power-of-two texture
* [Feature] remove macro define to .cpp file
* [Feature] add log info
* [Feature] add PipelineDescriptor for TriangleCommand
* [Feature] add PipelineDescriptor object as member of TriangleCommand
* [Feature] add getPipelineDescriptor method
* add renderbackend
* complete pipeline descriptor
* [Feature] add viewport in RenderCommand
* set viewport when rendrering
* [Feature] occur error when using RendererBackend, to be fixed.
* a workaround to fix black screen on macOS 10.14 (#19090)
* add rendererbackend init function
* fix typo
* [Feature] modify testFile
* [BugFix] modify shader path
* [Feature] set default viewport
* fix projection
* [Feature] modify log info
* [BugFix] change viewport data type to int
* [BugFix] add BindGroup to PipelienDescriptor
* [BugFix] change a_position to vec3 in sprite.vert
* [BugFix] set vertexLayout according to V3F_C4B_T2F structure
* [Feature] revert a_position to vec4
* [Feature] renderer should not use gl codes directly
* [Feature] it's better not use default value parameter
* fix depth test setting
* rendererbackend -> renderer
* clear color and depth at begin
* add metal backend
* metal support normalized attribute
* simplify codes
* update external
* add render pass desctriptor in pipeline descriptor
* fix warnings
* fix crash and memeory leak
* refactor Texture2D
* put pipeline descriptor into render command
* simplify codes
* [Feature] update Sprite
* fix crash when closing app
* [Feature] update SpriteBatchNode and TextureAtlas
* support render texture(not finish)
* [Feature] remove unused code
* make tests work on mac
* fix download-deps path error
* make tests work on iOS
* [Feature] support ttf under normal label effect
* refactor triangle command processing
* let renderer handle more common commands
* refactor backend
* make render texture work
* [Feature] refactor backend for GL
* [Feature]Renaming to make it easy to understand
* [Feature] change warp mode to CLAMP_TO_EDGE
* fix ghost
* simplify visit render queue logic
* support progress timer without rial mode
* support partcile system
* Feature/update label (#149)
* [BugFix] fix compile error
* [Feature] support outline effect in ios
* [Feature] add shader file
* [BugFix] fix begin and end RenderPass
* [Feature] update CustomCommand
* [Feature] revert project.pbxproj
* [Feature] simplify codes
* [BugFix] pack AI88 to RGBA8888 only when outline enable
* [Feature] support shadow effect in Label
* [Feature] support BMFont
* [Feature] support glow effect
* [Feature] simplify shader files
* LabelAtlas work
* handle blend function correctly
* support tile map
* don't share buffer in metal
* alloc buffer size as needed
* support more tilemap
* Merge branch 'minggo/metal-support' into feature/updateLabel
* minggo/metal-support:
support tile map
handle blend function correctly
LabelAtlas work
Feature/update label (#149)
support partcile system
# Conflicts:
# cocos/2d/CCLabel.cpp
# cocos/2d/CCSprite.cpp
# cocos/2d/CCSpriteBatchNode.cpp
# cocos/renderer/CCQuadCommand.cpp
# cocos/renderer/CCQuadCommand.h
* render texture work without saving file
* use global viewport
* grid3d works
* remove grabber
* tiled3d works
* [BugFix] fix label bug
* [Feature] add updateSubData for buffer
* [Feature] remove setVertexCount
* support depth test
* add callback command
* [Feature] add UITest
* [Feature] update UITest
* [Feature] remove unneeded codes
* fix custom command issue
* fix layer color blend issue
* [BugFix] fix iOS compile error
* [Feature] remove unneeded codes
* [Feature] fix updateVertexBuffer
* layerradial works
* add draw test back
* fix batch issue
* fix compiling error
* [BugFix] support ETC1
* [BugFix] get the correct pipelineDescriptor
* [BugFix] skip draw when backendTexture nullptr
* clipping node support
* [Feature] add shader files
* fix stencil issue in metal
* [Feature] update UILayoutTest
* [BugFix] skip drawing when vertexCount is zero
* refactor renderer
* add set global z order for stencil manager commands
* fix warnings caused by type
* remove viewport in render command
* [Feature] fix warnings caused by type
* [BugFix] clear vertexCount and indexCount for CustomComand when needed
* [Feature] update clear for CustomCommand
* ios use metal
* fix viewport issue
* fix LayerColorGradient crash
* [cmake] transport to android and windows (#160)
* save point 1
* compile on windows
* run on android
* revert useless change
* android set CC_ENABLE_CACHE_TEXTURE_DATA to 1
* add initGlew
* fix android crash
* add TODO new-renderer
* review update
* revert onGLFWWindowPosCallback
* fix android compiling error
* Impl progress radial (#162)
* progresstimer add radial impl
* default drawType to element
* dec invoke times of createVertexBuffer (#163)
* support depth/stencil format for gl backend
* simplify progress timer codes
* support motionstreak, effect is wrong
* fix motionstreak issue
* [Feature] update Scissor Test (#161)
* [Feature] update Scissor Test
* [Feature] update ScissorTest
* [Feature] rename function
* [Feature] get constant reference if needed
* [Feature] show render status (#164)
* improve performance
* fix depth state
* fill error that triangle vertex/index number bigger than buffer
* fix compiline error in release mode
* fix buffer conflict between CPU and GPU on iOS/macOS
* Renderer refactor (#165)
* use one vertes/index buffer with opengl
* fix error on windows
* custom command support index format config
* CCLayer: compact vertex data structure
* update comment
* fix doc
* support fast tilemap
* pass index format instead
* fix some wrong effect
* fix render texture error
* fix texture per-element size
* fix texture format error
* BlendFunc type refactor, GLenum -> backend::BlendFactor (#167)
* BlendFunc use backend::BlendFactor as inner field
* update comments
* use int to replace GLenum
* update xcode project fiel
* rename to GLBlendConst
* add ccConstants.h
* update xcode project file
* update copyright
* remove primitive command
* remove CCPrimitive.cpp/.h
* remove deprecated files
* remove unneeded files
* remove multiple view support
* remove multiple view support
* remove the usage of frame buffer in camera
* director don't use frame buffer
* remove FrameBuffer
* remove BatchCommand
* add some api reference
* add physics2d back
* fix crash when close app on mac
* improve render texture
* fix rendertexture issue
* fix rendertexture issue
* simplify codes
* CMake support for mac & ios (#169)
* update cmake
* fix compile error
* update 3rd libs version
* remove CCThread.h/.cpp
* remove ccthread
* use audio engine to implement simple audio engine
* remove unneeded codes
* remove deprecated codes
* remove winrt macro
* remove CC_USE_WIC
* set partcile blend function in more elegant way
* remove unneeded codes
* remove unneeded codes
* cmake works on windows
* update project setting
* improve performance
* GLFloat -> float
* sync v3 cmake improvements into metal-support (#172)
* pick: modern cmake, compile definitions improvement (#19139)
* modern cmake, use target_compile_definitions partly
* simplify macro define, remove USE_*
* modern cmake, macro define
* add physics 2d macro define into ccConfig.h
* remove USE_CHIPMUNK macro in build.gradle
* remove CocosSelectModule.cmake
* shrink useless define
* simplify compile options config, re-add if necessary
* update external for tmp CI test
* un-quote target_compile_options value
* add "-g" parameter only when debug mode
* keep single build type when generator Xcode & VS projecy
* update external for tmp CI tes
* add static_cast<char>(-1), fix -Wc++11-narrowing
* simplify win32 compile define
* not modify code, only improve compile options
# Conflicts:
# .gitignore
# cmake/Modules/CocosConfigDepend.cmake
# cocos/CMakeLists.txt
# external/config.json
# tests/cpp-tests/CMakeLists.txt
* modern cmake, improve cmake_compiler_flags (#19145)
* cmake_compiler_flags
* Fix typo
* Fix typo2
* Remove chanages from Android.mk
* correct lua template cmake build (#19149)
* don't add -Wno-deprecated into jsb target
* correct lua template cmake build
* fix win32 lua template compile error
* prevent cmake in-source-build friendly (#19151)
* pick: Copy resources to "Resources/" on win32 like in linux configuration
* add "/Z7" for cpp-tests on windows
* [cmake] fix iOS xcode property setting failed (#19208)
* fix iOS xcode property setting failed
* use search_depend_libs_recursive at dlls collect
* fix typo
* [cmake] add find_host_library into iOS toolchain file (#19230)
* pick: [lua android] use luajit & template cmake update (#19239)
* increase cmake stability , remove tests/CMakeLists.txt (#19261)
* cmake win32 Precompiled header (#19273)
* Precompiled header
* Fix
* Precompiled header for cocos
* Precompiled header jscocos2d
* Fix for COCOS2D_DEBUG is always 1 on Android (#19291)
Related #19289
* little build fix, tests cpp-tests works on mac
* sync v3 build related codes into metal-support (#173)
* strict initialization for std::array
* remove proj.win32 project configs
* modern cmake, cmake_cleanup_remove_unused_variables (#19146)
* Switch travis CI to xenial (#19207)
* Switch travis CI to xenial
* Remove language: android
* Set language: cpp
* Fix java problem
* Update sdkmanager
* Fix sdkmanger
* next sdkmanager fix
* Remove xenial from android
* revert to sdk-tools-{system}-3859397
* Remove linux cmake install
* Update before-install.sh
* Update .travis.yml
* Simplify install-deps-linux.sh, tested on Ubuntu 16.04 (#19212)
* Simplify install-deps-linux.sh
* Cleanup
* pick: install ninja
* update cocos2d-console submodule
* for metal-support alpha release, we only test cpp
* add HelloCpp into project(Cocos2d-x) for tmp test
* update extenal metal-support-4
* update uniform setting
* [Feature] update BindGroup
* [Feature] empty-test
* [Feature] cpp-test
* [Feature] fix GL compiler error
* [Feature] fix GL crash
* [Feature] empty-test
* [Feature] cpp-tests
* [feature] improve frameRate
* [feature] fix opengl compile error
* [feature] fix opengl compile error
* [BugFix] fix compute maxLocation error
* [Feature] update setting unifrom
* [Feature] fix namespace
* [Feature] remove unneeded code
* [Bugfix] fix project file
* [Feature] update review
* [texture2d] impl texture format support (#175)
* texture update
* update
* update texture
* commit
* compile on windows
* ddd
* rename
* rename methods
* no crash
* save gl
* save
* save
* rename
* move out pixel format convert functions
* metal crash
* update
* update android
* support gles compressed texture format
* support more compress format
* add more conversion methods
* ss
* save
* update conversion methods
* add PVRTC format support
* reformat
* add marco linux
* fix GL marcro
* pvrtc supported only by ios 8.0+
* remove unused cmake
* revert change
* refactor Texture2D::initWithData
* fix conversion log
* refactor Texture2D::initWithData
* remove some OpenGL constants for PVRTC
* add todo
* fix typo
* AutoTest works on mac/iOS by disable part cases, sync v3 bug fix (#174)
* review cpp-tests, and fix part issues on start auto test
* sync png format fix: Node:Particle3D abnormal texture effects #19204
* fix cpp-tests SpritePolygon crash, wrong png format (#19170)
* fix wrong png convert format from sRGB to Gray
* erase plist index if all frames was erased
* test_A8.png have I8 format, fix it
* [CCSpriteCache] allow re-add plist & add testcase (#19175)
* allow re-add plist & add testcase
* remove comments/rename method/update testcase
* fix isSpriteFramesWithFileLoaded & add testcase
* remove used variable
* remove unused variable
* fix double free issues when js/lua-tests exit on iOS (#19236)
* disable part cases, AutoTest works without crash on mac
* update cocos2dx files json, to test cocos new next
* fix spritecache plist parsing issue (#19269)
* [linux] Fix FileUtils::getContents with folder (#19157)
* fix FileUtils::getContents on linux/mac
* use stat.st_mode
* simplify
* [CCFileUtils] win32 getFileSize (#19176)
* win32 getFileSize
* fix stat
* [cpp test-Android]20:FileUtils/2 change title (#19197)
* sync #19200
* sync #19231
* [android lua] improve performance of lua loader (#19234)
* [lua] improve performance of lua loader
* remove cache fix
* Revert "fix spritecache plist parsing issue (#19269)"
This reverts commit f3a85ece4307a7b90816c34489d1ed2c8fd11baf.
* remove win32 project files ref in template.json
* add metal framework lnk ref into cpp template
* test on iOS, and disable part cases
* alBufferData instead of alBufferDataStatic for small audio file on Apple (#19227)
* changes AudioCache to use alBufferData instead of alBufferDataStatic
(also makes test 19 faster to trigger openal bugs faster)
The original problem: CrashIfClientProvidedBogusAudioBufferList
https://github.com/cocos2d/cocos2d-x/issues/18948
is not happening anymore, but there's still a not very frequent issue
that makes OpenAL crash with a call stack like this.
AudioCache::readDataTask > alBufferData > CleanUpDeadBufferList
It happes more frequently when the device is "cold", which means after
half an hour of not using the device (locked).
I could not find the actual source code for iOS OpenAL, so I used the
macOS versions:
https://opensource.apple.com/source/OpenAL/OpenAL-48.7/Source/OpenAL/oalImp.cpp.auto.html
They seem to use CAGuard.h to make sure the dead buffer list
has no threading issues. I'm worried because the CAGuard code I found
has macos and win32 define but no iOS, so I'm not sure. I guess the
iOS version is different and has the guard.
I could not find a place in the code that's unprotected by the locks
except the InitializeBufferMap() which should not be called more than
once from cocos, and there's a workaround in AudioEngine-impl for it.
I reduced the occurence of the CleanUpDeadBufferList crash by moving
the guard in ~AudioCache to cover the alDeleteBuffers call.
* remove hack method "setTimeout" on audio
* AutoTest works on iOS
* support set ios deployment target for root project
* enable all texture2d cases, since Jiang have fixed
* add CCTextureUtils to xcode project file (#176)
* add leak cases for SpriteFrameCache (#177)
* re-add SpriteFrameCache cases
* update template file json
* Update SpriteFrameCacheTest.cpp
* fix compiling error
2019-01-18 15:08:25 +08:00
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32)
|
2015-05-05 10:50:19 +08:00
|
|
|
WSADATA wsaData;
|
|
|
|
err = WSAStartup(MAKEWORD(2, 2),&wsaData);
|
|
|
|
#endif
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if ((err = getaddrinfo(NULL, portstr.str().c_str(), &hints, &result)) != 0) {
|
|
|
|
LOGD("getaddrinfo error : %s\n", gai_strerror(err));
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
for (rp = result; rp != NULL; rp = rp->ai_next) {
|
|
|
|
if ((s = socket(rp->ai_family, rp->ai_socktype, 0)) < 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
int optval = 1;
|
|
|
|
if ((setsockopt(s, SOL_SOCKET, SO_REUSEADDR, (char*)&optval, sizeof(optval))) < 0) {
|
|
|
|
cc_closesocket(s);
|
|
|
|
TRACE_DEBUGGER_SERVER("debug server : error setting socket option SO_REUSEADDR");
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
|
|
|
|
if ((setsockopt(s, SOL_SOCKET, SO_NOSIGPIPE, &optval, sizeof(optval))) < 0) {
|
|
|
|
close(s);
|
|
|
|
TRACE_DEBUGGER_SERVER("debug server : error setting socket option SO_NOSIGPIPE");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif //(CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if ((::bind(s, rp->ai_addr, rp->ai_addrlen)) == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cc_closesocket(s);
|
|
|
|
s = -1;
|
|
|
|
}
|
|
|
|
if (s < 0 || rp == NULL) {
|
|
|
|
TRACE_DEBUGGER_SERVER("debug server : error creating/binding socket");
|
|
|
|
return;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
freeaddrinfo(result);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
listen(s, 1);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-06-03 11:39:01 +08:00
|
|
|
#define MAX_RECEIVED_SIZE 1024
|
|
|
|
#define BUF_SIZE MAX_RECEIVED_SIZE + 1
|
|
|
|
|
|
|
|
char buf[BUF_SIZE] = {0};
|
|
|
|
int readBytes = 0;
|
2015-05-05 10:50:19 +08:00
|
|
|
while (true) {
|
|
|
|
clientSocket = accept(s, NULL, NULL);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
if (clientSocket < 0)
|
|
|
|
{
|
|
|
|
TRACE_DEBUGGER_SERVER("debug server : error on accept");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// read/write data
|
|
|
|
TRACE_DEBUGGER_SERVER("debug server : client connected");
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
inData = "connected";
|
|
|
|
// process any input, send any output
|
|
|
|
clearBuffers();
|
2016-06-03 11:39:01 +08:00
|
|
|
|
|
|
|
while ((readBytes = (int)::recv(clientSocket, buf, MAX_RECEIVED_SIZE, 0)) > 0)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
buf[readBytes] = '\0';
|
|
|
|
// TRACE_DEBUGGER_SERVER("debug server : received command >%s", buf);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// no other thread is using this
|
|
|
|
inData.append(buf);
|
|
|
|
// process any input, send any output
|
|
|
|
clearBuffers();
|
|
|
|
} // while(read)
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
cc_closesocket(clientSocket);
|
|
|
|
}
|
|
|
|
} // while(true)
|
2016-06-03 11:39:01 +08:00
|
|
|
|
|
|
|
#undef BUF_SIZE
|
|
|
|
#undef MAX_RECEIVED_SIZE
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool JSBDebug_BufferWrite(JSContext* cx, unsigned argc, jsval* vp)
|
|
|
|
{
|
|
|
|
if (argc == 1) {
|
|
|
|
JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
|
|
|
|
JSStringWrapper strWrapper(args.get(0));
|
|
|
|
// this is safe because we're already inside a lock (from clearBuffers)
|
|
|
|
outData.append(strWrapper.get());
|
|
|
|
_clientSocketWriteAndClearString(outData);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptingCore::enableDebugger(unsigned int port)
|
|
|
|
{
|
2016-08-19 16:28:47 +08:00
|
|
|
if (!_debugGlobal)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
JS_SetDebugMode(_cx, true);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
_debugGlobal = new (std::nothrow) JS::PersistentRootedObject(_cx, newGlobalObject(_cx, true));
|
2015-05-05 10:50:19 +08:00
|
|
|
// Adds the debugger object to root, otherwise it may be collected by GC.
|
|
|
|
//AddObjectRoot(_cx, &_debugGlobal.ref()); no need, it's persistent rooted now
|
|
|
|
//JS_WrapObject(_cx, &_debugGlobal.ref()); Not really needed, JS_WrapObject makes a cross-compartment wrapper for the given JS object
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject rootedDebugObj(_cx, _debugGlobal->get());
|
2016-01-28 01:55:10 +08:00
|
|
|
JSAutoCompartment acDebug(_cx, rootedDebugObj);
|
2015-05-05 10:50:19 +08:00
|
|
|
// these are used in the debug program
|
|
|
|
JS_DefineFunction(_cx, rootedDebugObj, "log", ScriptingCore::log, 0, JSPROP_READONLY | JSPROP_ENUMERATE | JSPROP_PERMANENT);
|
2016-02-02 16:45:17 +08:00
|
|
|
JS_DefineFunction(_cx, rootedDebugObj, "require", ScriptingCore::executeScript, 2, JSPROP_ENUMERATE | JSPROP_PERMANENT);
|
2015-05-05 10:50:19 +08:00
|
|
|
JS_DefineFunction(_cx, rootedDebugObj, "_bufferWrite", JSBDebug_BufferWrite, 1, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(_cx, rootedDebugObj, "_enterNestedEventLoop", JSBDebug_enterNestedEventLoop, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(_cx, rootedDebugObj, "_exitNestedEventLoop", JSBDebug_exitNestedEventLoop, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
|
|
|
JS_DefineFunction(_cx, rootedDebugObj, "_getEventLoopNestLevel", JSBDebug_getEventLoopNestLevel, 0, JSPROP_READONLY | JSPROP_PERMANENT);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
JS::RootedObject globalObj(_cx, _global->get());
|
2015-05-05 10:50:19 +08:00
|
|
|
JS_WrapObject(_cx, &globalObj);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-02-02 16:45:17 +08:00
|
|
|
runScript("script/jsb_debugger.js", rootedDebugObj);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// prepare the debugger
|
|
|
|
jsval argv = OBJECT_TO_JSVAL(globalObj);
|
|
|
|
JS::RootedValue outval(_cx);
|
2016-01-28 01:55:10 +08:00
|
|
|
JS::HandleValueArray args = JS::HandleValueArray::fromMarkedLocation(1, &argv);
|
|
|
|
bool ok = JS_CallFunctionName(_cx, rootedDebugObj, "_prepareDebugger", args, &outval);
|
2015-05-05 10:50:19 +08:00
|
|
|
if (!ok) {
|
|
|
|
JS_ReportPendingException(_cx);
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
// start bg thread
|
|
|
|
auto t = std::thread(&serverEntryPoint,port);
|
|
|
|
t.detach();
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
Scheduler* scheduler = Director::getInstance()->getScheduler();
|
|
|
|
scheduler->scheduleUpdate(this->_runLoop, 0, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
JSObject* ScriptingCore::newGlobalObject(JSContext* cx, bool debug)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
|
|
|
JS::CompartmentOptions options;
|
|
|
|
options.setVersion(JSVERSION_LATEST);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
JS::RootedObject glob(cx, JS_NewGlobalObject(cx, &global_class, &shellTrustedPrincipals, JS::DontFireOnNewGlobalHook, options));
|
|
|
|
if (!glob) {
|
2015-12-14 10:27:11 +08:00
|
|
|
return nullptr;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
2018-01-10 14:39:45 +08:00
|
|
|
|
|
|
|
JSCompartment* oldDebugCompartment = nullptr;
|
|
|
|
|
|
|
|
if (!debug)
|
|
|
|
{
|
|
|
|
_oldCompartment = JS_EnterCompartment(cx, glob);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
oldDebugCompartment = JS_EnterCompartment(cx, glob);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
bool ok = true;
|
|
|
|
ok = JS_InitStandardClasses(cx, glob);
|
|
|
|
if (ok)
|
|
|
|
JS_InitReflect(cx, glob);
|
|
|
|
if (ok && debug)
|
|
|
|
ok = JS_DefineDebuggerObject(cx, glob);
|
|
|
|
if (!ok)
|
2015-12-14 10:27:11 +08:00
|
|
|
return nullptr;
|
2015-05-05 10:50:19 +08:00
|
|
|
|
|
|
|
JS_FireOnNewGlobalObject(cx, glob);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
if (debug)
|
|
|
|
{
|
|
|
|
JS_LeaveCompartment(cx, oldDebugCompartment);
|
|
|
|
}
|
|
|
|
|
2015-05-05 10:50:19 +08:00
|
|
|
return glob;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool jsb_set_reserved_slot(JSObject *obj, uint32_t idx, jsval value)
|
|
|
|
{
|
|
|
|
const JSClass *klass = JS_GetClass(obj);
|
|
|
|
unsigned int slots = JSCLASS_RESERVED_SLOTS(klass);
|
|
|
|
if ( idx >= slots )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
JS_SetReservedSlot(obj, idx, value);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool jsb_get_reserved_slot(JSObject *obj, uint32_t idx, jsval& ret)
|
|
|
|
{
|
|
|
|
const JSClass *klass = JS_GetClass(obj);
|
|
|
|
unsigned int slots = JSCLASS_RESERVED_SLOTS(klass);
|
|
|
|
if ( idx >= slots )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ret = JS_GetReservedSlot(obj, idx);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
js_proxy_t* jsb_new_proxy(void* nativeObj, JS::HandleObject jsHandle)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
js_proxy_t* proxy = nullptr;
|
|
|
|
JSObject* jsObj = jsHandle.get();
|
|
|
|
|
|
|
|
if (nativeObj && jsObj)
|
|
|
|
{
|
|
|
|
// native to JS index
|
2018-01-10 14:39:45 +08:00
|
|
|
proxy = new (std::nothrow) js_proxy_t();
|
2015-12-17 09:37:44 +08:00
|
|
|
CC_ASSERT(proxy && "not enough memory");
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
if (_js_native_global_map.find(jsObj) != _js_native_global_map.end())
|
|
|
|
{
|
|
|
|
CCLOG("BUG: old:%s new:%s", JS_GetClass(_js_native_global_map.at(jsObj)->_jsobj)->name, JS_GetClass(jsObj)->name);
|
|
|
|
}
|
|
|
|
#endif
|
2015-12-16 04:27:53 +08:00
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
CC_ASSERT(_native_js_global_map.find(nativeObj) == _native_js_global_map.end() && "Native Key should not be present");
|
2015-12-18 19:25:56 +08:00
|
|
|
// If native proxy doesn't exist, and js proxy exist, means previous js object in this location have already been released.
|
|
|
|
// In some circumstances, js object may be released without calling its finalizer, so the proxy haven't been removed.
|
|
|
|
// For ex: var seq = cc.sequence(moveBy, cc.callFunc(this.callback, this));
|
|
|
|
// In this code, cc.callFunc is created in parameter, and directly released after the native function call, the finalizer won't be triggered.
|
|
|
|
// The current solution keep the game running with a warning because it may cause memory leak as the native object may have been retained.
|
|
|
|
auto existJSProxy = _js_native_global_map.find(jsObj);
|
|
|
|
if (existJSProxy != _js_native_global_map.end()) {
|
2016-12-14 09:27:47 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
2015-12-18 19:25:56 +08:00
|
|
|
CCLOG("jsbindings: Failed to remove proxy for native object: %p, force removing it, but it may cause memory leak", existJSProxy->second->ptr);
|
|
|
|
#endif
|
|
|
|
jsb_remove_proxy(existJSProxy->second);
|
|
|
|
}
|
2015-12-16 04:27:53 +08:00
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
proxy->ptr = nativeObj;
|
|
|
|
proxy->obj = jsObj;
|
|
|
|
proxy->_jsobj = jsObj;
|
2015-12-16 04:27:53 +08:00
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
// One Proxy in two entries
|
|
|
|
_native_js_global_map[nativeObj] = proxy;
|
|
|
|
_js_native_global_map[jsObj] = proxy;
|
|
|
|
}
|
|
|
|
else CCLOG("jsb_new_proxy: Invalid keys");
|
|
|
|
|
|
|
|
return proxy;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
js_proxy_t* jsb_get_native_proxy(void* nativeObj)
|
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
auto search = _native_js_global_map.find(nativeObj);
|
|
|
|
if(search != _native_js_global_map.end())
|
|
|
|
return search->second;
|
|
|
|
return nullptr;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
2016-08-19 16:28:47 +08:00
|
|
|
js_proxy_t* jsb_get_js_proxy(JS::HandleObject jsObj)
|
2015-05-05 10:50:19 +08:00
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
auto search = _js_native_global_map.find(jsObj);
|
|
|
|
if(search != _js_native_global_map.end())
|
|
|
|
return search->second;
|
|
|
|
return nullptr;
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void jsb_remove_proxy(js_proxy_t* nativeProxy, js_proxy_t* jsProxy)
|
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
js_proxy_t* proxy = nativeProxy ? nativeProxy : jsProxy;
|
|
|
|
jsb_remove_proxy(proxy);
|
|
|
|
}
|
|
|
|
|
|
|
|
void jsb_remove_proxy(js_proxy_t* proxy)
|
|
|
|
{
|
|
|
|
void* nativeKey = proxy->ptr;
|
|
|
|
JSObject* jsKey = proxy->_jsobj;
|
|
|
|
|
|
|
|
CC_ASSERT(nativeKey && "Invalid nativeKey");
|
|
|
|
CC_ASSERT(jsKey && "Invalid JSKey");
|
|
|
|
|
|
|
|
auto it_nat = _native_js_global_map.find(nativeKey);
|
|
|
|
auto it_js = _js_native_global_map.find(jsKey);
|
|
|
|
|
|
|
|
if (it_nat != _native_js_global_map.end())
|
2015-12-16 04:27:53 +08:00
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
_native_js_global_map.erase(it_nat);
|
2015-12-16 04:27:53 +08:00
|
|
|
}
|
2015-12-17 09:37:44 +08:00
|
|
|
else CCLOG("jsb_remove_proxy: failed. Native key not found");
|
2015-12-16 04:27:53 +08:00
|
|
|
|
2015-12-17 09:37:44 +08:00
|
|
|
if (it_js != _js_native_global_map.end())
|
2015-12-16 04:27:53 +08:00
|
|
|
{
|
2015-12-17 09:37:44 +08:00
|
|
|
// Free it once, since we only have one proxy alloced entry
|
2018-01-10 14:39:45 +08:00
|
|
|
delete it_js->second;
|
2015-12-17 09:37:44 +08:00
|
|
|
_js_native_global_map.erase(it_js);
|
2015-12-16 04:27:53 +08:00
|
|
|
}
|
2015-12-17 09:37:44 +08:00
|
|
|
else CCLOG("jsb_remove_proxy: failed. JS key not found");
|
2015-05-05 10:50:19 +08:00
|
|
|
}
|
|
|
|
|
2015-12-03 06:45:13 +08:00
|
|
|
//
|
|
|
|
// Ref functions
|
|
|
|
//
|
2015-12-08 11:05:23 +08:00
|
|
|
|
|
|
|
// ref_create
|
2015-12-21 11:07:40 +08:00
|
|
|
JSObject* jsb_ref_create_jsobject(JSContext *cx, cocos2d::Ref *ref, js_type_class_t *typeClass, const char* debug)
|
2015-12-03 06:45:13 +08:00
|
|
|
{
|
|
|
|
JS::RootedObject proto(cx, typeClass->proto.ref());
|
|
|
|
JS::RootedObject parent(cx, typeClass->parentProto.ref());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsObj(cx, JS_NewObject(cx, typeClass->jsclass, proto, parent));
|
|
|
|
js_proxy_t* newproxy = jsb_new_proxy(ref, jsObj);
|
2015-12-03 06:45:13 +08:00
|
|
|
jsb_ref_init(cx, &newproxy->obj, ref, debug);
|
2015-12-17 21:53:40 +08:00
|
|
|
return jsObj;
|
2015-12-03 06:45:13 +08:00
|
|
|
}
|
|
|
|
|
2015-12-21 11:07:40 +08:00
|
|
|
JSObject* jsb_ref_autoreleased_create_jsobject(JSContext *cx, cocos2d::Ref *ref, js_type_class_t *typeClass, const char* debug)
|
2015-12-05 10:42:31 +08:00
|
|
|
{
|
|
|
|
JS::RootedObject proto(cx, typeClass->proto.ref());
|
|
|
|
JS::RootedObject parent(cx, typeClass->parentProto.ref());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsObj(cx, JS_NewObject(cx, typeClass->jsclass, proto, parent));
|
|
|
|
js_proxy_t* newproxy = jsb_new_proxy(ref, jsObj);
|
2015-12-05 10:42:31 +08:00
|
|
|
jsb_ref_autoreleased_init(cx, &newproxy->obj, ref, debug);
|
2015-12-17 21:53:40 +08:00
|
|
|
return jsObj;
|
|
|
|
}
|
|
|
|
|
2015-12-21 11:07:40 +08:00
|
|
|
JSObject* jsb_create_weak_jsobject(JSContext *cx, void *native, js_type_class_t *typeClass, const char* debug)
|
2015-12-17 21:53:40 +08:00
|
|
|
{
|
|
|
|
JS::RootedObject proto(cx, typeClass->proto.ref());
|
|
|
|
JS::RootedObject parent(cx, typeClass->parentProto.ref());
|
|
|
|
JS::RootedObject jsObj(cx, JS_NewObject(cx, typeClass->jsclass, proto, parent));
|
2015-12-17 21:57:19 +08:00
|
|
|
auto proxy = jsb_new_proxy(native, jsObj);
|
2016-12-21 13:39:34 +08:00
|
|
|
js_add_FinalizeHook(cx, jsObj, false);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-04-07 01:26:46 +08:00
|
|
|
#if ! CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::AddNamedObjectRoot(cx, &proxy->obj, debug);
|
|
|
|
#else
|
2017-03-16 09:42:16 +08:00
|
|
|
CC_UNUSED_PARAM(proxy);
|
2016-02-15 17:19:37 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
2016-12-21 13:39:34 +08:00
|
|
|
if (debug != nullptr)
|
|
|
|
{
|
|
|
|
CCLOG("++++++WEAK_REF++++++ Cpp(%s): %p - JS: %p", debug, native, jsObj.get());
|
|
|
|
}
|
2015-12-17 21:53:40 +08:00
|
|
|
#endif // COCOS2D_DEBUG
|
|
|
|
#endif // CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
return jsObj;
|
2015-12-05 10:42:31 +08:00
|
|
|
}
|
|
|
|
|
2015-12-08 11:05:23 +08:00
|
|
|
// get_or_create
|
2015-12-21 11:07:40 +08:00
|
|
|
JSObject* jsb_ref_get_or_create_jsobject(JSContext *cx, cocos2d::Ref *ref, js_type_class_t *typeClass, const char* debug)
|
2015-12-04 07:40:52 +08:00
|
|
|
{
|
|
|
|
auto proxy = jsb_get_native_proxy(ref);
|
|
|
|
if (proxy)
|
2015-12-17 21:53:40 +08:00
|
|
|
{
|
|
|
|
JS::RootedObject obj(cx, proxy->obj);
|
|
|
|
return obj;
|
|
|
|
}
|
2015-12-11 14:15:44 +08:00
|
|
|
|
|
|
|
JS::RootedObject proto(cx, typeClass->proto.ref());
|
|
|
|
JS::RootedObject parent(cx, typeClass->parentProto.ref());
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsObj(cx, JS_NewObject(cx, typeClass->jsclass, proto, parent));
|
|
|
|
js_proxy_t* newproxy = jsb_new_proxy(ref, jsObj);
|
2015-12-11 14:15:44 +08:00
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
2017-03-16 09:42:16 +08:00
|
|
|
CC_UNUSED_PARAM(newproxy);
|
2015-12-16 04:27:53 +08:00
|
|
|
ref->retain();
|
2016-12-21 13:39:34 +08:00
|
|
|
js_add_FinalizeHook(cx, jsObj, true);
|
2016-02-15 17:19:37 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
2016-02-15 16:32:00 +08:00
|
|
|
CCLOG("++++++RETAINED++++++ Cpp(%s): %p - JS: %p", debug, ref, jsObj.get());
|
2015-12-17 21:53:40 +08:00
|
|
|
#endif // COCOS2D_DEBUG
|
2015-12-11 14:15:44 +08:00
|
|
|
#else
|
2017-03-15 16:09:02 +08:00
|
|
|
// don't auto-release, don't retain.
|
2015-12-11 14:15:44 +08:00
|
|
|
JS::AddNamedObjectRoot(cx, &newproxy->obj, debug);
|
2015-12-17 21:53:40 +08:00
|
|
|
#endif // CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
2015-12-11 14:15:44 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
return jsObj;
|
2015-12-04 07:40:52 +08:00
|
|
|
}
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
// get_or_create: Ref is already autoreleased (or created)
|
2015-12-21 11:07:40 +08:00
|
|
|
JSObject* jsb_ref_autoreleased_get_or_create_jsobject(JSContext *cx, cocos2d::Ref *ref, js_type_class_t *typeClass, const char* debug)
|
2015-12-08 11:05:23 +08:00
|
|
|
{
|
|
|
|
auto proxy = jsb_get_native_proxy(ref);
|
|
|
|
if (proxy)
|
2015-12-17 21:53:40 +08:00
|
|
|
{
|
|
|
|
JS::RootedObject obj(cx, proxy->obj);
|
|
|
|
return obj;
|
|
|
|
}
|
2015-12-08 11:05:23 +08:00
|
|
|
// else
|
2015-12-11 10:02:55 +08:00
|
|
|
return jsb_ref_autoreleased_create_jsobject(cx, ref, typeClass, debug);
|
2015-12-08 11:05:23 +08:00
|
|
|
}
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
// get_or_create: when native object isn't a ref object or when the native object life cycle don't need to be managed by js object
|
2015-12-21 11:07:40 +08:00
|
|
|
JSObject* jsb_get_or_create_weak_jsobject(JSContext *cx, void *native, js_type_class_t *typeClass, const char* debug)
|
2015-12-17 21:53:40 +08:00
|
|
|
{
|
|
|
|
auto proxy = jsb_get_native_proxy(native);
|
|
|
|
if (proxy)
|
|
|
|
{
|
|
|
|
JS::RootedObject obj(cx, proxy->obj);
|
|
|
|
return obj;
|
|
|
|
}
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
// don't auto-release, don't retain.
|
|
|
|
JS::RootedObject proto(cx, typeClass->proto.ref());
|
|
|
|
JS::RootedObject parent(cx, typeClass->parentProto.ref());
|
|
|
|
JS::RootedObject jsObj(cx, JS_NewObject(cx, typeClass->jsclass, proto, parent));
|
|
|
|
proxy = jsb_new_proxy(native, jsObj);
|
2016-04-18 15:09:21 +08:00
|
|
|
|
2016-10-28 09:29:56 +08:00
|
|
|
JS::RootedObject flag(cx, JS_NewObject(cx, nullptr, JS::NullPtr(), JS::NullPtr()));
|
|
|
|
JS::RootedValue flagVal(cx, OBJECT_TO_JSVAL(flag));
|
|
|
|
JS_SetProperty(cx, jsObj, "__cppCreated", flagVal);
|
|
|
|
|
2018-01-10 14:39:45 +08:00
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
2017-03-16 09:42:16 +08:00
|
|
|
js_add_FinalizeHook(cx, jsObj, false);
|
2016-02-15 17:19:37 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
2017-03-16 09:42:16 +08:00
|
|
|
if (debug != nullptr)
|
|
|
|
{
|
|
|
|
CCLOG("++++++WEAK_REF++++++ Cpp(%s): %p - JS: %p", debug, native, jsObj.get());
|
|
|
|
}
|
2015-12-17 21:53:40 +08:00
|
|
|
#endif // COCOS2D_DEBUG
|
|
|
|
#endif // CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
return jsObj;
|
|
|
|
}
|
|
|
|
|
2015-12-08 11:05:23 +08:00
|
|
|
// ref_init
|
2015-12-03 06:45:13 +08:00
|
|
|
void jsb_ref_init(JSContext* cx, JS::Heap<JSObject*> *obj, Ref* ref, const char* debug)
|
|
|
|
{
|
|
|
|
// CCLOG("jsb_ref_init: JSObject address = %p. %s", obj->get(), debug);
|
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
2015-12-17 21:53:40 +08:00
|
|
|
(void)ref;
|
|
|
|
JS::RootedObject jsObj(cx, *obj);
|
2016-12-21 13:39:34 +08:00
|
|
|
js_add_FinalizeHook(cx, jsObj, true);
|
2015-12-16 04:27:53 +08:00
|
|
|
// don't retain it, already retained
|
2016-02-15 17:19:37 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
2016-02-15 16:32:00 +08:00
|
|
|
CCLOG("++++++RETAINED++++++ Cpp(%s): %p - JS: %p", debug, ref, jsObj.get());
|
2015-12-17 21:53:40 +08:00
|
|
|
#endif // COCOS2D_DEBUG
|
2015-12-03 06:45:13 +08:00
|
|
|
#else
|
2015-12-08 11:05:23 +08:00
|
|
|
// autorelease it
|
2015-12-03 06:45:13 +08:00
|
|
|
ref->autorelease();
|
|
|
|
JS::AddNamedObjectRoot(cx, obj, debug);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-05 10:42:31 +08:00
|
|
|
void jsb_ref_autoreleased_init(JSContext* cx, JS::Heap<JSObject*> *obj, Ref* ref, const char* debug)
|
|
|
|
{
|
|
|
|
// CCLOG("jsb_ref_init: JSObject address = %p. %s", obj->get(), debug);
|
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
(void)cx;
|
|
|
|
(void)obj;
|
2015-12-11 10:02:55 +08:00
|
|
|
ref->retain();
|
2015-12-17 21:53:40 +08:00
|
|
|
JS::RootedObject jsObj(cx, *obj);
|
2016-12-21 13:39:34 +08:00
|
|
|
js_add_FinalizeHook(cx, jsObj, true);
|
2017-03-16 09:42:16 +08:00
|
|
|
#if COCOS2D_DEBUG > 1
|
|
|
|
CCLOG("++++++RETAINED++++++ Cpp(%s): %p - JS: %p", debug, ref, jsObj.get());
|
|
|
|
#endif // COCOS2D_DEBUG
|
2015-12-05 10:42:31 +08:00
|
|
|
#else
|
|
|
|
// don't autorelease it, since it is already autoreleased
|
|
|
|
JS::AddNamedObjectRoot(cx, obj, debug);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-08 11:05:23 +08:00
|
|
|
// rebind
|
2015-12-17 09:37:44 +08:00
|
|
|
void jsb_ref_rebind(JSContext* cx, JS::HandleObject jsobj, js_proxy_t *proxy, cocos2d::Ref* oldRef, cocos2d::Ref* newRef, const char* debug)
|
2015-12-03 06:45:13 +08:00
|
|
|
{
|
2015-12-17 21:53:40 +08:00
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
// Release the old reference as it have been retained by jsobj previously,
|
|
|
|
// and the jsobj won't have any chance to release it in the future
|
|
|
|
oldRef->release();
|
|
|
|
#else
|
2015-12-17 09:37:44 +08:00
|
|
|
JS::RemoveObjectRoot(cx, &proxy->obj);
|
2015-12-03 06:45:13 +08:00
|
|
|
#endif
|
2015-12-17 09:37:44 +08:00
|
|
|
jsb_remove_proxy(proxy);
|
2015-12-03 06:45:13 +08:00
|
|
|
|
|
|
|
// Rebind js obj with new action
|
2016-06-24 14:38:59 +08:00
|
|
|
js_proxy_t* newProxy = jsb_new_proxy(newRef, jsobj);
|
|
|
|
|
2016-12-21 13:39:34 +08:00
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
CC_UNUSED_PARAM(newProxy);
|
|
|
|
#else
|
2016-06-24 14:38:59 +08:00
|
|
|
JS::AddNamedObjectRoot(cx, &newProxy->obj, debug);
|
|
|
|
#endif
|
2015-12-17 21:53:40 +08:00
|
|
|
}
|
|
|
|
|
2017-03-15 16:09:02 +08:00
|
|
|
void jsb_non_ref_init(JSContext* cx, JS::Heap<JSObject*> *obj, void* native, const char* debug)
|
|
|
|
{
|
|
|
|
// CCLOG("jsb_non_ref_init: JSObject address = %p. %s", obj->get(), debug);
|
|
|
|
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
|
|
|
|
JS::RootedObject jsObj(cx, *obj);
|
|
|
|
js_add_FinalizeHook(cx, jsObj, false);
|
|
|
|
// don't retain it, already retained
|
|
|
|
#if COCOS2D_DEBUG > 1
|
|
|
|
CCLOG("++++++RETAINED++++++ Cpp(%s): %p - JS: %p", debug, native, jsObj.get());
|
|
|
|
#endif // COCOS2D_DEBUG
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-17 21:53:40 +08:00
|
|
|
// Register finalize hook
|
|
|
|
void jsb_register_finalize_hook(JSObject *hook, JSObject *owner)
|
|
|
|
{
|
|
|
|
_js_hook_owner_map.insert(std::make_pair(hook, owner));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove hook owner entry in _js_hook_owner_map
|
|
|
|
JSObject *jsb_get_and_remove_hook_owner(JSObject *hook)
|
|
|
|
{
|
|
|
|
auto ownerIt = _js_hook_owner_map.find(hook);
|
|
|
|
// Found
|
|
|
|
if (ownerIt != _js_hook_owner_map.cend())
|
|
|
|
{
|
2016-02-02 23:38:37 +08:00
|
|
|
JSObject *result = ownerIt->second;
|
2015-12-17 21:53:40 +08:00
|
|
|
_js_hook_owner_map.erase(ownerIt);
|
2016-02-02 23:38:37 +08:00
|
|
|
return result;
|
2015-12-17 21:53:40 +08:00
|
|
|
}
|
|
|
|
return nullptr;
|
2015-12-03 06:45:13 +08:00
|
|
|
}
|