2014-04-30 08:50:12 +08:00
|
|
|
/**
|
|
|
|
Copyright 2013 BlackBerry Inc.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
|
|
|
|
Original file from GamePlay3D: http://gameplay3d.org
|
|
|
|
|
|
|
|
This file was modified to fit the cocos2d-x project
|
|
|
|
*/
|
|
|
|
|
2014-04-02 10:33:07 +08:00
|
|
|
#include "MathUtil.h"
|
2014-04-30 08:37:36 +08:00
|
|
|
#include "base/ccMacros.h"
|
2014-04-02 10:33:07 +08:00
|
|
|
|
2014-09-24 17:21:38 +08:00
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
|
|
|
|
#include <cpu-features.h>
|
|
|
|
#endif
|
|
|
|
|
2014-09-25 10:46:25 +08:00
|
|
|
//#define USE_NEON32 : neon 32 code will be used
|
|
|
|
//#define USE_NEON64 : neon 64 code will be used
|
|
|
|
//#define INCLUDE_NEON32 : neon 32 code included
|
|
|
|
//#define INCLUDE_NEON64 : neon 64 code included
|
|
|
|
//#define USE_SSE : SSE code used
|
|
|
|
//#define INCLUDE_SSE : SSE code included
|
|
|
|
|
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
|
|
|
|
#if defined (__arm64__)
|
|
|
|
#define USE_NEON64
|
|
|
|
#define INCLUDE_NEON64
|
|
|
|
#elif defined (__ARM_NEON__)
|
|
|
|
#define USE_NEON32
|
|
|
|
#define INCLUDE_NEON32
|
|
|
|
#else
|
|
|
|
#endif
|
|
|
|
#elif (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
|
2014-11-22 15:50:17 +08:00
|
|
|
#if defined (__arm64__) || defined (__aarch64__)
|
|
|
|
#define USE_NEON64
|
2014-09-25 16:09:28 +08:00
|
|
|
#define INCLUDE_NEON64
|
|
|
|
#elif defined (__ARM_NEON__)
|
|
|
|
#define INCLUDE_NEON32
|
|
|
|
#else
|
2014-09-25 10:46:25 +08:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined (__SSE__)
|
|
|
|
#define USE_SSE
|
|
|
|
#define INCLUDE_SSE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef INCLUDE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
#include "MathUtilNeon.inl"
|
|
|
|
#endif
|
|
|
|
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef INCLUDE_NEON64
|
2014-09-25 10:53:44 +08:00
|
|
|
#include "MathUtilNeon64.inl"
|
2014-09-25 10:46:25 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef INCLUDE_SSE
|
2014-09-24 16:12:47 +08:00
|
|
|
#include "MathUtilSSE.inl"
|
2014-09-25 10:46:25 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "MathUtil.inl"
|
2014-09-24 16:12:47 +08:00
|
|
|
|
2014-04-02 12:09:51 +08:00
|
|
|
NS_CC_MATH_BEGIN
|
2014-04-02 10:33:07 +08:00
|
|
|
|
|
|
|
void MathUtil::smooth(float* x, float target, float elapsedTime, float responseTime)
|
|
|
|
{
|
|
|
|
GP_ASSERT(x);
|
2014-09-24 16:12:47 +08:00
|
|
|
|
2014-04-02 10:33:07 +08:00
|
|
|
if (elapsedTime > 0)
|
|
|
|
{
|
|
|
|
*x += (target - *x) * elapsedTime / (elapsedTime + responseTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::smooth(float* x, float target, float elapsedTime, float riseTime, float fallTime)
|
|
|
|
{
|
|
|
|
GP_ASSERT(x);
|
|
|
|
|
|
|
|
if (elapsedTime > 0)
|
|
|
|
{
|
|
|
|
float delta = target - *x;
|
|
|
|
*x += delta * elapsedTime / (elapsedTime + (delta > 0 ? riseTime : fallTime));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-25 10:46:25 +08:00
|
|
|
bool MathUtil::isNeon32Enabled()
|
2014-09-24 17:21:38 +08:00
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 17:21:38 +08:00
|
|
|
return true;
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif (defined (INCLUDE_NEON32) && (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID) )
|
2014-09-25 16:09:28 +08:00
|
|
|
class AnrdoidNeonChecker
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
AnrdoidNeonChecker()
|
|
|
|
{
|
|
|
|
if (android_getCpuFamily() == ANDROID_CPU_FAMILY_ARM && (android_getCpuFeatures() & ANDROID_CPU_ARM_FEATURE_NEON) != 0)
|
|
|
|
_isNeonEnabled = true;
|
|
|
|
else
|
|
|
|
_isNeonEnabled = false;
|
|
|
|
}
|
|
|
|
bool isNeonEnabled() const { return _isNeonEnabled; }
|
|
|
|
private:
|
|
|
|
bool _isNeonEnabled;
|
|
|
|
};
|
|
|
|
static AnrdoidNeonChecker checker;
|
|
|
|
return checker.isNeonEnabled();
|
2014-09-24 17:21:38 +08:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-09-25 10:46:25 +08:00
|
|
|
bool MathUtil::isNeon64Enabled()
|
2014-09-24 17:21:38 +08:00
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON64
|
2014-09-24 17:21:38 +08:00
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-09-24 10:32:29 +08:00
|
|
|
void MathUtil::addMatrix(const float* m, float scalar, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::addMatrix(m, scalar, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::addMatrix(m, scalar, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::addMatrix(m, scalar, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::addMatrix(m, scalar, dst);
|
2014-09-24 10:32:29 +08:00
|
|
|
#else
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilC::addMatrix(m, scalar, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::addMatrix(const float* m1, const float* m2, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::addMatrix(m1, m2, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::addMatrix(m1, m2, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::addMatrix(m1, m2, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::addMatrix(m1, m2, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::addMatrix(m1, m2, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::subtractMatrix(const float* m1, const float* m2, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::subtractMatrix(m1, m2, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::subtractMatrix(m1, m2, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::subtractMatrix(m1, m2, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::subtractMatrix(m1, m2, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::subtractMatrix(m1, m2, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::multiplyMatrix(const float* m, float scalar, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::multiplyMatrix(m, scalar, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::multiplyMatrix(m, scalar, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::multiplyMatrix(m, scalar, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::multiplyMatrix(m, scalar, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::multiplyMatrix(m, scalar, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::multiplyMatrix(const float* m1, const float* m2, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::multiplyMatrix(m1, m2, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::multiplyMatrix(m1, m2, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::multiplyMatrix(m1, m2, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::multiplyMatrix(m1, m2, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::multiplyMatrix(m1, m2, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::negateMatrix(const float* m, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::negateMatrix(m, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::negateMatrix(m, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::negateMatrix(m, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::negateMatrix(m, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::negateMatrix(m, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::transposeMatrix(const float* m, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::transposeMatrix(m, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::transposeMatrix(m, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::transposeMatrix(m, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::transposeMatrix(m, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::transposeMatrix(m, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::transformVec4(const float* m, float x, float y, float z, float w, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::transformVec4(m, x, y, z, w, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::transformVec4(m, x, y, z, w, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::transformVec4(m, x, y, z, w, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::transformVec4(m, x, y, z, w, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::transformVec4(m, x, y, z, w, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::transformVec4(const float* m, const float* v, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::transformVec4(m, v, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::transformVec4(m, v, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::transformVec4(m, v, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::transformVec4(m, v, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::transformVec4(m, v, dst);
|
|
|
|
#endif
|
2014-09-24 10:32:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MathUtil::crossVec3(const float* v1, const float* v2, float* dst)
|
|
|
|
{
|
2014-09-25 10:46:25 +08:00
|
|
|
#ifdef USE_NEON32
|
2014-09-24 16:12:47 +08:00
|
|
|
MathUtilNeon::crossVec3(v1, v2, dst);
|
2014-09-25 10:53:44 +08:00
|
|
|
#elif defined (USE_NEON64)
|
|
|
|
MathUtilNeon64::crossVec3(v1, v2, dst);
|
2014-09-25 10:46:25 +08:00
|
|
|
#elif defined (INCLUDE_NEON32)
|
|
|
|
if(isNeon32Enabled()) MathUtilNeon::crossVec3(v1, v2, dst);
|
2014-09-24 17:21:38 +08:00
|
|
|
else MathUtilC::crossVec3(v1, v2, dst);
|
2014-09-24 16:12:47 +08:00
|
|
|
#else
|
|
|
|
MathUtilC::crossVec3(v1, v2, dst);
|
2014-09-24 10:32:29 +08:00
|
|
|
#endif
|
2014-09-24 11:06:07 +08:00
|
|
|
}
|
|
|
|
|
2014-04-02 12:09:51 +08:00
|
|
|
NS_CC_MATH_END
|