forked from organicmaps/organicmaps
[ANDROID] using NVidia application framework and libraries to support old devices and cumbersome opengl initialization/finalization issues.
This commit is contained in:
parent
ad54bdeecc
commit
a13ee120c1
24 changed files with 3823 additions and 278 deletions
|
@ -20,7 +20,12 @@ LOCAL_HEADER_FILES := \
|
|||
com/mapswithme/core/render_context.hpp \
|
||||
com/mapswithme/maps/Framework.hpp \
|
||||
com/mapswithme/platform/Platform.hpp \
|
||||
com/mapswithme/platform/http_thread_android.hpp \
|
||||
com/mapswithme/platform/http_thread_android.hpp \
|
||||
nv_thread/nv_thread.hpp \
|
||||
nv_event/nv_event_queue.hpp \
|
||||
nv_event/nv_event.hpp \
|
||||
nv_event/nv_keycode_mapping.hpp \
|
||||
nv_event/scoped_profiler.hpp
|
||||
|
||||
LOCAL_SRC_FILES := \
|
||||
com/mapswithme/core/concurrent_runner.cpp \
|
||||
|
@ -33,15 +38,19 @@ LOCAL_SRC_FILES := \
|
|||
com/mapswithme/maps/Framework.cpp \
|
||||
com/mapswithme/maps/VideoTimer.cpp \
|
||||
com/mapswithme/maps/GesturesProcessor.cpp \
|
||||
com/mapswithme/maps/MainGLView.cpp \
|
||||
com/mapswithme/maps/MWMActivity.cpp \
|
||||
com/mapswithme/maps/Lifecycle.cpp \
|
||||
com/mapswithme/platform/Platform.cpp \
|
||||
com/mapswithme/platform/http_thread_android.cpp \
|
||||
nv_thread/nv_thread.cpp \
|
||||
nv_event/nv_event_queue.cpp \
|
||||
nv_event/nv_event.cpp \
|
||||
nv_time/nv_time.cpp
|
||||
|
||||
LOCAL_LDLIBS := -llog -lGLESv1_CM \
|
||||
-lwords -lmap -lversion -lsearch -lstorage -lindexer -lyg -lplatform \
|
||||
-lgeometry -lcoding -lbase -lexpat -lfreetype -lfribidi -lzlib -lbzip2 \
|
||||
-ljansson -ltomcrypt -lprotobuf ./obj/local/armeabi/libstdc++.a
|
||||
-lwords -lmap -lversion -lsearch -lstorage -lindexer -lyg -lplatform \
|
||||
-lgeometry -lcoding -lbase -lexpat -lfreetype -lfribidi -lzlib -lbzip2 \
|
||||
-ljansson -ltomcrypt -lprotobuf ./obj/local/armeabi/libstdc++.a
|
||||
|
||||
LOCAL_LDLIBS += -L../../omim-android-debug/out/debug
|
||||
#LOCAL_LDLIBS += -L../../omim-android-release/out/release
|
||||
|
|
|
@ -33,15 +33,15 @@ namespace android
|
|||
{
|
||||
void Framework::CallRepaint()
|
||||
{
|
||||
// Always get current env pointer, it's different for each thread
|
||||
JNIEnv * env;
|
||||
m_jvm->AttachCurrentThread(&env, NULL);
|
||||
env->CallVoidMethod(m_mainGLView, jni::GetJavaMethodID(env, m_mainGLView, "requestRender", "()V"));
|
||||
//LOG(LINFO, ("Calling Repaint"));
|
||||
}
|
||||
|
||||
Framework::Framework(JavaVM * jvm)
|
||||
: m_jvm(jvm),
|
||||
m_work()
|
||||
: m_work(),
|
||||
m_eventType(NVMultiTouchEventType(0)),
|
||||
m_hasFirst(false),
|
||||
m_hasSecond(false),
|
||||
m_mask(0)
|
||||
{
|
||||
ASSERT(g_framework == 0, ());
|
||||
g_framework = this;
|
||||
|
@ -57,11 +57,6 @@ namespace android
|
|||
delete m_videoTimer;
|
||||
}
|
||||
|
||||
void Framework::SetParentView(jobject view)
|
||||
{
|
||||
m_mainGLView = view;
|
||||
}
|
||||
|
||||
namespace
|
||||
{
|
||||
struct make_all_invalid
|
||||
|
@ -83,12 +78,16 @@ namespace android
|
|||
};
|
||||
}
|
||||
|
||||
void Framework::DeleteRenderPolicy()
|
||||
{
|
||||
drule::rules().ForEachRule(make_all_invalid(GetPlatform().CpuCores() + 1));
|
||||
m_work.SetRenderPolicy(0);
|
||||
}
|
||||
|
||||
void Framework::InitRenderPolicy()
|
||||
{
|
||||
LOG(LDEBUG, ("AF::InitRenderer 1"));
|
||||
|
||||
drule::rules().ForEachRule(make_all_invalid(GetPlatform().CpuCores() + 1));
|
||||
|
||||
DrawerYG::Params params;
|
||||
params.m_frameBuffer = make_shared_ptr(new yg::gl::FrameBuffer(true));
|
||||
|
||||
|
@ -100,6 +99,8 @@ namespace android
|
|||
|
||||
m_work.SetUpdatesEnabled(true);
|
||||
|
||||
DrawFrame();
|
||||
|
||||
LOG(LDEBUG, ("AF::InitRenderer 2"));
|
||||
|
||||
m_work.ShowAll();
|
||||
|
@ -154,6 +155,107 @@ namespace android
|
|||
}
|
||||
}
|
||||
|
||||
void Framework::Touch(int action, int mask, double x1, double y1, double x2, double y2)
|
||||
{
|
||||
NVMultiTouchEventType eventType = (NVMultiTouchEventType)action;
|
||||
|
||||
if (m_mask != mask)
|
||||
{
|
||||
if (m_mask == 0x0)
|
||||
{
|
||||
if (mask == 0x1)
|
||||
m_work.StartDrag(DragEvent(x1, y1));
|
||||
|
||||
if (mask == 0x2)
|
||||
m_work.StartDrag(DragEvent(x2, y2));
|
||||
|
||||
if (mask == 0x3)
|
||||
m_work.StartScale(ScaleEvent(x1, y1, x2, y2));
|
||||
}
|
||||
|
||||
if (m_mask == 0x1)
|
||||
{
|
||||
m_work.StopDrag(DragEvent(x1, y1));
|
||||
|
||||
if (mask == 0x0)
|
||||
{
|
||||
if ((eventType != NV_MULTITOUCH_UP) && (eventType != NV_MULTITOUCH_CANCEL))
|
||||
LOG(LINFO, ("should be NV_MULTITOUCH_UP or NV_MULTITOUCH_CANCEL"));
|
||||
}
|
||||
|
||||
if (m_mask == 0x2)
|
||||
m_work.StartDrag(DragEvent(x2, y2));
|
||||
|
||||
if (mask == 0x3)
|
||||
m_work.StartScale(ScaleEvent(x1, y1, x2, y2));
|
||||
}
|
||||
|
||||
if (m_mask == 0x2)
|
||||
{
|
||||
m_work.StopDrag(DragEvent(x2, y2));
|
||||
|
||||
if (mask == 0x0)
|
||||
{
|
||||
if ((eventType != NV_MULTITOUCH_UP) && (eventType != NV_MULTITOUCH_CANCEL))
|
||||
LOG(LINFO, ("should be NV_MULTITOUCH_UP or NV_MULTITOUCH_CANCEL"));
|
||||
}
|
||||
|
||||
if (mask == 0x1)
|
||||
m_work.StartDrag(DragEvent(x1, y1));
|
||||
|
||||
if (mask == 0x3)
|
||||
m_work.StartScale(ScaleEvent(x1, y1, x2, y2));
|
||||
}
|
||||
|
||||
if (m_mask == 0x3)
|
||||
{
|
||||
m_work.StopScale(ScaleEvent(m_x1, m_y1, m_x2, m_y2));
|
||||
|
||||
if ((eventType == NV_MULTITOUCH_MOVE))
|
||||
{
|
||||
if (mask == 0x1)
|
||||
m_work.StartDrag(DragEvent(x1, y1));
|
||||
|
||||
if (mask == 0x2)
|
||||
m_work.StartDrag(DragEvent(x2, y2));
|
||||
}
|
||||
else
|
||||
mask = 0;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (eventType == NV_MULTITOUCH_MOVE)
|
||||
{
|
||||
if (m_mask == 0x1)
|
||||
m_work.DoDrag(DragEvent(x1, y1));
|
||||
if (m_mask == 0x2)
|
||||
m_work.DoDrag(DragEvent(x2, y2));
|
||||
if (m_mask == 0x3)
|
||||
m_work.DoScale(ScaleEvent(x1, y1, x2, y2));
|
||||
}
|
||||
|
||||
if ((eventType == NV_MULTITOUCH_CANCEL) || (eventType == NV_MULTITOUCH_UP))
|
||||
{
|
||||
if (m_mask == 0x1)
|
||||
m_work.StopDrag(DragEvent(x1, y1));
|
||||
if (m_mask == 0x2)
|
||||
m_work.StopDrag(DragEvent(x2, y2));
|
||||
if (m_mask == 0x3)
|
||||
m_work.StopScale(ScaleEvent(m_x1, m_y1, m_x2, m_y2));
|
||||
mask = 0;
|
||||
}
|
||||
}
|
||||
|
||||
m_x1 = x1;
|
||||
m_y1 = y1;
|
||||
m_x2 = x2;
|
||||
m_y2 = y2;
|
||||
m_mask = mask;
|
||||
m_eventType = eventType;
|
||||
|
||||
}
|
||||
|
||||
void f()
|
||||
{
|
||||
// empty location stub
|
||||
|
@ -190,9 +292,4 @@ namespace android
|
|||
info.m_accuracy = accuracy;
|
||||
m_work.OnCompassUpdate(info);
|
||||
}
|
||||
|
||||
JavaVM * Framework::javaVM() const
|
||||
{
|
||||
return m_jvm;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -12,16 +12,13 @@
|
|||
#include "../../../../../map/drawer_yg.hpp"
|
||||
#include "../../../../../map/window_handle.hpp"
|
||||
#include "../../../../../map/feature_vec_model.hpp"
|
||||
#include "../../../nv_event/nv_event.hpp"
|
||||
|
||||
namespace android
|
||||
{
|
||||
class Framework
|
||||
{
|
||||
private:
|
||||
|
||||
JavaVM * m_jvm;
|
||||
jobject m_mainGLView;
|
||||
|
||||
::Framework m_work;
|
||||
|
||||
VideoTimer * m_videoTimer;
|
||||
|
@ -32,6 +29,17 @@ namespace android
|
|||
|
||||
void CreateResourceManager();
|
||||
|
||||
NVMultiTouchEventType m_eventType; //< multitouch action
|
||||
|
||||
double m_x1;
|
||||
double m_y1;
|
||||
double m_x2;
|
||||
double m_y2;
|
||||
|
||||
bool m_hasFirst;
|
||||
bool m_hasSecond;
|
||||
int m_mask;
|
||||
|
||||
public:
|
||||
|
||||
Framework(JavaVM * jvm);
|
||||
|
@ -39,9 +47,8 @@ namespace android
|
|||
|
||||
storage::Storage & Storage();
|
||||
|
||||
void SetParentView(jobject view);
|
||||
|
||||
void InitRenderPolicy();
|
||||
void DeleteRenderPolicy();
|
||||
|
||||
void Resize(int w, int h);
|
||||
|
||||
|
@ -49,12 +56,11 @@ namespace android
|
|||
|
||||
void Move(int mode, double x, double y);
|
||||
void Zoom(int mode, double x1, double y1, double x2, double y2);
|
||||
void Touch(int action, int mask, double x1, double y1, double x2, double y2);
|
||||
|
||||
void EnableLocation(bool enable);
|
||||
void UpdateLocation(uint64_t timestamp, double lat, double lon, float accuracy);
|
||||
void UpdateCompass(uint64_t timestamp, double magneticNorth, double trueNorth, float accuracy);
|
||||
|
||||
JavaVM * javaVM() const;
|
||||
};
|
||||
}
|
||||
|
||||
|
|
321
android/jni/com/mapswithme/maps/Lifecycle.cpp
Normal file
321
android/jni/com/mapswithme/maps/Lifecycle.cpp
Normal file
|
@ -0,0 +1,321 @@
|
|||
/*
|
||||
* Lifecycle.cpp
|
||||
*
|
||||
* Created on: Nov 17, 2011
|
||||
* Author: siarheirachytski
|
||||
*/
|
||||
|
||||
#include <unistd.h>
|
||||
|
||||
#include <math.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <time.h>
|
||||
#include <jni.h>
|
||||
#include <android/log.h>
|
||||
#include <GLES/gl.h>
|
||||
#include "../../../nv_time/nv_time.hpp"
|
||||
#include "../../../nv_event/nv_event.hpp"
|
||||
#include "../../../nv_thread/nv_thread.hpp"
|
||||
#include "../../../../../base/logging.hpp"
|
||||
#include "Framework.hpp"
|
||||
#include "../platform/Platform.hpp"
|
||||
|
||||
#define MODULE "MapsWithMe"
|
||||
#define NVDEBUG(args...) __android_log_print(ANDROID_LOG_DEBUG, MODULE, ## args)
|
||||
|
||||
static float s_aspect = 1.0f;
|
||||
static int32_t s_winWidth = 1;
|
||||
static int32_t s_winHeight = 1;
|
||||
static unsigned int s_swapCount = 0;
|
||||
|
||||
static bool s_glesLoaded = false;
|
||||
static bool s_glesAutopaused = false;
|
||||
|
||||
static bool renderGameplay()
|
||||
{
|
||||
g_framework->DrawFrame();
|
||||
//glClearColor(0.0f, 1.0f, 1.0f, 1.0f);
|
||||
//glClear(GL_COLOR_BUFFER_BIT);
|
||||
|
||||
return true; //return true to update screen
|
||||
}
|
||||
|
||||
static bool renderPauseScreen()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
bool SetupGLESResources()
|
||||
{
|
||||
if (s_glesLoaded)
|
||||
return true;
|
||||
|
||||
NVDEBUG(reinterpret_cast<char const *>(glGetString(GL_EXTENSIONS)));
|
||||
NVDEBUG(reinterpret_cast<char const *>(glGetString(GL_VENDOR)));
|
||||
NVDEBUG(reinterpret_cast<char const *>(glGetString(GL_RENDERER)));
|
||||
NVDEBUG(reinterpret_cast<char const *>(glGetString(GL_VERSION)));
|
||||
// NVDEBUG(reinterpret_cast<char const *>(glGetString(GL_SHADING_LANGUAGE_VERSION)));
|
||||
|
||||
g_framework->DeleteRenderPolicy();
|
||||
g_framework->InitRenderPolicy();
|
||||
|
||||
s_glesLoaded = true;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool ShutdownGLESResources()
|
||||
{
|
||||
if (!s_glesLoaded)
|
||||
return true;
|
||||
|
||||
// We cannot use GLES calls to release the resources if the context is
|
||||
// not bound. In that case, we simply shut down EGL, which has code to
|
||||
// explicitly delete the context
|
||||
if (!NVEventStatusEGLIsBound())
|
||||
{
|
||||
NVDEBUG("ShutdownGLESResources: GLES not bound, shutting down EGL to release");
|
||||
if (NVEventCleanupEGL())
|
||||
{
|
||||
s_glesLoaded = false;
|
||||
return true;
|
||||
}
|
||||
else
|
||||
{
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
NVDEBUG("ShutdownGLESResources event: GLES bound, manually deleting GLES resources");
|
||||
|
||||
g_framework->DeleteRenderPolicy();/// delete RenderPolicy and ResourceManager here
|
||||
|
||||
s_glesLoaded = false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool renderFrame(bool allocateIfNeeded)
|
||||
{
|
||||
if (!NVEventReadyToRenderEGL(allocateIfNeeded))
|
||||
return false;
|
||||
|
||||
// We've gotten this far, so EGL is ready for us. Have we loaded our assets?
|
||||
// Note that we cannot use APP_STATUS_GLES_LOADED to imply that EGL is
|
||||
// ready to render. We can have a valid context with all GLES resources loaded
|
||||
// into it but no surface and thus the context not bound. These are semi-
|
||||
// independent states.
|
||||
if (!s_glesLoaded)
|
||||
{
|
||||
if (!allocateIfNeeded)
|
||||
return false;
|
||||
|
||||
if (!SetupGLESResources())
|
||||
return false;
|
||||
}
|
||||
|
||||
// If we're not paused, "animate" the scene
|
||||
if (!s_glesAutopaused)
|
||||
{
|
||||
// clock ticks, so animate something.
|
||||
}
|
||||
|
||||
// For this simple app, we render the gameplay every time
|
||||
// we render, even if it is paused. When we are paused, the
|
||||
/// gameplay is not animated and the pause "screen" is on top
|
||||
g_framework->DrawFrame();
|
||||
|
||||
// // If we're paused, draw the pause screen on top
|
||||
// if (s_glesAutopaused)
|
||||
// renderPauseScreen();
|
||||
|
||||
NVEventSwapBuffersEGL();
|
||||
|
||||
// A debug printout every 256 frames so we can see when we're
|
||||
// actively rendering and swapping
|
||||
if (!(s_swapCount++ & 0x00ff))
|
||||
{
|
||||
NVDEBUG("Swap count is %d", s_swapCount);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
// Add any initialization that requires the app Java classes
|
||||
// to be accessible (such as nv_shader_init, NvAPKInit, etc,
|
||||
// as listed in the docs)
|
||||
int32_t NVEventAppInit(int32_t argc, char** argv)
|
||||
{
|
||||
/*if (!g_framework)
|
||||
{
|
||||
android::Platform::Instance().Initialize(env, apkPath, storagePath);
|
||||
g_framework = new android::Framework(g_jvm);
|
||||
}*/
|
||||
|
||||
// nv_shader_init();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int32_t NVEventAppMain(int32_t argc, char** argv)
|
||||
{
|
||||
s_swapCount = 0;
|
||||
|
||||
s_glesLoaded = false;
|
||||
//s_glesAutopaused = false;
|
||||
|
||||
NVDEBUG( "App entering main loop");
|
||||
|
||||
while (NVEventStatusIsRunning())
|
||||
{
|
||||
const NVEvent* ev = NULL;
|
||||
while (NVEventStatusIsRunning() &&
|
||||
(ev = NVEventGetNextEvent(NVEventStatusIsFocused() ? 1 : 100)))
|
||||
{
|
||||
switch (ev->m_type)
|
||||
{
|
||||
case NV_EVENT_KEY:
|
||||
NVDEBUG( "Key event: 0x%02x %s",
|
||||
ev->m_data.m_key.m_code,
|
||||
(ev->m_data.m_key.m_action == NV_KEYACTION_DOWN) ? "down" : "up");
|
||||
|
||||
if ((ev->m_data.m_key.m_code == NV_KEYCODE_BACK)/* && !s_glesAutopaused*/)
|
||||
{
|
||||
// s_glesAutopaused = true;
|
||||
renderFrame(false);
|
||||
NVEventDoneWithEvent(true);
|
||||
}
|
||||
else
|
||||
{
|
||||
NVEventDoneWithEvent(false);
|
||||
}
|
||||
ev = NULL;
|
||||
break;
|
||||
|
||||
case NV_EVENT_CHAR:
|
||||
NVDEBUG("Char event: 0x%02x", ev->m_data.m_char.m_unichar);
|
||||
ev = NULL;
|
||||
NVEventDoneWithEvent(false);
|
||||
break;
|
||||
|
||||
case NV_EVENT_TOUCH:
|
||||
// if (!s_glesAutopaused)
|
||||
{
|
||||
switch (ev->m_data.m_touch.m_action)
|
||||
{
|
||||
case NV_TOUCHACTION_DOWN:
|
||||
g_framework->Move(0, ev->m_data.m_touch.m_x, ev->m_data.m_touch.m_y);
|
||||
break;
|
||||
case NV_TOUCHACTION_MOVE:
|
||||
g_framework->Move(1, ev->m_data.m_touch.m_x, ev->m_data.m_touch.m_y);
|
||||
break;
|
||||
case NV_TOUCHACTION_UP:
|
||||
g_framework->Move(2, ev->m_data.m_touch.m_x, ev->m_data.m_touch.m_y);
|
||||
break;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case NV_EVENT_MULTITOUCH:
|
||||
{
|
||||
int maskOnly = (ev->m_data.m_multi.m_action & NV_MULTITOUCH_POINTER_MASK) >> (NV_MULTITOUCH_POINTER_SHIFT);
|
||||
int action = ev->m_data.m_multi.m_action & NV_MULTITOUCH_ACTION_MASK;
|
||||
|
||||
/* ostringstream out;
|
||||
out << "mask: " << maskOnly
|
||||
<< ", action: " << action
|
||||
<< ", pts: " << ev->m_data.m_multi.m_x1 << " " << ev->m_data.m_multi.m_y1 << " "
|
||||
<< ev->m_data.m_multi.m_x2 << " " << ev->m_data.m_multi.m_y2;
|
||||
|
||||
NVDEBUG(out.str().c_str());*/
|
||||
|
||||
// if (!s_glesAutopaused)
|
||||
// {
|
||||
g_framework->Touch(action, maskOnly, ev->m_data.m_multi.m_x1, ev->m_data.m_multi.m_y1,
|
||||
ev->m_data.m_multi.m_x2, ev->m_data.m_multi.m_y2);
|
||||
// }
|
||||
}
|
||||
break;
|
||||
|
||||
case NV_EVENT_SURFACE_CREATED:
|
||||
case NV_EVENT_SURFACE_SIZE:
|
||||
s_winWidth = ev->m_data.m_size.m_w;
|
||||
s_winHeight = ev->m_data.m_size.m_h;
|
||||
|
||||
g_framework->Resize(s_winWidth, s_winHeight);
|
||||
|
||||
NVDEBUG( "Surface create/resize event: %d x %d", s_winWidth, s_winHeight);
|
||||
|
||||
if ((s_winWidth > 0) && (s_winHeight > 0))
|
||||
s_aspect = (float)s_winWidth / (float)s_winHeight;
|
||||
break;
|
||||
|
||||
case NV_EVENT_SURFACE_DESTROYED:
|
||||
NVDEBUG("Surface destroyed event");
|
||||
NVEventDestroySurfaceEGL();
|
||||
break;
|
||||
|
||||
case NV_EVENT_FOCUS_LOST:
|
||||
NVDEBUG("Focus lost event");
|
||||
// s_glesAutopaused = true;
|
||||
renderFrame(false);
|
||||
break;
|
||||
|
||||
case NV_EVENT_PAUSE:
|
||||
NVDEBUG("Pause event");
|
||||
//s_glesAutopaused = true;
|
||||
renderFrame(false);
|
||||
break;
|
||||
|
||||
case NV_EVENT_STOP:
|
||||
NVDEBUG("Stop event");
|
||||
// As per Google's recommendation, we release GLES resources here
|
||||
ShutdownGLESResources();
|
||||
break;
|
||||
case NV_EVENT_ACCEL:
|
||||
case NV_EVENT_START:
|
||||
case NV_EVENT_RESTART:
|
||||
case NV_EVENT_RESUME:
|
||||
case NV_EVENT_FOCUS_GAINED:
|
||||
case NV_EVENT_QUIT:
|
||||
NVDEBUG("%s event: no specific app action", NVEventGetEventStr(ev->m_type));
|
||||
break;
|
||||
|
||||
default:
|
||||
NVDEBUG("UNKNOWN event");
|
||||
break;
|
||||
};
|
||||
|
||||
// if we do not NULL out the event, then we return that
|
||||
// we handled it by default
|
||||
if (ev)
|
||||
NVEventDoneWithEvent(true);
|
||||
}
|
||||
|
||||
// Do not bother to initialize _any_ of EGL, much less go ahead
|
||||
// and render to the screen unless we have all we need to go
|
||||
// ahead and do our thing. In many cases,
|
||||
// devices will bring us part-way up and then take us down.
|
||||
// So, before we bother to init EGL (much less the rendering
|
||||
// surface, check that:
|
||||
// - we are focused
|
||||
// - we have a rendering surface
|
||||
// - the surface size is not 0x0
|
||||
// - we are resumed, not paused
|
||||
if (NVEventStatusIsInteractable())
|
||||
{
|
||||
// This will try to set up EGL if it isn't set up
|
||||
// When we first set up EGL completely, we also load our GLES resources
|
||||
// If these are already set up or we succeed at setting them all up now, then
|
||||
// we go ahead and render.
|
||||
renderFrame(true);
|
||||
}
|
||||
}
|
||||
|
||||
NVDEBUG("cleanup!!!");
|
||||
|
||||
NVEventCleanupEGL();
|
||||
|
||||
return 0;
|
||||
}
|
|
@ -15,6 +15,7 @@
|
|||
|
||||
#include "Framework.hpp"
|
||||
#include "../platform/Platform.hpp"
|
||||
#include "../../../nv_event/nv_event.hpp"
|
||||
|
||||
JavaVM * g_jvm;
|
||||
|
||||
|
@ -23,6 +24,7 @@ extern "C"
|
|||
JNIEXPORT jint JNICALL
|
||||
JNI_OnLoad(JavaVM * jvm, void * reserved)
|
||||
{
|
||||
InitNVEvent(jvm);
|
||||
g_jvm = jvm;
|
||||
jni::InitSystemLog();
|
||||
jni::InitAssertLog();
|
||||
|
|
|
@ -1,48 +0,0 @@
|
|||
/*
|
||||
* MainGLView.cpp
|
||||
*
|
||||
* Created on: Oct 13, 2011
|
||||
* Author: siarheirachytski
|
||||
*/
|
||||
|
||||
#include <jni.h>
|
||||
#include "Framework.hpp"
|
||||
|
||||
extern "C"
|
||||
{
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// MainGLView
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
JNIEXPORT void JNICALL
|
||||
Java_com_mapswithme_maps_MainGLView_nativeInit(JNIEnv * env, jobject thiz)
|
||||
{
|
||||
ASSERT ( g_framework, () );
|
||||
g_framework->SetParentView(thiz);
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// MainRenderer
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
JNIEXPORT void JNICALL
|
||||
Java_com_mapswithme_maps_MainRenderer_nativeInit(JNIEnv * env, jobject thiz)
|
||||
{
|
||||
ASSERT ( g_framework, () );
|
||||
g_framework->InitRenderPolicy();
|
||||
}
|
||||
|
||||
JNIEXPORT void JNICALL
|
||||
Java_com_mapswithme_maps_MainRenderer_nativeResize(JNIEnv * env, jobject thiz, jint w, jint h)
|
||||
{
|
||||
ASSERT ( g_framework, () );
|
||||
g_framework->Resize(w, h);
|
||||
}
|
||||
|
||||
JNIEXPORT void JNICALL
|
||||
Java_com_mapswithme_maps_MainRenderer_nativeDraw(JNIEnv * env, jobject thiz)
|
||||
{
|
||||
ASSERT ( g_framework, () );
|
||||
g_framework->DrawFrame();
|
||||
}
|
||||
}
|
|
@ -33,39 +33,39 @@ namespace android
|
|||
|
||||
void VideoTimer::start()
|
||||
{
|
||||
JNIEnv * env;
|
||||
/* JNIEnv * env;
|
||||
m_javaVM->AttachCurrentThread(&env, NULL);
|
||||
env->CallVoidMethod(m_videoTimer, jni::GetJavaMethodID(env, m_videoTimer, "start", "()V"));
|
||||
m_state = ERunning;
|
||||
m_state = ERunning;*/
|
||||
}
|
||||
|
||||
void VideoTimer::resume()
|
||||
{
|
||||
JNIEnv * env;
|
||||
/* JNIEnv * env;
|
||||
m_javaVM->AttachCurrentThread(&env, NULL);
|
||||
env->CallVoidMethod(m_videoTimer, jni::GetJavaMethodID(env, m_videoTimer, "resume", "()V"));
|
||||
m_state = ERunning;
|
||||
m_state = ERunning;*/
|
||||
}
|
||||
|
||||
void VideoTimer::pause()
|
||||
{
|
||||
JNIEnv * env;
|
||||
/* JNIEnv * env;
|
||||
m_javaVM->AttachCurrentThread(&env, NULL);
|
||||
env->CallVoidMethod(m_videoTimer, jni::GetJavaMethodID(env, m_videoTimer, "pause", "()V"));
|
||||
m_state = EPaused;
|
||||
m_state = EPaused;*/
|
||||
}
|
||||
|
||||
void VideoTimer::stop()
|
||||
{
|
||||
JNIEnv * env;
|
||||
/* JNIEnv * env;
|
||||
m_javaVM->AttachCurrentThread(&env, NULL);
|
||||
env->CallVoidMethod(m_videoTimer, jni::GetJavaMethodID(env, m_videoTimer, "stop", "()V"));
|
||||
m_state = EStopped;
|
||||
m_state = EStopped;*/
|
||||
}
|
||||
|
||||
void VideoTimer::perform()
|
||||
{
|
||||
m_frameFn();
|
||||
//m_frameFn();
|
||||
}
|
||||
}
|
||||
|
||||
|
|
77
android/jni/nv_debug/nv_debug.hpp
Normal file
77
android/jni/nv_debug/nv_debug.hpp
Normal file
|
@ -0,0 +1,77 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_debug.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
#ifndef __INCLUDED_NV_DEBUG_H
|
||||
#define __INCLUDED_NV_DEBUG_H
|
||||
|
||||
#define CT_ASSERT(tag,cond) \
|
||||
enum { COMPILE_TIME_ASSERT__ ## tag = 1/(cond) }
|
||||
|
||||
#define dimof( x ) ( sizeof(x) / sizeof(x[0]) )
|
||||
#include <android/log.h>
|
||||
|
||||
#define DBG_DETAILED 0
|
||||
|
||||
#if 0
|
||||
|
||||
// the detailed prefix can be customised by setting DBG_DETAILED_PREFIX. See
|
||||
// below as a reference.
|
||||
// NOTE: fmt is the desired format string and must be in the prefix.
|
||||
//#ifndef DBG_DETAILED_PREFIX
|
||||
// #define DBG_DETAILED_PREFIX "%s, %s, line %d: " fmt, __FILE__, __FUNCTION__, __LINE__,
|
||||
//#endif
|
||||
//#define DEBUG_D_(fmt, args...)
|
||||
//#define DEBUG_D(fmt, args...) __android_log_print(ANDROID_LOG_DEBUG, MODULE, (DBG_DETAILED_PREFIX) ## args)
|
||||
|
||||
#else
|
||||
|
||||
#ifdef STRINGIFY
|
||||
#pragma push_macro("STRINGIFY")
|
||||
#undef STRINGIFY
|
||||
#define STRINGIFYPUSHED_____
|
||||
#endif
|
||||
#define STRINGIFY(x) #x
|
||||
|
||||
// debug macro, includes file name function name and line number
|
||||
#define TO(x) typeof(x)
|
||||
#define DEBUG_D_(file, line, fmt, args...) __android_log_print(ANDROID_LOG_DEBUG, MODULE, file ", %s, line(" STRINGIFY(line) "): " fmt, __FUNCTION__, ## args)
|
||||
#define DEBUG_D(fmt, args...) DEBUG_D_( __FILE__ , __LINE__ , fmt, ## args)
|
||||
|
||||
#ifdef STRINGIFYPUSHED_____
|
||||
#undef STRINGIFYPUSHED_____
|
||||
#pragma pop_macro("STRINGIFY")
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
||||
// basic debug macro
|
||||
#define NVDEBUG_(fmt, args...) (__android_log_print(ANDROID_LOG_DEBUG, MODULE, fmt, ## args))
|
||||
|
||||
// Debug macro that can be switched to spew a file name,
|
||||
// function and line number using DEBUG_DETAILED
|
||||
#if DBG_DETAILED == 1
|
||||
#define NVDEBUG(fmt, args...) NVDEBUG_D(fmt, ## args)
|
||||
#else
|
||||
#define NVDEBUG(fmt, args...) __android_log_print(ANDROID_LOG_DEBUG, MODULE, fmt, ## args)
|
||||
#endif
|
||||
|
||||
|
||||
#endif
|
907
android/jni/nv_event/nv_event.cpp
Normal file
907
android/jni/nv_event/nv_event.cpp
Normal file
|
@ -0,0 +1,907 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_event\nv_event.cpp
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#define MODULE "NVEvent"
|
||||
#define DBG_DETAILED 0
|
||||
|
||||
#include "nv_event.hpp"
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <jni.h>
|
||||
#include <pthread.h>
|
||||
#include <android/log.h>
|
||||
#include <GLES2/gl2.h>
|
||||
//#include <EGL/egl.h>
|
||||
|
||||
#include "../nv_time/nv_time.hpp"
|
||||
#include "../nv_thread/nv_thread.hpp"
|
||||
#include "../nv_debug/nv_debug.hpp"
|
||||
#include "scoped_profiler.hpp"
|
||||
#include "nv_keycode_mapping.hpp"
|
||||
#include "nv_event_queue.hpp"
|
||||
|
||||
// TODO TBD - this should be done in NVTimeInit(), but we use a different
|
||||
// class than most apps. Need to clean this up, as it is fragile w.r.t.
|
||||
// changes in nv_time
|
||||
extern void nvAcquireTimeExtensionJNI(JNIEnv*, jobject);
|
||||
extern jlong nvGetSystemTimeJNI(JNIEnv*, jobject);
|
||||
|
||||
#define CT_ASSERT(tag,cond) \
|
||||
enum { COMPILE_TIME_ASSERT__ ## tag = 1/(cond) }
|
||||
|
||||
enum
|
||||
{
|
||||
// Android lifecycle status flags. Not app-specific
|
||||
// Set between onCreate and onDestroy
|
||||
NVEVENT_STATUS_RUNNING = 0x00000001,
|
||||
// Set between onResume and onPause
|
||||
NVEVENT_STATUS_ACTIVE = 0x00000002,
|
||||
// Set between onWindowFocusChanged(true) and (false)
|
||||
NVEVENT_STATUS_FOCUSED = 0x00000004,
|
||||
// Set when the app's SurfaceHolder points to a
|
||||
// valid, nonzero-sized surface
|
||||
NVEVENT_STATUS_HAS_REAL_SURFACE = 0x00000008,
|
||||
|
||||
// Mask of all app lifecycle status flags, useful for checking when is it
|
||||
// a reasonable time to be setting up EGL and rendering
|
||||
NVEVENT_STATUS_INTERACTABLE = 0x0000000f,
|
||||
|
||||
// NvEvent EGL status flags. Not app-specific
|
||||
// Set between calls to NVEventInitEGL and NVEventCleanupEGL
|
||||
NVEVENT_STATUS_EGL_INITIALIZED = 0x00000010,
|
||||
// Set when the EGL surface is allocated
|
||||
NVEVENT_STATUS_EGL_HAS_SURFACE = 0x00000020,
|
||||
// Set when a surface and context are available and bound
|
||||
NVEVENT_STATUS_EGL_BOUND = 0x00000040,
|
||||
};
|
||||
|
||||
static unsigned int s_appStatus = 0;
|
||||
|
||||
static void ZeroAppFlags()
|
||||
{
|
||||
s_appStatus = 0;
|
||||
}
|
||||
|
||||
static void SetAppFlag(unsigned int status)
|
||||
{
|
||||
s_appStatus |= status;
|
||||
}
|
||||
|
||||
static void ClearAppFlag(unsigned int status)
|
||||
{
|
||||
s_appStatus &= ~status;
|
||||
}
|
||||
|
||||
static bool QueryAppFlag(unsigned int status)
|
||||
{
|
||||
return (s_appStatus & status) ? true : false;
|
||||
}
|
||||
|
||||
static bool QueryAppFlagsEqualMasked(unsigned int status, unsigned int mask)
|
||||
{
|
||||
return ((s_appStatus & mask) == status) ? true : false;
|
||||
}
|
||||
|
||||
static NVKeyCodeMapping s_keyMapping;
|
||||
static NVEventQueue s_eventQueue;
|
||||
static jobject s_globalThiz;
|
||||
static jfieldID s_lengthId;
|
||||
static jfieldID s_dataId;
|
||||
static jfieldID s_widthId;
|
||||
static jfieldID s_heightId;
|
||||
static jfieldID s_texDataId;
|
||||
static pthread_t s_mainThread;
|
||||
static bool s_appThreadExited = false;
|
||||
static bool s_javaPostedQuit = false;
|
||||
|
||||
static int NVEVENT_ACTION_DOWN = 0;
|
||||
static int NVEVENT_ACTION_UP = 0;
|
||||
static int NVEVENT_ACTION_CANCEL = 0;
|
||||
static int NVEVENT_ACTION_POINTER_INDEX_MASK = 0;
|
||||
static int NVEVENT_ACTION_POINTER_INDEX_SHIFT = 0;
|
||||
static int NVEVENT_ACTION_KEY_UP = 0;
|
||||
|
||||
class MethodRef
|
||||
{
|
||||
public:
|
||||
MethodRef(const char* name,
|
||||
const char* signature) :
|
||||
m_name(name),
|
||||
m_signature(signature),
|
||||
m_index(NULL)
|
||||
{}
|
||||
|
||||
bool QueryID(JNIEnv *env, jclass k)
|
||||
{
|
||||
m_index = env->GetMethodID(k, m_name, m_signature);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool CallBoolean()
|
||||
{
|
||||
JNIEnv* jniEnv = NVThreadGetCurrentJNIEnv();
|
||||
|
||||
if (!jniEnv || !s_globalThiz)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error: No valid JNI env in %s", m_name);
|
||||
return false;
|
||||
}
|
||||
if (!m_index)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error: No valid function pointer in %s", m_name);
|
||||
return false;
|
||||
}
|
||||
// __android_log_print(ANDROID_LOG_DEBUG, MODULE, "Calling JNI up to %s", m_name);
|
||||
return jniEnv->CallBooleanMethod(s_globalThiz, m_index);
|
||||
}
|
||||
|
||||
bool CallInt()
|
||||
{
|
||||
JNIEnv* jniEnv = NVThreadGetCurrentJNIEnv();
|
||||
|
||||
if (!jniEnv || !s_globalThiz)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error: No valid JNI env in %s", m_name);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (!m_index)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error: No valid function pointer in %s", m_name);
|
||||
return false;
|
||||
}
|
||||
|
||||
return (int)jniEnv->CallIntMethod(s_globalThiz, m_index);
|
||||
}
|
||||
|
||||
const char* m_name;
|
||||
const char* m_signature;
|
||||
jmethodID m_index;
|
||||
|
||||
};
|
||||
|
||||
static MethodRef s_InitEGL("InitEGL", "()Z");
|
||||
static MethodRef s_CleanupEGL("CleanupEGL", "()Z");
|
||||
static MethodRef s_CreateSurfaceEGL("CreateSurfaceEGL", "()Z");
|
||||
static MethodRef s_DestroySurfaceEGL("DestroySurfaceEGL", "()Z");
|
||||
static MethodRef s_SwapBuffersEGL("SwapBuffersEGL", "()Z");
|
||||
static MethodRef s_BindSurfaceAndContextEGL("BindSurfaceAndContextEGL", "()Z");
|
||||
static MethodRef s_UnbindSurfaceAndContextEGL("UnbindSurfaceAndContextEGL", "()Z");
|
||||
static MethodRef s_GetErrorEGL("GetErrorEGL", "()I");
|
||||
static MethodRef s_finish("finish", "()V");
|
||||
|
||||
// True between onCreate and onDestroy
|
||||
bool NVEventStatusIsRunning()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_RUNNING);
|
||||
}
|
||||
|
||||
// True between onResume and onPause
|
||||
bool NVEventStatusIsActive()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_ACTIVE);
|
||||
}
|
||||
|
||||
// True between onWindowFocusChanged(true) and (false)
|
||||
bool NVEventStatusIsFocused()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_FOCUSED);
|
||||
}
|
||||
|
||||
// True when the app's SurfaceHolder points to a
|
||||
// valid, nonzero-sized window
|
||||
bool NVEventStatusHasRealSurface()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_HAS_REAL_SURFACE);
|
||||
}
|
||||
|
||||
// True when all of IsRunning, IsActive, IsFocused, HasRealSurface are true
|
||||
// useful for checking when is it a reasonable time to be setting up EGL and rendering
|
||||
bool NVEventStatusIsInteractable()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlagsEqualMasked(NVEVENT_STATUS_INTERACTABLE, NVEVENT_STATUS_INTERACTABLE);
|
||||
}
|
||||
|
||||
// True between calls to NVEventInitEGL and NVEventCleanupEGL
|
||||
bool NVEventStatusEGLInitialized()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_EGL_INITIALIZED);
|
||||
}
|
||||
|
||||
// True when the EGL surface is allocated
|
||||
bool NVEventStatusEGLHasSurface()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_EGL_HAS_SURFACE);
|
||||
}
|
||||
|
||||
// True when a surface and context are available and bound
|
||||
bool NVEventStatusEGLIsBound()
|
||||
{
|
||||
// TBD - need to lock a mutex?
|
||||
return QueryAppFlag(NVEVENT_STATUS_EGL_BOUND);
|
||||
}
|
||||
|
||||
static void NVEventInitInputFields(JNIEnv *env)
|
||||
{
|
||||
jclass Motion_class = env->FindClass("android/view/MotionEvent");
|
||||
jfieldID ACTION_DOWN_id = env->GetStaticFieldID(Motion_class, "ACTION_DOWN", "I");
|
||||
jfieldID ACTION_UP_id = env->GetStaticFieldID(Motion_class, "ACTION_UP", "I");
|
||||
jfieldID ACTION_CANCEL_id = env->GetStaticFieldID(Motion_class, "ACTION_CANCEL", "I");
|
||||
jfieldID ACTION_POINTER_INDEX_SHIFT_id = env->GetStaticFieldID(Motion_class, "ACTION_POINTER_ID_SHIFT", "I");
|
||||
jfieldID ACTION_POINTER_INDEX_MASK_id = env->GetStaticFieldID(Motion_class, "ACTION_POINTER_ID_MASK", "I");
|
||||
NVEVENT_ACTION_DOWN = env->GetStaticIntField(Motion_class, ACTION_DOWN_id);
|
||||
NVEVENT_ACTION_UP = env->GetStaticIntField(Motion_class, ACTION_UP_id);
|
||||
NVEVENT_ACTION_CANCEL = env->GetStaticIntField(Motion_class, ACTION_CANCEL_id);
|
||||
NVEVENT_ACTION_POINTER_INDEX_MASK = env->GetStaticIntField(Motion_class, ACTION_POINTER_INDEX_MASK_id);
|
||||
NVEVENT_ACTION_POINTER_INDEX_SHIFT = env->GetStaticIntField(Motion_class, ACTION_POINTER_INDEX_SHIFT_id);
|
||||
|
||||
jclass KeyCode_class = env->FindClass("android/view/KeyEvent");
|
||||
jfieldID ACTION_KEY_UP_id = env->GetStaticFieldID(KeyCode_class, "ACTION_UP", "I");
|
||||
NVEVENT_ACTION_KEY_UP = env->GetStaticIntField(KeyCode_class, ACTION_KEY_UP_id);
|
||||
}
|
||||
|
||||
static void* NVEventMainLoopThreadFunc(void*)
|
||||
{
|
||||
NVEventAppMain(0, NULL);
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "NvEvent native app Main returned");
|
||||
|
||||
// signal the condition variable to unblock
|
||||
// java from waiting on pause or quit
|
||||
s_eventQueue.UnblockProducer();
|
||||
|
||||
s_appThreadExited = true;
|
||||
|
||||
// IF that app main returned because we posted a QUIT, then Java knows what to
|
||||
// do regarding lifecycle. But, if the app returned from main of its own accord,
|
||||
// we need to call finish.
|
||||
if (!s_javaPostedQuit)
|
||||
{
|
||||
JNIEnv* env = NVThreadGetCurrentJNIEnv();
|
||||
env->CallVoidMethod(s_globalThiz, s_finish.m_index);
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
NVEventPlatformAppHandle NVEventGetPlatformAppHandle()
|
||||
{
|
||||
return s_globalThiz;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// Native event-handling functions
|
||||
|
||||
const char* NVEventGetEventStr(NVEventType eventType)
|
||||
{
|
||||
switch(eventType)
|
||||
{
|
||||
case NV_EVENT_KEY: return "NV_EVENT_KEY";
|
||||
case NV_EVENT_CHAR: return "NV_EVENT_CHAR";
|
||||
case NV_EVENT_TOUCH: return "NV_EVENT_TOUCH";
|
||||
case NV_EVENT_MULTITOUCH: return "NV_EVENT_MULTITOUCH";
|
||||
case NV_EVENT_ACCEL: return "NV_EVENT_ACCEL";
|
||||
case NV_EVENT_START: return "NV_EVENT_START";
|
||||
case NV_EVENT_RESTART: return "NV_EVENT_RESTART";
|
||||
case NV_EVENT_RESUME: return "NV_EVENT_RESUME";
|
||||
case NV_EVENT_FOCUS_GAINED: return "NV_EVENT_FOCUS_GAINED";
|
||||
case NV_EVENT_SURFACE_CREATED: return "NV_EVENT_SURFACE_CREATED";
|
||||
case NV_EVENT_SURFACE_SIZE: return "NV_EVENT_SURFACE_SIZE";
|
||||
case NV_EVENT_SURFACE_DESTROYED: return "NV_EVENT_SURFACE_DESTROYED";
|
||||
case NV_EVENT_FOCUS_LOST: return "NV_EVENT_FOCUS_LOST";
|
||||
case NV_EVENT_PAUSE: return "NV_EVENT_PAUSE";
|
||||
case NV_EVENT_STOP: return "NV_EVENT_STOP";
|
||||
case NV_EVENT_QUIT: return "NV_EVENT_QUIT";
|
||||
case NV_EVENT_USER: return "NV_EVENT_USER";
|
||||
}
|
||||
|
||||
// update this if you end up having to edit something.
|
||||
CT_ASSERT(NEED_TO_ADD_STRING_HERE, NV_EVENT_NUM_EVENTS == 17);
|
||||
return "unknown event type!";
|
||||
}
|
||||
|
||||
const NVEvent* NVEventGetNextEvent(int waitMSecs)
|
||||
{
|
||||
return s_eventQueue.RemoveOldest(waitMSecs);
|
||||
}
|
||||
|
||||
void NVEventDoneWithEvent(bool handled)
|
||||
{
|
||||
return s_eventQueue.DoneWithEvent(handled);
|
||||
}
|
||||
|
||||
static void NVEventInsert(NVEvent* ev)
|
||||
{
|
||||
if(!s_appThreadExited)
|
||||
s_eventQueue.Insert(ev);
|
||||
}
|
||||
|
||||
static bool NVEventInsertBlocking(NVEvent* ev)
|
||||
{
|
||||
if(!s_appThreadExited)
|
||||
return s_eventQueue.InsertBlocking(ev);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// Native to Java EGL call-up functions
|
||||
|
||||
bool NVEventInitEGL()
|
||||
{
|
||||
if(s_InitEGL.CallBoolean())
|
||||
{
|
||||
SetAppFlag(NVEVENT_STATUS_EGL_INITIALIZED);
|
||||
return true;
|
||||
}
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
bool NVEventCleanupEGL()
|
||||
{
|
||||
ClearAppFlag(NVEVENT_STATUS_EGL_BOUND);
|
||||
ClearAppFlag(NVEVENT_STATUS_EGL_HAS_SURFACE);
|
||||
ClearAppFlag(NVEVENT_STATUS_EGL_INITIALIZED);
|
||||
return s_CleanupEGL.CallBoolean();
|
||||
}
|
||||
|
||||
bool NVEventCreateSurfaceEGL()
|
||||
{
|
||||
if (s_CreateSurfaceEGL.CallBoolean())
|
||||
{
|
||||
SetAppFlag(NVEVENT_STATUS_EGL_HAS_SURFACE);
|
||||
return true;
|
||||
}
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
bool NVEventDestroySurfaceEGL()
|
||||
{
|
||||
if (!QueryAppFlag(NVEVENT_STATUS_EGL_HAS_SURFACE))
|
||||
return true;
|
||||
|
||||
if (QueryAppFlag(NVEVENT_STATUS_EGL_BOUND))
|
||||
NVEventUnbindSurfaceAndContextEGL();
|
||||
|
||||
ClearAppFlag(NVEVENT_STATUS_EGL_HAS_SURFACE);
|
||||
return s_DestroySurfaceEGL.CallBoolean();
|
||||
}
|
||||
|
||||
bool NVEventBindSurfaceAndContextEGL()
|
||||
{
|
||||
if (s_BindSurfaceAndContextEGL.CallBoolean())
|
||||
{
|
||||
SetAppFlag(NVEVENT_STATUS_EGL_BOUND);
|
||||
return true;
|
||||
}
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
bool NVEventUnbindSurfaceAndContextEGL()
|
||||
{
|
||||
ClearAppFlag(NVEVENT_STATUS_EGL_BOUND);
|
||||
return s_UnbindSurfaceAndContextEGL.CallBoolean();
|
||||
}
|
||||
|
||||
bool NVEventSwapBuffersEGL()
|
||||
{
|
||||
if (!s_SwapBuffersEGL.CallBoolean())
|
||||
return false;
|
||||
RESET_PROFILING();
|
||||
return true;
|
||||
}
|
||||
|
||||
int NVEventGetErrorEGL()
|
||||
{
|
||||
return s_GetErrorEGL.CallInt();
|
||||
}
|
||||
|
||||
bool NVEventReadyToRenderEGL(bool allocateIfNeeded)
|
||||
{
|
||||
// If we have a bound context and surface, then EGL is ready
|
||||
if (!NVEventStatusEGLIsBound())
|
||||
{
|
||||
if (!allocateIfNeeded)
|
||||
return false;
|
||||
|
||||
// If we have not bound the context and surface, do we even _have_ a surface?
|
||||
if (!NVEventStatusEGLHasSurface())
|
||||
{
|
||||
// No surface, so we need to check if EGL is set up at all
|
||||
if (!NVEventStatusEGLInitialized())
|
||||
{
|
||||
if (!NVEventInitEGL())
|
||||
return false;
|
||||
}
|
||||
|
||||
// Create the rendering surface now that we have a context
|
||||
if (!NVEventCreateSurfaceEGL())
|
||||
return false;
|
||||
|
||||
}
|
||||
|
||||
// We have a surface and context, so bind them
|
||||
if (NVEventBindSurfaceAndContextEGL())
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// Input event-related Java to Native callback functions
|
||||
|
||||
static jboolean NVEventTouchEvent(JNIEnv* env, jobject thiz, jint action, jint mx, jint my)
|
||||
{
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_TOUCH;
|
||||
ev.m_data.m_touch.m_action = (NVEVENT_ACTION_UP == action)
|
||||
? NV_TOUCHACTION_UP : (
|
||||
(NVEVENT_ACTION_DOWN == action) ? NV_TOUCHACTION_DOWN : NV_TOUCHACTION_MOVE);
|
||||
ev.m_data.m_touch.m_x = mx;
|
||||
ev.m_data.m_touch.m_y = my;
|
||||
NVEventInsert(&ev);
|
||||
}
|
||||
|
||||
return JNI_TRUE;
|
||||
}
|
||||
|
||||
static jboolean NVEventMultiTouchEvent(JNIEnv* env, jobject thiz, jint action,
|
||||
jboolean hasFirst, jboolean hasSecond, jint mx1, jint my1, jint mx2, jint my2)
|
||||
{
|
||||
{
|
||||
NVEvent ev;
|
||||
|
||||
int actionOnly = action & (~NVEVENT_ACTION_POINTER_INDEX_MASK);
|
||||
|
||||
int maskOnly = 0;
|
||||
|
||||
if (hasFirst)
|
||||
maskOnly |= 0x1;
|
||||
if (hasSecond)
|
||||
maskOnly |= 0x2;
|
||||
|
||||
ev.m_type = NV_EVENT_MULTITOUCH;
|
||||
|
||||
if (actionOnly == NVEVENT_ACTION_UP)
|
||||
{
|
||||
ev.m_data.m_multi.m_action = NV_MULTITOUCH_UP;
|
||||
}
|
||||
else if (actionOnly == NVEVENT_ACTION_DOWN)
|
||||
{
|
||||
ev.m_data.m_multi.m_action = NV_MULTITOUCH_DOWN;
|
||||
}
|
||||
else if (actionOnly == NVEVENT_ACTION_CANCEL)
|
||||
{
|
||||
ev.m_data.m_multi.m_action = NV_MULTITOUCH_CANCEL;
|
||||
}
|
||||
else
|
||||
{
|
||||
ev.m_data.m_multi.m_action = NV_MULTITOUCH_MOVE;
|
||||
}
|
||||
ev.m_data.m_multi.m_action =
|
||||
(NVMultiTouchEventType)(ev.m_data.m_multi.m_action | (maskOnly << NV_MULTITOUCH_POINTER_SHIFT));
|
||||
ev.m_data.m_multi.m_x1 = mx1;
|
||||
ev.m_data.m_multi.m_y1 = my1;
|
||||
ev.m_data.m_multi.m_x2 = mx2;
|
||||
ev.m_data.m_multi.m_y2 = my2;
|
||||
NVEventInsert(&ev);
|
||||
}
|
||||
|
||||
return JNI_TRUE;
|
||||
}
|
||||
|
||||
static jboolean NVEventKeyEvent(JNIEnv* env, jobject thiz, jint action, jint keycode, jint unichar)
|
||||
{
|
||||
// TBD - remove these or make them resettable for safety...
|
||||
static int lastKeyAction = 0;
|
||||
static int lastKeyCode = 0;
|
||||
bool ret = false;
|
||||
|
||||
NVKeyCode code = NV_KEYCODE_NULL;
|
||||
|
||||
if (s_keyMapping.MapKey((int)keycode, code))
|
||||
{
|
||||
if ((code != NV_KEYCODE_NULL) &&
|
||||
((code != lastKeyCode) || (action != lastKeyAction)))
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_KEY;
|
||||
ev.m_data.m_key.m_action = (NVEVENT_ACTION_UP == action)
|
||||
? NV_KEYACTION_UP : NV_KEYACTION_DOWN;
|
||||
ev.m_data.m_key.m_code = code;
|
||||
ret = NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
lastKeyAction = action;
|
||||
lastKeyCode = code;
|
||||
}
|
||||
|
||||
if (unichar && (NVEVENT_ACTION_UP != action))
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_CHAR;
|
||||
ev.m_data.m_char.m_unichar = unichar;
|
||||
NVEventInsert(&ev);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static jboolean NVEventAccelerometerEvent(JNIEnv* env, jobject thiz, jfloat values0, jfloat values1, jfloat values2)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_ACCEL;
|
||||
ev.m_data.m_accel.m_x = values0;
|
||||
ev.m_data.m_accel.m_y = values1;
|
||||
ev.m_data.m_accel.m_z = values2;
|
||||
NVEventInsert(&ev);
|
||||
return JNI_TRUE;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// Java to Native app lifecycle callback functions
|
||||
|
||||
static jboolean onCreateNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
ZeroAppFlags();
|
||||
|
||||
if (!s_globalThiz)
|
||||
{
|
||||
s_globalThiz = env->NewGlobalRef(thiz);
|
||||
if (!s_globalThiz)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error: Thiz NewGlobalRef failed!");
|
||||
}
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Thiz NewGlobalRef: 0x%p", s_globalThiz);
|
||||
}
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Init KeyCode Map");
|
||||
s_keyMapping.Init(env, thiz);
|
||||
|
||||
NVEventInitInputFields(env);
|
||||
|
||||
s_eventQueue.Init();
|
||||
|
||||
s_javaPostedQuit = false;
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Calling NVEventAppInit");
|
||||
|
||||
if (NVEventAppInit(0, NULL))
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "NVEventAppInit error");
|
||||
return JNI_FALSE;
|
||||
}
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "spawning thread");
|
||||
|
||||
s_appThreadExited = false;
|
||||
SetAppFlag(NVEVENT_STATUS_RUNNING);
|
||||
|
||||
/* Launch thread with main loop */
|
||||
NVThreadSpawnJNIThread(&s_mainThread, NULL, NVEventMainLoopThreadFunc, NULL);
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "thread spawned");
|
||||
|
||||
return JNI_TRUE;
|
||||
}
|
||||
|
||||
static jboolean onStartNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_START;
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onRestartNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_RESTART;
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onResumeNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_RESUME;
|
||||
SetAppFlag(NVEVENT_STATUS_ACTIVE);
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onSurfaceCreatedNative(JNIEnv* env, jobject thiz, int w, int h)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_SURFACE_CREATED;
|
||||
ev.m_data.m_size.m_w = w;
|
||||
ev.m_data.m_size.m_h = h;
|
||||
if ((w > 0) && (h > 0))
|
||||
SetAppFlag(NVEVENT_STATUS_HAS_REAL_SURFACE);
|
||||
else
|
||||
ClearAppFlag(NVEVENT_STATUS_HAS_REAL_SURFACE);
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onFocusChangedNative(JNIEnv* env, jobject thiz, jboolean focused)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = (focused == JNI_TRUE) ? NV_EVENT_FOCUS_GAINED : NV_EVENT_FOCUS_LOST;
|
||||
if (focused)
|
||||
SetAppFlag(NVEVENT_STATUS_FOCUSED);
|
||||
else
|
||||
ClearAppFlag(NVEVENT_STATUS_FOCUSED);
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onSurfaceChangedNative(JNIEnv* env, jobject thiz, int w, int h)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_SURFACE_SIZE;
|
||||
ev.m_data.m_size.m_w = w;
|
||||
ev.m_data.m_size.m_h = h;
|
||||
if (w * h)
|
||||
SetAppFlag(NVEVENT_STATUS_HAS_REAL_SURFACE);
|
||||
else
|
||||
ClearAppFlag(NVEVENT_STATUS_HAS_REAL_SURFACE);
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onSurfaceDestroyedNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_SURFACE_DESTROYED;
|
||||
ClearAppFlag(NVEVENT_STATUS_HAS_REAL_SURFACE);
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onPauseNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
// TODO: we could selectively flush here to
|
||||
// improve responsiveness to the pause
|
||||
s_eventQueue.Flush();
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_PAUSE;
|
||||
ClearAppFlag(NVEVENT_STATUS_ACTIVE);
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onStopNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_STOP;
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
|
||||
static jboolean onDestroyNative(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
if (!env || !s_globalThiz)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error: DestroyingRegisteredObjectInstance no TLS data!");
|
||||
}
|
||||
|
||||
if (!s_appThreadExited)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Posting quit event");
|
||||
|
||||
// flush ALL events
|
||||
s_eventQueue.Flush();
|
||||
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_QUIT;
|
||||
ClearAppFlag(NVEVENT_STATUS_RUNNING);
|
||||
|
||||
// We're posting quit, so we need to mark that; when the main loop
|
||||
// thread exits, we check this flag to ensure that we only call "finish"
|
||||
// if the app returned of its own accord, not if we posted it
|
||||
s_javaPostedQuit = true;
|
||||
NVEventInsert(&ev);
|
||||
|
||||
// ensure that the native side
|
||||
// isn't blocked waiting for an event -- since we've flushed
|
||||
// all the events save quit, we must artificially unblock native
|
||||
s_eventQueue.UnblockConsumer();
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Waiting for main loop exit");
|
||||
pthread_join(s_mainThread, NULL);
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Main loop exited");
|
||||
}
|
||||
|
||||
env->DeleteGlobalRef(s_globalThiz);
|
||||
s_globalThiz = NULL;
|
||||
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Released global thiz!");
|
||||
|
||||
s_eventQueue.Shutdown();
|
||||
|
||||
return JNI_TRUE;
|
||||
}
|
||||
|
||||
static jboolean postUserEvent(JNIEnv* env, jobject thiz,
|
||||
jint u0, jint u1, jint u2, jint u3,
|
||||
jboolean blocking)
|
||||
{
|
||||
NVEvent ev;
|
||||
ev.m_type = NV_EVENT_USER;
|
||||
ev.m_data.m_user.m_u0 = u0;
|
||||
ev.m_data.m_user.m_u1 = u1;
|
||||
ev.m_data.m_user.m_u2 = u2;
|
||||
ev.m_data.m_user.m_u3 = u3;
|
||||
if (blocking == JNI_TRUE)
|
||||
{
|
||||
return NVEventInsertBlocking(&ev);
|
||||
}
|
||||
else
|
||||
{
|
||||
NVEventInsert(&ev);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// File and APK handling functions
|
||||
|
||||
char* NVEventLoadFile(const char* file)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "loadFile is not implemented");
|
||||
return 0;
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// JVM Initialization functions
|
||||
|
||||
void InitNVEvent(JavaVM* vm)
|
||||
{
|
||||
JNIEnv *env;
|
||||
|
||||
NVThreadInit(vm);
|
||||
|
||||
NVDEBUG("InitNVEvent called");
|
||||
|
||||
if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK)
|
||||
{
|
||||
NVDEBUG("Failed to get the environment using GetEnv()");
|
||||
return;
|
||||
}
|
||||
|
||||
JNINativeMethod methods[] =
|
||||
{
|
||||
{
|
||||
"onCreateNative",
|
||||
"()Z",
|
||||
(void *) onCreateNative
|
||||
},
|
||||
{
|
||||
"onStartNative",
|
||||
"()Z",
|
||||
(void *) onStartNative
|
||||
},
|
||||
{
|
||||
"onRestartNative",
|
||||
"()Z",
|
||||
(void *) onRestartNative
|
||||
},
|
||||
{
|
||||
"onResumeNative",
|
||||
"()Z",
|
||||
(void *) onResumeNative
|
||||
},
|
||||
{
|
||||
"onSurfaceCreatedNative",
|
||||
"(II)Z",
|
||||
(void *) onSurfaceCreatedNative
|
||||
},
|
||||
{
|
||||
"onFocusChangedNative",
|
||||
"(Z)Z",
|
||||
(void *) onFocusChangedNative
|
||||
},
|
||||
{
|
||||
"onSurfaceChangedNative",
|
||||
"(II)Z",
|
||||
(void *) onSurfaceChangedNative
|
||||
},
|
||||
{
|
||||
"onSurfaceDestroyedNative",
|
||||
"()Z",
|
||||
(void *) onSurfaceDestroyedNative
|
||||
},
|
||||
{
|
||||
"onPauseNative",
|
||||
"()Z",
|
||||
(void *) onPauseNative
|
||||
},
|
||||
{
|
||||
"onStopNative",
|
||||
"()Z",
|
||||
(void *) onStopNative
|
||||
},
|
||||
{
|
||||
"onDestroyNative",
|
||||
"()Z",
|
||||
(void *) onDestroyNative
|
||||
},
|
||||
{
|
||||
"postUserEvent",
|
||||
"(IIIIZ)Z",
|
||||
(void *) postUserEvent
|
||||
},
|
||||
{
|
||||
"touchEvent",
|
||||
"(IIILandroid/view/MotionEvent;)Z",
|
||||
(void *) NVEventTouchEvent
|
||||
},
|
||||
{
|
||||
"multiTouchEvent",
|
||||
"(IZZIIIILandroid/view/MotionEvent;)Z",
|
||||
(void *) NVEventMultiTouchEvent
|
||||
},
|
||||
{
|
||||
"keyEvent",
|
||||
"(IIILandroid/view/KeyEvent;)Z",
|
||||
(void *) NVEventKeyEvent
|
||||
},
|
||||
{
|
||||
"accelerometerEvent",
|
||||
"(FFF)Z",
|
||||
(void *) NVEventAccelerometerEvent
|
||||
},
|
||||
|
||||
// TODO TBD - this should be done in NVTimeInit(), but we use a different
|
||||
// class than most apps. Need to clean this up, as it is fragile w.r.t.
|
||||
// changes in nv_time
|
||||
{
|
||||
"nvAcquireTimeExtension",
|
||||
"()V",
|
||||
(void *) nvAcquireTimeExtensionJNI
|
||||
},
|
||||
{
|
||||
"nvGetSystemTime",
|
||||
"()J",
|
||||
(void *) nvGetSystemTimeJNI
|
||||
},
|
||||
};
|
||||
|
||||
jclass k;
|
||||
k = (env)->FindClass ("com/nvidia/devtech/NvEventQueueActivity");
|
||||
(env)->RegisterNatives(k, methods, dimof(methods));
|
||||
|
||||
s_InitEGL.QueryID(env, k);
|
||||
s_CleanupEGL.QueryID(env, k);
|
||||
s_CreateSurfaceEGL.QueryID(env, k);
|
||||
s_DestroySurfaceEGL.QueryID(env, k);
|
||||
s_SwapBuffersEGL.QueryID(env, k);
|
||||
s_BindSurfaceAndContextEGL.QueryID(env, k);
|
||||
s_UnbindSurfaceAndContextEGL.QueryID(env, k);
|
||||
s_GetErrorEGL.QueryID(env, k);
|
||||
s_finish.QueryID(env, k);
|
||||
}
|
||||
|
530
android/jni/nv_event/nv_event.hpp
Normal file
530
android/jni/nv_event/nv_event.hpp
Normal file
|
@ -0,0 +1,530 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_event\nv_event.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
#ifndef _NV_EVENT_H
|
||||
#define _NV_EVENT_H
|
||||
#include <jni.h>
|
||||
#include <sys/types.h>
|
||||
//#include <EGL/egl.h>
|
||||
|
||||
/** @file nv_event.h
|
||||
Contains a framework for event loop-based applications. This library replaces
|
||||
most or all of the normally-required JNI code for Android NDK applications,
|
||||
presenting the application developer with two callbacks into which they can
|
||||
write their application. The framework runs in a natively-created thread,
|
||||
allowing the application to implement a classic "event loop and render" structure
|
||||
without having to return to Java code to avoid ANR warnings. The library
|
||||
includes support for input and system event passing as well as JNI initialization
|
||||
and exposes basic EGL functionality to native as well. Put together, the library
|
||||
can form the basis of a simple interactive 3D application. All applications that
|
||||
are based on this library must also be subclassed on the Java side from the
|
||||
provided NvEventQueueActivity. Additional external documentation on the use of
|
||||
this library, the related Java code and the tool provided to create a template
|
||||
application based on the library are provided with the SDK.
|
||||
@see NvEventQueueActivity
|
||||
*/
|
||||
|
||||
#ifdef ANDROID
|
||||
|
||||
#include <jni.h>
|
||||
typedef jobject NVEventPlatformAppHandle;
|
||||
|
||||
#else // unknown platform
|
||||
|
||||
typedef void* NVEventPlatformAppHandle;
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
enum
|
||||
{
|
||||
/** Timeout argument to NVEventGetNextEvent() that indicates the function should
|
||||
block until there is an event pending or the app exits
|
||||
@see NVEventGetNextEvent()
|
||||
*/
|
||||
NV_EVENT_WAIT_FOREVER = -1
|
||||
};
|
||||
|
||||
/** Event type values
|
||||
*/
|
||||
typedef enum NVEventType
|
||||
{
|
||||
/** Key up/down events */
|
||||
NV_EVENT_KEY = 1,
|
||||
/** Translated character events */
|
||||
NV_EVENT_CHAR,
|
||||
/** Single-touch pointer events */
|
||||
NV_EVENT_TOUCH,
|
||||
/** Multi-touch events */
|
||||
NV_EVENT_MULTITOUCH,
|
||||
/** Accelerometer events */
|
||||
NV_EVENT_ACCEL,
|
||||
/** onStart lifecycle events */
|
||||
NV_EVENT_START,
|
||||
/** onRestart lifecycle events */
|
||||
NV_EVENT_RESTART,
|
||||
/** onResume lifecycle events */
|
||||
NV_EVENT_RESUME,
|
||||
/** onWindowFocusChanged(TRUE) lifecycle events */
|
||||
NV_EVENT_FOCUS_GAINED,
|
||||
/** Surface created events */
|
||||
NV_EVENT_SURFACE_CREATED,
|
||||
/** Surface size changed events */
|
||||
NV_EVENT_SURFACE_SIZE,
|
||||
/** Surface destroyed events */
|
||||
NV_EVENT_SURFACE_DESTROYED,
|
||||
/** onWindowFocusChanged(FALSE) lifecycle events */
|
||||
NV_EVENT_FOCUS_LOST,
|
||||
/** onPause lifecycle events */
|
||||
NV_EVENT_PAUSE,
|
||||
/** onStop lifecycle events */
|
||||
NV_EVENT_STOP,
|
||||
/** (onDestroy) Quit request events */
|
||||
NV_EVENT_QUIT,
|
||||
/** App-specific events */
|
||||
NV_EVENT_USER,
|
||||
/* a dummy enum value used to compute num_events */
|
||||
NV_EVENT_NUM_EVENT_DUMMY_DONTUSEME,
|
||||
/* total number of events */
|
||||
NV_EVENT_NUM_EVENTS = NV_EVENT_NUM_EVENT_DUMMY_DONTUSEME - 1,
|
||||
NV_EVENT_FORCE_32BITS = 0x7fffffff
|
||||
} NVEventType;
|
||||
|
||||
/** Touch event actions
|
||||
*/
|
||||
typedef enum NVTouchEventType
|
||||
{
|
||||
/** Pointer has just left the screen */
|
||||
NV_TOUCHACTION_UP,
|
||||
/** Pointer has just gone down onto the screen */
|
||||
NV_TOUCHACTION_DOWN,
|
||||
/** Pointer is moving on the screen */
|
||||
NV_TOUCHACTION_MOVE,
|
||||
NV_TOUCHACTION_FORCE_32BITS = 0x7fffffff
|
||||
} NVTouchEventType;
|
||||
|
||||
/** Multitouch event flags
|
||||
*/
|
||||
typedef enum NVMultiTouchEventType
|
||||
{
|
||||
/** Indicated pointers are leaving the screen */
|
||||
NV_MULTITOUCH_UP = 0x00000001,
|
||||
/** Indicated pointers have just touched the screen */
|
||||
NV_MULTITOUCH_DOWN = 0x00000002,
|
||||
/** Indicated pointers are moving on the screen */
|
||||
NV_MULTITOUCH_MOVE = 0x00000003,
|
||||
/** Indicated pointers have halted the current gesture
|
||||
app should cancel any actions implied by the gesture */
|
||||
NV_MULTITOUCH_CANCEL = 0x00000004,
|
||||
/** Mask to be AND'ed with the flag value
|
||||
to get the active pointer bits */
|
||||
NV_MULTITOUCH_POINTER_MASK = 0x0000ff00,
|
||||
/** Number of bits to right-shift the masked value
|
||||
to get the active pointer bits */
|
||||
NV_MULTITOUCH_POINTER_SHIFT = 0x00000008,
|
||||
/** Mask to be AND'ed with the flag value
|
||||
to get the event action */
|
||||
NV_MULTITOUCH_ACTION_MASK = 0x000000ff,
|
||||
NV_MULTITOUCH_FORCE_32BITS = 0x7fffffff
|
||||
} NVMultiTouchEventType;
|
||||
|
||||
/** Key event types
|
||||
*/
|
||||
typedef enum NVKeyEventType
|
||||
{
|
||||
/** Key has just been pressed (no repeats) */
|
||||
NV_KEYACTION_UP,
|
||||
/** Key has just been released */
|
||||
NV_KEYACTION_DOWN,
|
||||
NV_KEYACTION_FORCE_32BITS = 0x7fffffff
|
||||
} NVKeyEventType;
|
||||
|
||||
/** Key event key codes
|
||||
*/
|
||||
#define NV_MAX_KEYCODE 256
|
||||
typedef enum NVKeyCode
|
||||
{
|
||||
NV_KEYCODE_NULL = 0,
|
||||
NV_KEYCODE_BACK,
|
||||
NV_KEYCODE_TAB,
|
||||
NV_KEYCODE_ENTER,
|
||||
NV_KEYCODE_DEL,
|
||||
NV_KEYCODE_SPACE,
|
||||
NV_KEYCODE_ENDCALL,
|
||||
NV_KEYCODE_HOME,
|
||||
|
||||
NV_KEYCODE_STAR,
|
||||
NV_KEYCODE_PLUS,
|
||||
NV_KEYCODE_MINUS,
|
||||
NV_KEYCODE_NUM,
|
||||
|
||||
NV_KEYCODE_DPAD_LEFT,
|
||||
NV_KEYCODE_DPAD_UP,
|
||||
NV_KEYCODE_DPAD_RIGHT,
|
||||
NV_KEYCODE_DPAD_DOWN,
|
||||
|
||||
NV_KEYCODE_0,
|
||||
NV_KEYCODE_1,
|
||||
NV_KEYCODE_2,
|
||||
NV_KEYCODE_3,
|
||||
NV_KEYCODE_4,
|
||||
NV_KEYCODE_5,
|
||||
NV_KEYCODE_6,
|
||||
NV_KEYCODE_7,
|
||||
NV_KEYCODE_8,
|
||||
NV_KEYCODE_9,
|
||||
|
||||
NV_KEYCODE_A,
|
||||
NV_KEYCODE_B,
|
||||
NV_KEYCODE_C,
|
||||
NV_KEYCODE_D,
|
||||
NV_KEYCODE_E,
|
||||
NV_KEYCODE_F,
|
||||
NV_KEYCODE_G,
|
||||
NV_KEYCODE_H,
|
||||
NV_KEYCODE_I,
|
||||
NV_KEYCODE_J,
|
||||
NV_KEYCODE_K,
|
||||
NV_KEYCODE_L,
|
||||
NV_KEYCODE_M,
|
||||
NV_KEYCODE_N,
|
||||
NV_KEYCODE_O,
|
||||
NV_KEYCODE_P,
|
||||
NV_KEYCODE_Q,
|
||||
NV_KEYCODE_R,
|
||||
NV_KEYCODE_S,
|
||||
NV_KEYCODE_T,
|
||||
NV_KEYCODE_U,
|
||||
NV_KEYCODE_V,
|
||||
NV_KEYCODE_W,
|
||||
NV_KEYCODE_X,
|
||||
NV_KEYCODE_Y,
|
||||
NV_KEYCODE_Z,
|
||||
|
||||
NV_KEYCODE_ALT_LEFT,
|
||||
NV_KEYCODE_ALT_RIGHT,
|
||||
|
||||
NV_KEYCODE_SHIFT_LEFT,
|
||||
NV_KEYCODE_SHIFT_RIGHT,
|
||||
|
||||
NV_KEYCODE_APOSTROPHE,
|
||||
NV_KEYCODE_SEMICOLON,
|
||||
NV_KEYCODE_EQUALS,
|
||||
NV_KEYCODE_COMMA,
|
||||
NV_KEYCODE_PERIOD,
|
||||
NV_KEYCODE_SLASH,
|
||||
NV_KEYCODE_GRAVE,
|
||||
NV_KEYCODE_BACKSLASH,
|
||||
|
||||
NV_KEYCODE_LEFT_BRACKET,
|
||||
NV_KEYCODE_RIGHT_BRACKET,
|
||||
|
||||
NV_KEYCODE_FORCE_32BIT = 0x7fffffff
|
||||
} NVKeyCode;
|
||||
|
||||
/** Single-touch event data
|
||||
*/
|
||||
typedef struct NVEventTouch
|
||||
{
|
||||
/** The action code */
|
||||
NVTouchEventType m_action;
|
||||
/** The window-relative X position (in pixels) */
|
||||
float m_x;
|
||||
/** The window-relative Y position (in pixels) */
|
||||
float m_y;
|
||||
} NVEventTouch;
|
||||
|
||||
/** Multi-touch event data
|
||||
*/
|
||||
typedef struct NVEventMultiTouch
|
||||
{
|
||||
/** The action flags */
|
||||
NVMultiTouchEventType m_action;
|
||||
/** The window-relative X position of the first pointer (in pixels)
|
||||
only valid if bit 0 of the pointer bits is set */
|
||||
float m_x1;
|
||||
/** The window-relative Y position of the first pointer (in pixels)
|
||||
only valid if bit 0 of the pointer bits is set */
|
||||
float m_y1;
|
||||
/** The window-relative X position of the second pointer (in pixels)
|
||||
only valid if bit 1 of the pointer bits is set */
|
||||
float m_x2;
|
||||
/** The window-relative Y position of the second pointer (in pixels)
|
||||
only valid if bit 1 of the pointer bits is set */
|
||||
float m_y2;
|
||||
} NVEventMultiTouch;
|
||||
|
||||
/** Key down/up event data
|
||||
*/
|
||||
typedef struct NVEventKey
|
||||
{
|
||||
/** The action flags */
|
||||
NVKeyEventType m_action;
|
||||
/** The code of the key pressed or released */
|
||||
NVKeyCode m_code;
|
||||
} NVEventKey;
|
||||
|
||||
/** Translated character event data
|
||||
*/
|
||||
typedef struct NVEventChar
|
||||
{
|
||||
/** The UNICODE character represented */
|
||||
int32_t m_unichar;
|
||||
} NVEventChar;
|
||||
|
||||
/** Accelerometer event data
|
||||
*/
|
||||
typedef struct NVEventAccel
|
||||
{
|
||||
/** Signed X magnitude of the force vector */
|
||||
float m_x;
|
||||
/** Signed Y magnitude of the force vector */
|
||||
float m_y;
|
||||
/** Signed Z magnitude of the force vector */
|
||||
float m_z;
|
||||
} NVEventAccel;
|
||||
|
||||
/** Surface size change event data
|
||||
*/
|
||||
typedef struct NVEventSurfaceSize
|
||||
{
|
||||
/** New surface client area width (in pixels) */
|
||||
int32_t m_w;
|
||||
/** New surface client area height (in pixels) */
|
||||
int32_t m_h;
|
||||
} NVEventSurfaceSize;
|
||||
|
||||
/** User/App-specific event data
|
||||
*/
|
||||
typedef struct NVEventUser
|
||||
{
|
||||
/** First 32-bit user data item */
|
||||
int32_t m_u0;
|
||||
/** Second 32-bit user data item */
|
||||
int32_t m_u1;
|
||||
/** Third 32-bit user data item */
|
||||
int32_t m_u2;
|
||||
/** Fourth 32-bit user data item */
|
||||
int32_t m_u3;
|
||||
} NVEventUser;
|
||||
|
||||
/** All-encompassing event structure
|
||||
*/
|
||||
typedef struct NVEvent
|
||||
{
|
||||
/** The type of the event, which also indicates which m_data union holds the data */
|
||||
NVEventType m_type;
|
||||
/** Union containing all possible event type data */
|
||||
union NVEventData
|
||||
{
|
||||
/** Data for single-touch events */
|
||||
NVEventTouch m_touch;
|
||||
/** Data for multi-touch events */
|
||||
NVEventMultiTouch m_multi;
|
||||
/** Data for key up/down events */
|
||||
NVEventKey m_key;
|
||||
/** Data for charcter events */
|
||||
NVEventChar m_char;
|
||||
/** Data for accelerometer events */
|
||||
NVEventAccel m_accel;
|
||||
/** Data for surface size events */
|
||||
NVEventSurfaceSize m_size;
|
||||
/** Data for user/app events */
|
||||
NVEventUser m_user;
|
||||
} m_data;
|
||||
} NVEvent;
|
||||
|
||||
/** Returns a string describing the event
|
||||
@param eventType The event type
|
||||
@return Returns a string containing a description of the event. Do not free or delete this memory.
|
||||
@see NVEvent */
|
||||
const char* NVEventGetEventStr(NVEventType eventType);
|
||||
|
||||
/** Returns the next pending event for the application to process. Can return immediately if there
|
||||
is no event, or can wait a fixed number of milisecs (or "forever") if desired.
|
||||
The application should always pair calls to this function that return non-NULL events with calls
|
||||
to NVEventDoneWithEvent()
|
||||
@param waitMSecs The maximum time (in milisecs) to wait for an event before returning "no event".
|
||||
Pass NV_EVENT_WAIT_FOREVER to wait indefinitely for an event. Note that NV_EVENT_WAIT_FOREVER
|
||||
does not gaurantee an event on return. The function can still return on error or if the
|
||||
app is exiting. Default is to return immediately, event or not.
|
||||
@return Non-NULL pointer to a constant event structure if an event was pending, NULL if no event was
|
||||
pending in the requested timeout period
|
||||
@see NVEvent
|
||||
@see NVEventDoneWithEvent
|
||||
*/
|
||||
const NVEvent* NVEventGetNextEvent(int waitMSecs = 0);
|
||||
|
||||
/** Indicates that the application has finished handling the event returned from the last
|
||||
call to NVEventGetNextEvent. This function should always be called prior to the next call
|
||||
to NVEventGetNextEvent. If the current event is a blocking event, this call will unblock
|
||||
the posting thread (normally in Java). This is particularly important for application
|
||||
lifecycle events like onPause, as calling this function indicates that the native code has
|
||||
completed the handling of the lifecycle callback. Failure to call this function promptly
|
||||
for all events can lead to Application Not Responding errors.
|
||||
@param handled The return value that should be passed back to Java for blocking events. For non-blocking
|
||||
events, this parameter is discard.
|
||||
@see NVEvent
|
||||
@see NVEventGetNextEvent
|
||||
*/
|
||||
void NVEventDoneWithEvent(bool handled);
|
||||
|
||||
/** The app-supplied "callback" for initialization during JNI_OnLoad.
|
||||
Declares the application's pre-main initialization function. Does not define the
|
||||
function. <b>The app must define this in its own code</b>, even if the function is empty.
|
||||
JNI init code can be safely called from here, as it WILL be called from
|
||||
within a JNI function thread
|
||||
@parm argc Passes the number of command line arguments.
|
||||
This is currently unsupported and is always passed 0
|
||||
@parm argv Passes the array of command line arguments.
|
||||
This is currently unsupported and is always passed NULL
|
||||
@return The function should return 0 on success and nonzero on failure.
|
||||
*/
|
||||
extern int32_t NVEventAppInit(int32_t argc, char** argv);
|
||||
|
||||
/** The app-supplied "callback" for running the application's main loop.
|
||||
Declares the application's main loop function. Does not define the
|
||||
function. <b>The app must define this in its own code</b>
|
||||
This function will be spawned in its own thread.
|
||||
@parm argc Passes the number of command line arguments.
|
||||
This is currently unsupported and is always passed 0
|
||||
@parm argv Passes the array of command line arguments.
|
||||
This is currently unsupported and is always passed NULL
|
||||
@return The function should return 0 on success and nonzero on failure.
|
||||
*/
|
||||
extern int32_t NVEventAppMain(int32_t argc, char** argv);
|
||||
|
||||
|
||||
/** Initializes EGL, queries a valid ES2 config and creates (but does not bind)
|
||||
an ES2-compatible EGLContext
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventInitEGL();
|
||||
|
||||
/** Releases any existing EGLSurface and EGLContext and terminates EGL
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventCleanupEGL();
|
||||
|
||||
/** Creates an EGLSurface for the current Android surface. Will attempt to initialize
|
||||
EGL and an EGLContext if not already done. Fails if there is no valid Android surface
|
||||
or if there is an EGL error.
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventCreateSurfaceEGL();
|
||||
|
||||
/** Unbinds (if needed) and releases the app's EGLSurface
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventDestroySurfaceEGL();
|
||||
|
||||
/** Binds the app's EGLSurface and EGLContext to the calling thread. The EGLSurface and
|
||||
EGLContext must both exist already, or else the function will fail.
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventBindSurfaceAndContextEGL();
|
||||
|
||||
/** Un-binds the app's EGLSurface and EGLContext from the calling thread.
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventUnbindSurfaceAndContextEGL();
|
||||
|
||||
/** Swaps the currently-bound EGLSurface
|
||||
@return true on success, false on failure
|
||||
*/
|
||||
bool NVEventSwapBuffersEGL();
|
||||
|
||||
/** Accessor for the last EGL error
|
||||
@return the EGL error
|
||||
*/
|
||||
int NVEventGetErrorEGL();
|
||||
|
||||
/** Utility function: checks if EGl is completely ready to render, including
|
||||
initialization, surface creation and context/surface binding.
|
||||
@parm allocateIfNeeded If the parameter is false, then the function immediately returns if any
|
||||
part of the requirements have not already been satisfied. If the parameter is
|
||||
true, then the function attempts to initialize any of the steps needed, failing
|
||||
and returning false only if a step cannot be completed at this time.
|
||||
@return The function returns true if EGL/GLES is ready to render/load content (i.e.
|
||||
a context and surface are bound) and false it not
|
||||
*/
|
||||
bool NVEventReadyToRenderEGL(bool allocateIfNeeded);
|
||||
|
||||
/** Convenience conditional function to determine if the app is between onCreate
|
||||
and onDestroy callbacks (i.e. the app is in a running state).
|
||||
@return true if the application is between onCreate and onDestroy and false after
|
||||
an onDestroy event has been delivered.
|
||||
*/
|
||||
bool NVEventStatusIsRunning();
|
||||
|
||||
/** Convenience conditional function to determine if the app is between onResume and onPause
|
||||
callbacks (i.e. the app is in a "resumed" state).
|
||||
@return true if the application is between onResume and onPause and false if the application
|
||||
has not yet been resumed, or is currently paused.
|
||||
*/
|
||||
bool NVEventStatusIsActive();
|
||||
|
||||
/** Convenience conditional function to determine if the app's window is focused (between
|
||||
calls to onWindowFocusChanged(true) and onWindowFocusChanged(false))
|
||||
@return true between onWindowFocusChanged(true) and onWindowFocusChanged(false)
|
||||
*/
|
||||
bool NVEventStatusIsFocused();
|
||||
|
||||
/** Convenience conditional function to determine if the app has a surface and that surface
|
||||
has non-zero area
|
||||
@return true if the app is between surfaceCreated and surfaceDestroyed callbacks and
|
||||
the surface has non-zero pixel area (not 0x0 pixels)
|
||||
*/
|
||||
bool NVEventStatusHasRealSurface();
|
||||
|
||||
/** Convenience conditional function to determine if the app is in a fully-focused, visible
|
||||
state. This is a logical "AND" of IsRunning, IsActive, IsFocused and HasRealSurface
|
||||
@return true if IsRunning, IsActive, IsFocused and HasRealSurface are all currently true,
|
||||
false otherwise
|
||||
*/
|
||||
bool NVEventStatusIsInteractable();
|
||||
|
||||
/** Convenience conditional function to determine if the app has active EGL
|
||||
@return true between successful calls to NVEventInitEGL and NVEventCleanupEGL,
|
||||
false otherwise
|
||||
*/
|
||||
bool NVEventStatusEGLInitialized();
|
||||
|
||||
/** Convenience conditional function to determine if the app has an EGLSurface (need not be bound)
|
||||
@return true if the EGLSurface for the app is allocated, false otherwise
|
||||
*/
|
||||
bool NVEventStatusEGLHasSurface();
|
||||
|
||||
/** Convenience conditional function to determine if the app has an EGLSurface and EGLContext
|
||||
and they are bound
|
||||
@return true if the EGLSurface and EGLContext for the app are allocated and bound, false otherwise
|
||||
*/
|
||||
bool NVEventStatusEGLIsBound();
|
||||
|
||||
|
||||
/** Returns the platform-specific handle to the application instance, if supported. This
|
||||
function is, by definition platform-specific.
|
||||
@return A platform-specific handle to the application. */
|
||||
NVEventPlatformAppHandle NVEventGetPlatformAppHandle();
|
||||
|
||||
void InitNVEvent(JavaVM * jvm);
|
||||
|
||||
#endif
|
277
android/jni/nv_event/nv_event_queue.cpp
Normal file
277
android/jni/nv_event/nv_event_queue.cpp
Normal file
|
@ -0,0 +1,277 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_event\nv_event_queue.cpp
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
#define MODULE "NVEventQueue"
|
||||
#define DBG_DETAILED 1
|
||||
#include "../nv_debug/nv_debug.hpp"
|
||||
|
||||
#include "nv_event_queue.hpp"
|
||||
#include <assert.h>
|
||||
#include <stdlib.h>
|
||||
#include <jni.h>
|
||||
#include <pthread.h>
|
||||
#include <android/log.h>
|
||||
|
||||
#define NVNextWrapped(index) (((index) + 1) & QUEUE_MASK)
|
||||
#define NVPrevWrapped(index) (((index) - 1) & QUEUE_MASK)
|
||||
|
||||
/* you must be inside a m_accessLock lock to invoke this! */
|
||||
static void unlockAll(NVEventSync* sem)
|
||||
{
|
||||
sem->m_block = false;
|
||||
pthread_cond_broadcast(&(sem->m_cond));
|
||||
}
|
||||
|
||||
/* you must be inside mutex lock to invoke this! */
|
||||
static int32_t wait(NVEventSync* sem, pthread_mutex_t* mutex, int waitMS)
|
||||
{
|
||||
// TBD - spec is dodgy; do we definitely release the mutex even if
|
||||
// wait fails?
|
||||
if(sem->m_block)
|
||||
{
|
||||
if( waitMS < 0 )
|
||||
return pthread_cond_wait(&sem->m_cond, mutex);
|
||||
else
|
||||
return pthread_cond_timeout_np(&sem->m_cond, mutex, (unsigned)waitMS);
|
||||
}
|
||||
else
|
||||
{
|
||||
// must release this, as failure assumes no lock
|
||||
pthread_mutex_unlock(mutex);
|
||||
return 1; // do not return 0 - we do not own the lock!
|
||||
}
|
||||
}
|
||||
|
||||
static void signal(NVEventSync* sem)
|
||||
{
|
||||
pthread_cond_signal(&sem->m_cond);
|
||||
}
|
||||
|
||||
static void broadcast(NVEventSync* sem)
|
||||
{
|
||||
pthread_cond_broadcast(&sem->m_cond);
|
||||
}
|
||||
|
||||
static void syncInit( NVEventSync* sync )
|
||||
{
|
||||
pthread_cond_init(&(sync->m_cond), NULL);
|
||||
sync->m_block = true;
|
||||
}
|
||||
|
||||
static void syncDestroy( NVEventSync* sync )
|
||||
{
|
||||
pthread_cond_destroy( &sync->m_cond );
|
||||
}
|
||||
|
||||
/* you must be inside a m_accessLock lock to invoke this! */
|
||||
bool NVEventQueue::insert(const NVEvent* ev)
|
||||
{
|
||||
// Is the queue full?
|
||||
int32_t nextNext = NVNextWrapped(m_nextInsertIndex);
|
||||
if (nextNext == m_headIndex)
|
||||
{
|
||||
// TBD - what to do when we cannot insert (full queue)
|
||||
return false;
|
||||
}
|
||||
|
||||
NVEvent* evDest = m_events + m_nextInsertIndex;
|
||||
memcpy(evDest, ev, sizeof(NVEvent));
|
||||
|
||||
m_nextInsertIndex = nextNext;
|
||||
return true;
|
||||
}
|
||||
|
||||
void NVEventQueue::Init()
|
||||
{
|
||||
m_nextInsertIndex = 0;
|
||||
m_headIndex = 0;
|
||||
pthread_mutex_init(&(m_accessLock), NULL);
|
||||
syncInit(&m_consumerSync);
|
||||
syncInit(&m_blockerSync);
|
||||
|
||||
m_blocker = NULL;
|
||||
m_blockerState = NO_BLOCKER;
|
||||
m_blockerReturnVal = false;
|
||||
}
|
||||
|
||||
void NVEventQueue::Shutdown()
|
||||
{
|
||||
pthread_mutex_destroy(&(m_accessLock));
|
||||
|
||||
// free everyone...
|
||||
unlockAll(&m_consumerSync);
|
||||
unlockAll(&m_blockerSync);
|
||||
syncDestroy(&(m_consumerSync));
|
||||
syncDestroy(&(m_blockerSync));
|
||||
}
|
||||
|
||||
void NVEventQueue::Flush()
|
||||
{
|
||||
// TBD: Lock the mutex????
|
||||
m_headIndex = m_nextInsertIndex;
|
||||
}
|
||||
|
||||
void NVEventQueue::UnblockConsumer()
|
||||
{
|
||||
unlockAll(&(m_consumerSync));
|
||||
}
|
||||
|
||||
void NVEventQueue::UnblockProducer()
|
||||
{
|
||||
unlockAll(&(m_blockerSync));
|
||||
}
|
||||
|
||||
void NVEventQueue::Insert(const NVEvent* ev)
|
||||
{
|
||||
pthread_mutex_lock(&(m_accessLock));
|
||||
|
||||
// insert the event and unblock a waiter
|
||||
insert(ev);
|
||||
signal(&m_consumerSync);
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
}
|
||||
|
||||
bool NVEventQueue::InsertBlocking(const NVEvent* ev)
|
||||
{
|
||||
// TBD - how to handle the destruction of these mutexes
|
||||
|
||||
pthread_mutex_lock(&(m_accessLock));
|
||||
while (m_blocker)
|
||||
{
|
||||
if (wait(&(m_blockerSync), &(m_accessLock), NV_EVENT_WAIT_FOREVER))
|
||||
return false;
|
||||
}
|
||||
|
||||
assert(!m_blocker);
|
||||
assert(m_blockerState == NO_BLOCKER);
|
||||
|
||||
// we have the mutex _and_ the blocking event is NULL
|
||||
// So now we can push a new one
|
||||
m_blocker = ev;
|
||||
m_blockerState = PENDING_BLOCKER;
|
||||
|
||||
// Release the consumer, as we just posted a new event
|
||||
signal(&(m_consumerSync));
|
||||
|
||||
// Loop on the condition variable until we find out that
|
||||
// there is a return value waiting for us. Since only we
|
||||
// will null the blocker pointer, we will not let anyone
|
||||
// else start to post a blocking event
|
||||
while (m_blockerState != RETURNED_BLOCKER)
|
||||
{
|
||||
if (wait(&(m_blockerSync), &(m_accessLock), NV_EVENT_WAIT_FOREVER))
|
||||
return false;
|
||||
}
|
||||
|
||||
bool handled = m_blockerReturnVal;
|
||||
m_blocker = NULL;
|
||||
m_blockerState = NO_BLOCKER;
|
||||
|
||||
NVDEBUG("producer unblocking from consumer handling blocking event (%s)", handled ? "true" : "false");
|
||||
|
||||
// We've handled the event, so the producer can release the
|
||||
// next thread to potentially post a blocking event
|
||||
signal(&(m_blockerSync));
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
|
||||
return handled;
|
||||
}
|
||||
|
||||
|
||||
const NVEvent* NVEventQueue::RemoveOldest(int waitMSecs)
|
||||
{
|
||||
pthread_mutex_lock(&(m_accessLock));
|
||||
|
||||
// Hmm - the last event we got from RemoveOldest was a
|
||||
// blocker, and DoneWithEvent not called.
|
||||
// Default to "false" as a return value
|
||||
if (m_blockerState == PROCESSING_BLOCKER)
|
||||
{
|
||||
m_blockerReturnVal = false;
|
||||
m_blockerState = RETURNED_BLOCKER;
|
||||
broadcast(&(m_blockerSync));
|
||||
}
|
||||
|
||||
// Blocker is waiting - return it
|
||||
// And push the blocker pipeline forward
|
||||
if(m_blockerState == PENDING_BLOCKER)
|
||||
{
|
||||
m_blockerState = PROCESSING_BLOCKER;
|
||||
const NVEvent* ev = m_blocker;
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
|
||||
return ev;
|
||||
}
|
||||
else if (m_nextInsertIndex == m_headIndex)
|
||||
{
|
||||
// We're empty - so what do we do?
|
||||
if (waitMSecs == 0)
|
||||
{
|
||||
goto no_event;
|
||||
}
|
||||
else
|
||||
{
|
||||
// wait for the specified time
|
||||
wait(&(m_consumerSync), &(m_accessLock), (unsigned)waitMSecs);
|
||||
}
|
||||
|
||||
// check again after exiting cond waits, either we had a timeout
|
||||
if(m_blockerState == PENDING_BLOCKER)
|
||||
{
|
||||
m_blockerState = PROCESSING_BLOCKER;
|
||||
const NVEvent* ev = m_blocker;
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
|
||||
return ev;
|
||||
}
|
||||
else if (m_nextInsertIndex == m_headIndex)
|
||||
{
|
||||
goto no_event;
|
||||
}
|
||||
}
|
||||
|
||||
{
|
||||
// One way or another, we have an event...
|
||||
const NVEvent* ev = m_events + m_headIndex;
|
||||
m_headIndex = NVNextWrapped(m_headIndex);
|
||||
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
return ev;
|
||||
}
|
||||
|
||||
no_event:
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
return NULL;
|
||||
}
|
||||
|
||||
void NVEventQueue::DoneWithEvent(bool ret)
|
||||
{
|
||||
// We only care about blockers for now.
|
||||
// All other events just NOP
|
||||
pthread_mutex_lock(&(m_accessLock));
|
||||
if (m_blockerState == PROCESSING_BLOCKER)
|
||||
{
|
||||
m_blockerReturnVal = ret;
|
||||
m_blockerState = RETURNED_BLOCKER;
|
||||
broadcast(&(m_blockerSync));
|
||||
}
|
||||
pthread_mutex_unlock(&(m_accessLock));
|
||||
}
|
92
android/jni/nv_event/nv_event_queue.hpp
Normal file
92
android/jni/nv_event/nv_event_queue.hpp
Normal file
|
@ -0,0 +1,92 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_event\nv_event_queue.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#ifndef NV_EVENT_QUEUE
|
||||
#define NV_EVENT_QUEUE
|
||||
|
||||
#include "nv_event.hpp"
|
||||
#include <pthread.h>
|
||||
|
||||
class NVEventSync
|
||||
{
|
||||
public:
|
||||
pthread_cond_t m_cond;
|
||||
bool m_block;
|
||||
};
|
||||
|
||||
class NVEventQueue {
|
||||
public:
|
||||
void Init();
|
||||
void Shutdown();
|
||||
void Flush();
|
||||
void UnblockConsumer();
|
||||
void UnblockProducer();
|
||||
// Events are copied, so caller can reuse ev immediately
|
||||
void Insert(const NVEvent* ev);
|
||||
// Waits until the event is consumed. Returns whether the
|
||||
// consumer indicates handling the event or ignoring it
|
||||
bool InsertBlocking(const NVEvent* ev);
|
||||
|
||||
// Returned event is valid only until the next call to
|
||||
// RemoveOldest or until a call to DoneWithEvent
|
||||
// Calling RemoveOldest again without calling DoneWithEvent
|
||||
// indicates that the last event returned was NOT handled, and
|
||||
// thus InsertNewestAndWait for that even would return false
|
||||
const NVEvent* RemoveOldest(int waitMSecs);
|
||||
|
||||
// Indicates that all processing of the last event returned
|
||||
// from RemoveOldest is complete. Also allows the app to indicate
|
||||
// whether it handled the event or not.
|
||||
// Do not dereference the last event pointer after calling this function
|
||||
void DoneWithEvent(bool ret);
|
||||
|
||||
protected:
|
||||
bool insert(const NVEvent* ev);
|
||||
|
||||
enum { QUEUE_ELEMS = 256 };
|
||||
enum { QUEUE_MASK = 0x000000ff };
|
||||
|
||||
int32_t m_nextInsertIndex;
|
||||
int32_t m_headIndex;
|
||||
|
||||
pthread_mutex_t m_accessLock;
|
||||
|
||||
NVEventSync m_blockerSync;
|
||||
NVEventSync m_consumerSync;
|
||||
|
||||
NVEvent m_events[QUEUE_ELEMS];
|
||||
const NVEvent* m_blocker;
|
||||
enum BlockerState
|
||||
{
|
||||
NO_BLOCKER,
|
||||
PENDING_BLOCKER,
|
||||
PROCESSING_BLOCKER,
|
||||
RETURNED_BLOCKER
|
||||
};
|
||||
BlockerState m_blockerState;
|
||||
bool m_blockerReturnVal;
|
||||
};
|
||||
|
||||
|
||||
|
||||
|
||||
#endif // #ifndef NV_EVENT_QUEUE
|
147
android/jni/nv_event/nv_keycode_mapping.hpp
Normal file
147
android/jni/nv_event/nv_keycode_mapping.hpp
Normal file
|
@ -0,0 +1,147 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_event\nv_keycode_mapping.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
class NVKeyCodeMapping
|
||||
{
|
||||
public:
|
||||
NVKeyCodeMapping()
|
||||
{
|
||||
memset(m_keyMapping, 0, sizeof(NVKeyCode) * NV_MAX_KEYCODE);
|
||||
}
|
||||
|
||||
void Init(JNIEnv* env, jobject thiz);
|
||||
|
||||
bool MapKey(int key, NVKeyCode& code);
|
||||
|
||||
protected:
|
||||
void AddKeyMapping(JNIEnv* env, jobject thiz, jclass KeyCode_class, const char* name, NVKeyCode value);
|
||||
NVKeyCode m_keyMapping[NV_MAX_KEYCODE];
|
||||
};
|
||||
|
||||
/* Init the mapping array, set up the event queue */
|
||||
void NVKeyCodeMapping::AddKeyMapping(JNIEnv* env, jobject thiz, jclass KeyCode_class, const char* name, NVKeyCode value)
|
||||
{
|
||||
// Add a new mapping...
|
||||
jfieldID id = env->GetStaticFieldID(KeyCode_class, name, "I");
|
||||
int keyID = env->GetStaticIntField(KeyCode_class, id);
|
||||
|
||||
if (keyID < NV_MAX_KEYCODE)
|
||||
{
|
||||
/* TODO TBD Should check for collision */
|
||||
m_keyMapping[keyID] = value;
|
||||
}
|
||||
}
|
||||
|
||||
#define AddKeymappingMacro(name, value) \
|
||||
AddKeyMapping(env, thiz, KeyCode_class, name, value)
|
||||
|
||||
void NVKeyCodeMapping::Init(JNIEnv* env, jobject thiz)
|
||||
{
|
||||
jclass KeyCode_class = env->FindClass("android/view/KeyEvent");
|
||||
|
||||
AddKeymappingMacro("KEYCODE_BACK",NV_KEYCODE_BACK);
|
||||
AddKeymappingMacro("KEYCODE_TAB",NV_KEYCODE_TAB);
|
||||
AddKeymappingMacro("KEYCODE_ENTER",NV_KEYCODE_ENTER);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_SPACE",NV_KEYCODE_SPACE);
|
||||
AddKeymappingMacro("KEYCODE_ENDCALL",NV_KEYCODE_ENDCALL);
|
||||
AddKeymappingMacro("KEYCODE_HOME",NV_KEYCODE_HOME);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_DPAD_LEFT",NV_KEYCODE_DPAD_LEFT);
|
||||
AddKeymappingMacro("KEYCODE_DPAD_UP",NV_KEYCODE_DPAD_UP);
|
||||
AddKeymappingMacro("KEYCODE_DPAD_RIGHT",NV_KEYCODE_DPAD_RIGHT);
|
||||
AddKeymappingMacro("KEYCODE_DPAD_DOWN",NV_KEYCODE_DPAD_DOWN);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_DEL",NV_KEYCODE_DEL);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_0",NV_KEYCODE_0);
|
||||
AddKeymappingMacro("KEYCODE_1",NV_KEYCODE_1);
|
||||
AddKeymappingMacro("KEYCODE_2",NV_KEYCODE_2);
|
||||
AddKeymappingMacro("KEYCODE_3",NV_KEYCODE_3);
|
||||
AddKeymappingMacro("KEYCODE_4",NV_KEYCODE_4);
|
||||
AddKeymappingMacro("KEYCODE_5",NV_KEYCODE_5);
|
||||
AddKeymappingMacro("KEYCODE_6",NV_KEYCODE_6);
|
||||
AddKeymappingMacro("KEYCODE_7",NV_KEYCODE_7);
|
||||
AddKeymappingMacro("KEYCODE_8",NV_KEYCODE_8);
|
||||
AddKeymappingMacro("KEYCODE_9",NV_KEYCODE_9);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_A",NV_KEYCODE_A);
|
||||
AddKeymappingMacro("KEYCODE_B",NV_KEYCODE_B);
|
||||
AddKeymappingMacro("KEYCODE_C",NV_KEYCODE_C);
|
||||
AddKeymappingMacro("KEYCODE_D",NV_KEYCODE_D);
|
||||
AddKeymappingMacro("KEYCODE_E",NV_KEYCODE_E);
|
||||
AddKeymappingMacro("KEYCODE_F",NV_KEYCODE_F);
|
||||
AddKeymappingMacro("KEYCODE_G",NV_KEYCODE_G);
|
||||
AddKeymappingMacro("KEYCODE_H",NV_KEYCODE_H);
|
||||
AddKeymappingMacro("KEYCODE_I",NV_KEYCODE_I);
|
||||
AddKeymappingMacro("KEYCODE_J",NV_KEYCODE_J);
|
||||
AddKeymappingMacro("KEYCODE_K",NV_KEYCODE_K);
|
||||
AddKeymappingMacro("KEYCODE_L",NV_KEYCODE_L);
|
||||
AddKeymappingMacro("KEYCODE_M",NV_KEYCODE_M);
|
||||
AddKeymappingMacro("KEYCODE_N",NV_KEYCODE_N);
|
||||
AddKeymappingMacro("KEYCODE_O",NV_KEYCODE_O);
|
||||
AddKeymappingMacro("KEYCODE_P",NV_KEYCODE_P);
|
||||
AddKeymappingMacro("KEYCODE_Q",NV_KEYCODE_Q);
|
||||
AddKeymappingMacro("KEYCODE_R",NV_KEYCODE_R);
|
||||
AddKeymappingMacro("KEYCODE_S",NV_KEYCODE_S);
|
||||
AddKeymappingMacro("KEYCODE_T",NV_KEYCODE_T);
|
||||
AddKeymappingMacro("KEYCODE_U",NV_KEYCODE_U);
|
||||
AddKeymappingMacro("KEYCODE_V",NV_KEYCODE_V);
|
||||
AddKeymappingMacro("KEYCODE_W",NV_KEYCODE_W);
|
||||
AddKeymappingMacro("KEYCODE_X",NV_KEYCODE_X);
|
||||
AddKeymappingMacro("KEYCODE_Y",NV_KEYCODE_Y);
|
||||
AddKeymappingMacro("KEYCODE_Z",NV_KEYCODE_Z);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_STAR",NV_KEYCODE_STAR);
|
||||
AddKeymappingMacro("KEYCODE_PLUS",NV_KEYCODE_PLUS);
|
||||
AddKeymappingMacro("KEYCODE_MINUS",NV_KEYCODE_MINUS);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_NUM",NV_KEYCODE_NUM);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_ALT_LEFT",NV_KEYCODE_ALT_LEFT);
|
||||
AddKeymappingMacro("KEYCODE_ALT_RIGHT",NV_KEYCODE_ALT_RIGHT);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_SHIFT_LEFT",NV_KEYCODE_SHIFT_LEFT);
|
||||
AddKeymappingMacro("KEYCODE_SHIFT_RIGHT",NV_KEYCODE_SHIFT_RIGHT);
|
||||
|
||||
AddKeymappingMacro("KEYCODE_APOSTROPHE",NV_KEYCODE_APOSTROPHE);
|
||||
AddKeymappingMacro("KEYCODE_SEMICOLON",NV_KEYCODE_SEMICOLON);
|
||||
AddKeymappingMacro("KEYCODE_EQUALS",NV_KEYCODE_EQUALS);
|
||||
AddKeymappingMacro("KEYCODE_COMMA",NV_KEYCODE_COMMA);
|
||||
AddKeymappingMacro("KEYCODE_PERIOD",NV_KEYCODE_PERIOD);
|
||||
AddKeymappingMacro("KEYCODE_SLASH",NV_KEYCODE_SLASH);
|
||||
AddKeymappingMacro("KEYCODE_GRAVE",NV_KEYCODE_GRAVE);
|
||||
AddKeymappingMacro("KEYCODE_LEFT_BRACKET",NV_KEYCODE_LEFT_BRACKET);
|
||||
AddKeymappingMacro("KEYCODE_BACKSLASH",NV_KEYCODE_BACKSLASH);
|
||||
AddKeymappingMacro("KEYCODE_RIGHT_BRACKET",NV_KEYCODE_RIGHT_BRACKET);
|
||||
}
|
||||
|
||||
bool NVKeyCodeMapping::MapKey(int key, NVKeyCode& code)
|
||||
{
|
||||
if (key < NV_MAX_KEYCODE)
|
||||
{
|
||||
code = m_keyMapping[key];
|
||||
return true;
|
||||
}
|
||||
else
|
||||
return false;
|
||||
}
|
82
android/jni/nv_event/scoped_profiler.hpp
Normal file
82
android/jni/nv_event/scoped_profiler.hpp
Normal file
|
@ -0,0 +1,82 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_event\scoped_profiler.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#ifndef SCOPED_PROFILER_H
|
||||
#define SCOPED_PROFILER_H
|
||||
|
||||
#define PERF_STMTS 0
|
||||
#if PERF_STMTS == 1
|
||||
#include "../nv_time/nv_time.hpp"
|
||||
#include "stdio.h"
|
||||
#include "stdlib.h"
|
||||
|
||||
static char s_bigString[4096];
|
||||
static int s_bigStringSize;
|
||||
|
||||
static char s_tmpBuf[1024];
|
||||
|
||||
class ScopedProfiler
|
||||
{
|
||||
public:
|
||||
ScopedProfiler(const char* text)
|
||||
{
|
||||
_text = text;
|
||||
_startTime = nvGetSystemTime();
|
||||
__last = this;
|
||||
}
|
||||
~ScopedProfiler()
|
||||
{
|
||||
stop();
|
||||
}
|
||||
inline void stop()
|
||||
{
|
||||
if(_text)
|
||||
{
|
||||
int size = snprintf(s_tmpBuf, dimof(s_tmpBuf)-1, "%d ms spent in %s" , (int)(nvGetSystemTime() - _startTime), _text);
|
||||
strcat(s_bigString + s_bigStringSize, s_tmpBuf);
|
||||
s_bigStringSize += size;
|
||||
_text = 0;
|
||||
}
|
||||
static void stopLast()
|
||||
{
|
||||
if(__last)
|
||||
__last->stop();
|
||||
__last = 0;
|
||||
}
|
||||
const char* _text;
|
||||
long _startTime;
|
||||
static ScopedProfiler* __last;
|
||||
};
|
||||
ScopedProfiler* ScopedProfiler::__last = 0;
|
||||
|
||||
#define STRINGIFIER(s) #s
|
||||
#define CONCAT_(a,b) a ## b
|
||||
#define CONCAT(a,b) CONCAT_(a,b)
|
||||
#define PERFBLURB(s) static const char CONCAT(___str,__LINE__)[] = s "\n"; ScopedProfiler CONCAT(SCOPED_PROFILER,__LINE__)(CONCAT(___str,__LINE__));
|
||||
#define RESET_PROFILING() { DEBUG_D("%s", s_bigString); s_bigString[0] = 0; s_bigStringSize = 0; }
|
||||
#else
|
||||
#define PERFBLURB(s)
|
||||
#define RESET_PROFILING()
|
||||
#endif
|
||||
|
||||
#endif // SCOPED_PROFILER_H
|
||||
|
159
android/jni/nv_thread/nv_thread.cpp
Normal file
159
android/jni/nv_thread/nv_thread.cpp
Normal file
|
@ -0,0 +1,159 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_thread\nv_thread.c
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#include <jni.h>
|
||||
|
||||
#include "nv_thread.hpp"
|
||||
#include <android/log.h>
|
||||
#include <pthread.h>
|
||||
|
||||
static JavaVM* s_vm = NULL;
|
||||
static pthread_key_t s_jniEnvKey = 0;
|
||||
|
||||
#define MODULE "NVThread"
|
||||
|
||||
#include "../nv_debug/nv_debug.hpp"
|
||||
|
||||
void NVThreadInit(JavaVM* vm)
|
||||
{
|
||||
s_vm = vm;
|
||||
}
|
||||
|
||||
JNIEnv* NVThreadGetCurrentJNIEnv()
|
||||
{
|
||||
JNIEnv* env = NULL;
|
||||
if (s_jniEnvKey)
|
||||
{
|
||||
env = (JNIEnv*)pthread_getspecific(s_jniEnvKey);
|
||||
}
|
||||
else
|
||||
{
|
||||
pthread_key_create(&s_jniEnvKey, NULL);
|
||||
}
|
||||
|
||||
if (!env)
|
||||
{
|
||||
// do we have a VM cached?
|
||||
if (!s_vm)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error - could not find JVM!");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
// Hmm - no env for this thread cached yet
|
||||
int error = s_vm->AttachCurrentThread(&env, NULL);
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "AttachCurrentThread: %d, 0x%p", error, env);
|
||||
if (error || !env)
|
||||
{
|
||||
__android_log_print(ANDROID_LOG_DEBUG, MODULE, "Error - could not attach thread to JVM!");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
pthread_setspecific(s_jniEnvKey, env);
|
||||
}
|
||||
|
||||
return env;
|
||||
}
|
||||
|
||||
typedef struct NVThreadInitStruct
|
||||
{
|
||||
void* m_arg;
|
||||
void *(*m_startRoutine)(void *);
|
||||
} NVThreadInitStruct;
|
||||
|
||||
static void* NVThreadSpawnProc(void* arg)
|
||||
{
|
||||
NVThreadInitStruct* init = (NVThreadInitStruct*)arg;
|
||||
void *(*start_routine)(void *) = init->m_startRoutine;
|
||||
void* data = init->m_arg;
|
||||
void* ret;
|
||||
|
||||
free(arg);
|
||||
|
||||
NVThreadGetCurrentJNIEnv();
|
||||
|
||||
ret = start_routine(data);
|
||||
|
||||
if (s_vm)
|
||||
s_vm->DetachCurrentThread();
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int NVThreadSpawnJNIThread(pthread_t *thread, pthread_attr_t const * attr,
|
||||
void *(*start_routine)(void *), void * arg)
|
||||
{
|
||||
if (!start_routine)
|
||||
return -1;
|
||||
|
||||
NVThreadInitStruct * initData = new NVThreadInitStruct;
|
||||
|
||||
initData->m_startRoutine = start_routine;
|
||||
initData->m_arg = arg;
|
||||
|
||||
int err = pthread_create(thread, attr, NVThreadSpawnProc, initData);
|
||||
|
||||
// If the thread was not started, then we need to delete the init data ourselves
|
||||
if (err)
|
||||
free(initData);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
// on linuces, signals can interrupt sleep functions, so you might need to
|
||||
// retry to get the full sleep going. I'm not entirely sure this is necessary
|
||||
// *here* clients could retry themselves when the exposed function returns
|
||||
// nonzero
|
||||
inline int __sleep(const struct timespec *req, struct timespec *rem)
|
||||
{
|
||||
int ret = 1;
|
||||
int i;
|
||||
static const int sleepTries = 2;
|
||||
|
||||
struct timespec req_tmp={0}, rem_tmp={0};
|
||||
|
||||
rem_tmp = *req;
|
||||
for(i = 0; i < sleepTries; ++i)
|
||||
{
|
||||
req_tmp = rem_tmp;
|
||||
int ret = nanosleep(&req_tmp, &rem_tmp);
|
||||
if(ret == 0)
|
||||
{
|
||||
ret = 0;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if(rem)
|
||||
*rem = rem_tmp;
|
||||
return ret;
|
||||
}
|
||||
|
||||
int NVThreadSleep(unsigned long millisec)
|
||||
{
|
||||
struct timespec req={0},rem={0};
|
||||
time_t sec =(int)(millisec/1000);
|
||||
|
||||
millisec = millisec-(sec*1000);
|
||||
req.tv_sec = sec;
|
||||
req.tv_nsec = millisec*1000000L;
|
||||
return __sleep(&req,&rem);
|
||||
}
|
97
android/jni/nv_thread/nv_thread.hpp
Normal file
97
android/jni/nv_thread/nv_thread.hpp
Normal file
|
@ -0,0 +1,97 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_thread\nv_thread.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#ifndef __INCLUDED_NV_THREAD_H
|
||||
#define __INCLUDED_NV_THREAD_H
|
||||
|
||||
#include <jni.h>
|
||||
#include <pthread.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
extern "C"
|
||||
{
|
||||
#endif
|
||||
|
||||
/** @file nv_thread.h
|
||||
The NVThread library makes it easy to create native threads that can acess
|
||||
JNI objects. By default, pthreads created in the Android NDK are NOT connected
|
||||
to the JVM and JNI calls will fail. This library wraps thread creation in
|
||||
such a way that pthreads created using it will connect to and disconnect from
|
||||
the JVM as appropriate. Applications creating all of their threads with these
|
||||
interfaces can use the provided NVThreadGetCurrentJNIEnv() function to
|
||||
get the current thread's JNI context at any time.
|
||||
|
||||
Note that native-created threads still have JNI limitations over threads
|
||||
that are calls down to native from Java. The JNI function FindClass will
|
||||
NOT find application-specific classes when called from native threads.
|
||||
Native code that needs to call FindClass and record the indices of Java
|
||||
class members for later access must call FindClass and Get*FieldID/Get*MethodID
|
||||
in threads calling from Java, such as JNI_OnLoad
|
||||
*/
|
||||
|
||||
/**
|
||||
Initializes the nv_thread system by connecting it to the JVM. This
|
||||
function must be called as early as possible in the native code's
|
||||
JNI_OnLoad function, so that the thread system is prepared for any
|
||||
JNI-dependent library initialization calls.
|
||||
@param vm The VM pointer - should be the JavaVM pointer sent to JNI_OnLoad.
|
||||
*/
|
||||
void NVThreadInit(JavaVM* vm);
|
||||
|
||||
/**
|
||||
Retrieves the JNIEnv object associated with the current thread, allowing
|
||||
any thread that was creating with NVThreadSpawnJNIThread() to access the
|
||||
JNI at will. This JNIEnv is NOT usable across multiple calls or threads
|
||||
The function should be called in each function that requires a JNIEnv
|
||||
@return The current thread's JNIEnv, or NULL if the thread was not created
|
||||
by NVThreadSpawnJNIThread
|
||||
@see NVThreadSpawnJNIThread
|
||||
*/
|
||||
JNIEnv* NVThreadGetCurrentJNIEnv();
|
||||
|
||||
/**
|
||||
Spwans a new native thread that is registered for use with JNI. Threads
|
||||
created with this function will have access to JNI data via the JNIEnv
|
||||
available from NVThreadGetCurrentJNIEnv().
|
||||
@param thread is the same as in pthread_create
|
||||
@param attr is the same as in pthread_create
|
||||
@param start_routine is the same as in pthread_create
|
||||
@param arg is the same as in pthread_create
|
||||
@return 0 on success, -1 on failure
|
||||
@see NVThreadGetCurrentJNIEnv
|
||||
*/
|
||||
int NVThreadSpawnJNIThread(pthread_t *thread, pthread_attr_t const * attr,
|
||||
void *(*start_routine)(void *), void * arg);
|
||||
|
||||
/**
|
||||
Sleeps the current thread for the specified number of milliseconds
|
||||
@param millisec Sleep time in ms
|
||||
@return 0 on success, -1 on failure
|
||||
*/
|
||||
int NVThreadSleep(unsigned long millisec);
|
||||
|
||||
#if defined(__cplusplus)
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif
|
164
android/jni/nv_time/nv_time.cpp
Normal file
164
android/jni/nv_time/nv_time.cpp
Normal file
|
@ -0,0 +1,164 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_time\nv_time.cpp
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#include "nv_time.hpp"
|
||||
#include "../nv_thread/nv_thread.hpp"
|
||||
/*#include <EGL/egl.h>
|
||||
#include <EGL/eglext.h>
|
||||
#include <EGL/eglplatform.h>*/
|
||||
#include <time.h>
|
||||
#include <jni.h>
|
||||
#include <android/log.h>
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
|
||||
/*#ifndef EGL_NV_system_time
|
||||
#define EGL_NV_system_time 1
|
||||
typedef khronos_uint64_t EGLuint64NV;
|
||||
#ifdef EGL_EGLEXT_PROTOTYPES
|
||||
EGLAPI EGLuint64NV EGLAPIENTRY eglGetSystemTimeFrequencyNV(void);
|
||||
EGLAPI EGLuint64NV EGLAPIENTRY eglGetSystemTimeNV(void);
|
||||
#endif
|
||||
typedef EGLuint64NV (EGLAPIENTRYP PFNEGLGETSYSTEMTIMEFREQUENCYNVPROC)(void);
|
||||
typedef EGLuint64NV (EGLAPIENTRYP PFNEGLGETSYSTEMTIMENVPROC)(void);
|
||||
#endif
|
||||
*/
|
||||
|
||||
void nvAcquireTimeExtensionJNI(JNIEnv*, jobject)
|
||||
{
|
||||
nvAcquireTimeExtension();
|
||||
}
|
||||
|
||||
jlong nvGetSystemTimeJNI(JNIEnv*, jobject)
|
||||
{
|
||||
return (jlong)nvGetSystemTime();
|
||||
}
|
||||
|
||||
/*static PFNEGLGETSYSTEMTIMEFREQUENCYNVPROC eglGetSystemTimeFrequencyNVProc = NULL;
|
||||
static PFNEGLGETSYSTEMTIMENVPROC eglGetSystemTimeNVProc = NULL;
|
||||
static EGLuint64NV eglSystemTimeFrequency = 0;
|
||||
static bool timeExtensionQueried = false;*/
|
||||
|
||||
void nvAcquireTimeExtension()
|
||||
{
|
||||
/* if (timeExtensionQueried)
|
||||
return;
|
||||
timeExtensionQueried = true;
|
||||
|
||||
eglGetSystemTimeFrequencyNVProc = (PFNEGLGETSYSTEMTIMEFREQUENCYNVPROC) eglGetProcAddress("eglGetSystemTimeFrequencyNV");
|
||||
eglGetSystemTimeNVProc = (PFNEGLGETSYSTEMTIMENVPROC) eglGetProcAddress("eglGetSystemTimeNV");
|
||||
|
||||
// now, we'll proceed through a series of sanity checking.
|
||||
// if they all succeed, we'll return.
|
||||
// if any fail, we fall out of conditional tests to end of function, null pointers, and return.
|
||||
if (eglGetSystemTimeFrequencyNVProc &&
|
||||
eglGetSystemTimeNVProc)
|
||||
{
|
||||
eglSystemTimeFrequency = eglGetSystemTimeFrequencyNVProc();
|
||||
if (eglSystemTimeFrequency>0) // assume okay. quick-check it works.
|
||||
{
|
||||
EGLuint64NV time1, time2;
|
||||
time1 = eglGetSystemTimeNVProc();
|
||||
usleep(2000); // 2ms should be MORE than sufficient, right?
|
||||
time2 = eglGetSystemTimeNVProc();
|
||||
if (time1 != time2) // quick sanity only...
|
||||
{
|
||||
// we've sanity checked:
|
||||
// - fn pointers non-null
|
||||
// - freq non-zero
|
||||
// - two calls to time sep'd by sleep non-equal
|
||||
// safe to return now.
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// fall back if we've not returned already.
|
||||
eglGetSystemTimeFrequencyNVProc = (PFNEGLGETSYSTEMTIMEFREQUENCYNVPROC) NULL;
|
||||
eglGetSystemTimeNVProc = (PFNEGLGETSYSTEMTIMENVPROC) NULL;*/
|
||||
}
|
||||
|
||||
bool nvValidTimeExtension()
|
||||
{
|
||||
/* if (NULL == eglGetSystemTimeFrequencyNVProc ||
|
||||
NULL == eglGetSystemTimeNVProc)
|
||||
return false;
|
||||
else
|
||||
return true;*/
|
||||
}
|
||||
|
||||
long nvGetSystemTime()
|
||||
{
|
||||
static struct timeval start_time, end_time;
|
||||
static int isinit = 0;
|
||||
jlong curr_time = 0;
|
||||
|
||||
/* if(eglGetSystemTimeNVProc)
|
||||
{
|
||||
EGLuint64NV egltime;
|
||||
EGLuint64NV egltimequot;
|
||||
EGLuint64NV egltimerem;
|
||||
|
||||
egltime = eglGetSystemTimeNVProc();
|
||||
|
||||
egltimequot = egltime / eglSystemTimeFrequency;
|
||||
egltimerem = egltime - (eglSystemTimeFrequency * egltimequot);
|
||||
egltimequot *= 1000;
|
||||
egltimerem *= 1000;
|
||||
egltimerem /= eglSystemTimeFrequency;
|
||||
egltimequot += egltimerem;
|
||||
return (jlong) egltimequot;
|
||||
}
|
||||
*/
|
||||
if (!isinit)
|
||||
{
|
||||
gettimeofday(&start_time, 0);
|
||||
isinit = 1;
|
||||
}
|
||||
gettimeofday(&end_time, 0);
|
||||
curr_time = (end_time.tv_sec - start_time.tv_sec) * 1000;
|
||||
curr_time += (end_time.tv_usec - start_time.tv_usec) / 1000;
|
||||
|
||||
return curr_time;
|
||||
}
|
||||
|
||||
void NVTimeInit()
|
||||
{
|
||||
JNIEnv* env = NVThreadGetCurrentJNIEnv();
|
||||
|
||||
JNINativeMethod methods_time[] =
|
||||
{
|
||||
{
|
||||
"nvAcquireTimeExtension",
|
||||
"()V",
|
||||
(void *) nvAcquireTimeExtension
|
||||
},
|
||||
{
|
||||
"nvGetSystemTime",
|
||||
"()J",
|
||||
(void *) nvGetSystemTime
|
||||
},
|
||||
};
|
||||
jclass k_time;
|
||||
k_time = (env)->FindClass ("com/nvidia/devtech/NvActivity");
|
||||
(env)->RegisterNatives(k_time, methods_time, 2);
|
||||
}
|
34
android/jni/nv_time/nv_time.hpp
Normal file
34
android/jni/nv_time/nv_time.hpp
Normal file
|
@ -0,0 +1,34 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\jni\nv_time\nv_time.h
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#ifndef __INCLUDED_NV_TIME_H
|
||||
#define __INCLUDED_NV_TIME_H
|
||||
|
||||
#include <stdlib.h>
|
||||
|
||||
void NVTimeInit();
|
||||
|
||||
void nvAcquireTimeExtension();
|
||||
bool nvValidTimeExtension();
|
||||
long nvGetSystemTime();
|
||||
|
||||
#endif
|
|
@ -1,83 +0,0 @@
|
|||
package com.mapswithme.maps;
|
||||
|
||||
import android.graphics.PointF;
|
||||
import android.view.MotionEvent;
|
||||
|
||||
public class GesturesProcessor
|
||||
{
|
||||
private final int NONE = 0;
|
||||
private final int MOVE = 1;
|
||||
private final int ZOOM = 2;
|
||||
|
||||
// Do not modify this constant values (or take into account native code,
|
||||
// please).
|
||||
private final int START = 0;
|
||||
private final int PROCESS = 1;
|
||||
private final int END = 2;
|
||||
|
||||
private PointF m_pt1;
|
||||
private PointF m_pt2;
|
||||
private int m_mode;
|
||||
|
||||
private void getPointsMove(MotionEvent e)
|
||||
{
|
||||
m_pt1.set(e.getX(), e.getY());
|
||||
}
|
||||
|
||||
private void getPointsZoom(MotionEvent e)
|
||||
{
|
||||
m_pt1.set(e.getX(0), e.getY(0));
|
||||
m_pt2.set(e.getX(1), e.getY(1));
|
||||
}
|
||||
|
||||
public GesturesProcessor()
|
||||
{
|
||||
m_pt1 = new PointF();
|
||||
m_pt2 = new PointF();
|
||||
m_mode = NONE;
|
||||
}
|
||||
|
||||
public void onTouchEvent(MotionEvent e)
|
||||
{
|
||||
switch (e.getAction() & MotionEvent.ACTION_MASK)
|
||||
{
|
||||
case MotionEvent.ACTION_DOWN:
|
||||
m_mode = MOVE;
|
||||
getPointsMove(e);
|
||||
nativeMove(START, m_pt1.x, m_pt1.y);
|
||||
break;
|
||||
|
||||
case MotionEvent.ACTION_POINTER_DOWN:
|
||||
m_mode = ZOOM;
|
||||
getPointsZoom(e);
|
||||
nativeZoom(START, m_pt1.x, m_pt1.y, m_pt2.x, m_pt2.y);
|
||||
break;
|
||||
|
||||
case MotionEvent.ACTION_UP:
|
||||
nativeMove(END, m_pt1.x, m_pt1.y);
|
||||
m_mode = NONE;
|
||||
break;
|
||||
|
||||
case MotionEvent.ACTION_POINTER_UP:
|
||||
nativeZoom(END, m_pt1.x, m_pt1.y, m_pt2.x, m_pt2.y);
|
||||
m_mode = NONE;
|
||||
break;
|
||||
|
||||
case MotionEvent.ACTION_MOVE:
|
||||
if (m_mode == MOVE)
|
||||
{
|
||||
getPointsMove(e);
|
||||
nativeMove(PROCESS, m_pt1.x, m_pt1.y);
|
||||
}
|
||||
else if (m_mode == ZOOM)
|
||||
{
|
||||
getPointsZoom(e);
|
||||
nativeZoom(PROCESS, m_pt1.x, m_pt1.y, m_pt2.x, m_pt2.y);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private native void nativeMove(int mode, double x, double y);
|
||||
private native void nativeZoom(int mode, double x1, double y1, double x2, double y2);
|
||||
}
|
|
@ -2,11 +2,10 @@ package com.mapswithme.maps;
|
|||
|
||||
import java.io.File;
|
||||
|
||||
import com.mapswithme.maps.MainGLView;
|
||||
import com.mapswithme.maps.R;
|
||||
import com.mapswithme.maps.location.LocationService;
|
||||
import com.nvidia.devtech.NvEventQueueActivity;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.ApplicationInfo;
|
||||
import android.content.pm.PackageManager;
|
||||
|
@ -16,15 +15,13 @@ import android.os.Environment;
|
|||
import android.view.Menu;
|
||||
import android.view.MenuInflater;
|
||||
import android.view.MenuItem;
|
||||
import android.util.Log;
|
||||
|
||||
public class MWMActivity extends Activity
|
||||
public class MWMActivity extends NvEventQueueActivity
|
||||
{
|
||||
VideoTimer m_timer;
|
||||
private static String TAG = "MWMActivity";
|
||||
private final static String PACKAGE_NAME = "com.mapswithme.maps";
|
||||
|
||||
private MainGLView m_view;
|
||||
|
||||
private boolean m_locationEnabled = false;
|
||||
|
||||
private String getAppBundlePath() throws NameNotFoundException
|
||||
|
@ -58,13 +55,16 @@ public class MWMActivity extends Activity
|
|||
{
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
m_view = new MainGLView(getApplication());
|
||||
|
||||
setContentView(m_view);
|
||||
m_timer = new VideoTimer();
|
||||
|
||||
|
||||
/*m_view = new MainGLView(getApplication());
|
||||
|
||||
setContentView(m_view);*/
|
||||
}
|
||||
|
||||
@Override
|
||||
/* @Override
|
||||
protected void onPause()
|
||||
{
|
||||
super.onPause();
|
||||
|
@ -80,7 +80,7 @@ public class MWMActivity extends Activity
|
|||
m_view.onResume();
|
||||
if (m_locationEnabled)
|
||||
LocationService.start(this);
|
||||
}
|
||||
}*/
|
||||
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu)
|
||||
|
|
|
@ -1,82 +0,0 @@
|
|||
package com.mapswithme.maps;
|
||||
|
||||
import com.mapswithme.maps.GesturesProcessor;
|
||||
import com.mapswithme.maps.VideoTimer;
|
||||
|
||||
import javax.microedition.khronos.egl.EGLConfig;
|
||||
import javax.microedition.khronos.opengles.GL10;
|
||||
|
||||
import android.content.Context;
|
||||
import android.opengl.GLSurfaceView;
|
||||
import android.util.Log;
|
||||
import android.view.MotionEvent;
|
||||
|
||||
public class MainGLView extends GLSurfaceView
|
||||
{
|
||||
private static String TAG = "MainGLView";
|
||||
|
||||
GesturesProcessor m_gestures;
|
||||
VideoTimer m_timer;
|
||||
|
||||
public MainGLView(Context context)
|
||||
{
|
||||
super(context);
|
||||
init();
|
||||
}
|
||||
|
||||
private void init()
|
||||
{
|
||||
m_gestures = new GesturesProcessor();
|
||||
|
||||
// Do native initialization with OpenGL.
|
||||
nativeInit();
|
||||
|
||||
m_timer = new VideoTimer();
|
||||
|
||||
setRenderer(new MainRenderer());
|
||||
|
||||
// When renderMode is RENDERMODE_WHEN_DIRTY, the renderer only rendered
|
||||
// when the surface is created, or when requestRender() is called.
|
||||
setRenderMode(RENDERMODE_WHEN_DIRTY);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onTouchEvent(MotionEvent e)
|
||||
{
|
||||
m_gestures.onTouchEvent(e);
|
||||
return true; // if handled
|
||||
}
|
||||
|
||||
private native void nativeInit();
|
||||
}
|
||||
|
||||
|
||||
class MainRenderer implements GLSurfaceView.Renderer
|
||||
{
|
||||
private static String TAG = "MainGLView.MainRenderer";
|
||||
|
||||
//@Override
|
||||
public void onDrawFrame(GL10 gl)
|
||||
{
|
||||
//Log.i(TAG, "onDrawFrame");
|
||||
nativeDraw();
|
||||
}
|
||||
|
||||
//@Override
|
||||
public void onSurfaceChanged(GL10 gl, int w, int h)
|
||||
{
|
||||
Log.i(TAG, "onSurfaceChanged");
|
||||
nativeResize(w, h);
|
||||
}
|
||||
|
||||
//@Override
|
||||
public void onSurfaceCreated(GL10 gl, EGLConfig config)
|
||||
{
|
||||
Log.i(TAG, "onSurfaceCreated");
|
||||
nativeInit();
|
||||
}
|
||||
|
||||
private native void nativeInit();
|
||||
private native void nativeResize(int w, int h);
|
||||
private native void nativeDraw();
|
||||
}
|
737
android/src/com/nvidia/devtech/NvEventQueueActivity.java
Normal file
737
android/src/com/nvidia/devtech/NvEventQueueActivity.java
Normal file
|
@ -0,0 +1,737 @@
|
|||
//----------------------------------------------------------------------------------
|
||||
// File: libs\src\com\nvidia\devtech\NvEventQueueActivity.java
|
||||
// Samples Version: NVIDIA Android Lifecycle samples 1_0beta
|
||||
// Email: tegradev@nvidia.com
|
||||
// Web: http://developer.nvidia.com/category/zone/mobile-development
|
||||
//
|
||||
// Copyright 2009-2011 NVIDIA® Corporation
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//----------------------------------------------------------------------------------
|
||||
package com.nvidia.devtech;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.hardware.Sensor;
|
||||
import android.hardware.SensorEvent;
|
||||
import android.hardware.SensorEventListener;
|
||||
import android.hardware.SensorManager;
|
||||
import android.os.Bundle;
|
||||
import android.view.KeyEvent;
|
||||
import android.view.MotionEvent;
|
||||
|
||||
import javax.microedition.khronos.egl.EGL10;
|
||||
import javax.microedition.khronos.egl.EGLConfig;
|
||||
import javax.microedition.khronos.egl.EGLContext;
|
||||
import javax.microedition.khronos.egl.EGLDisplay;
|
||||
import javax.microedition.khronos.egl.EGLSurface;
|
||||
import javax.microedition.khronos.opengles.GL11;
|
||||
|
||||
import android.view.SurfaceHolder;
|
||||
import android.view.SurfaceView;
|
||||
import android.view.SurfaceHolder.Callback;
|
||||
|
||||
/**
|
||||
A base class used to provide a native-code event-loop interface to an
|
||||
application. This class is designed to be subclassed by the application
|
||||
with very little need to extend the Java. Paired with its native static-link
|
||||
library, libnv_event.a, this package makes it possible for native applciations
|
||||
to avoid any direct use of Java code. In addition, input and other events are
|
||||
automatically queued and provided to the application in native code via a
|
||||
classic event queue-like API. EGL functionality such as bind/unbind and swap
|
||||
are also made available to the native code for ease of application porting.
|
||||
Please see the external SDK documentation for an introduction to the use of
|
||||
this class and its paired native library.
|
||||
*/
|
||||
public abstract class NvEventQueueActivity
|
||||
extends Activity
|
||||
implements SensorEventListener
|
||||
{
|
||||
protected boolean wantsMultitouch = true;
|
||||
|
||||
protected boolean wantsAccelerometer = false;
|
||||
protected SensorManager mSensorManager = null;
|
||||
protected int mSensorDelay = SensorManager.SENSOR_DELAY_UI; //other options: SensorManager.SENSOR_DELAY_FASTEST, SensorManager.SENSOR_DELAY_NORMAL and SensorManager.SENSOR_DELAY_UI
|
||||
|
||||
protected SurfaceView view3d = null;
|
||||
|
||||
//private static final int EGL_RENDERABLE_TYPE = 0x3040;
|
||||
//private static final int EGL_OPENGL_ES_BIT = 0x0001;
|
||||
//private static final int EGL_OPENGL_ES2_BIT = 0x0004;
|
||||
private static final int EGL_CONTEXT_CLIENT_VERSION = 0x3098;
|
||||
EGL10 egl = null;
|
||||
GL11 gl = null;
|
||||
|
||||
protected boolean eglInitialized = false;
|
||||
protected EGLSurface eglSurface = null;
|
||||
protected EGLDisplay eglDisplay = null;
|
||||
protected EGLContext eglContext = null;
|
||||
protected EGLConfig eglConfig = null;
|
||||
|
||||
protected SurfaceHolder cachedSurfaceHolder = null;
|
||||
private int surfaceWidth = 0;
|
||||
private int surfaceHeight = 0;
|
||||
|
||||
private int fixedWidth = 0;
|
||||
private int fixedHeight = 0;
|
||||
|
||||
private boolean nativeLaunched = false;
|
||||
|
||||
/* *
|
||||
* Helper function to select fixed window size.
|
||||
* */
|
||||
public void setFixedSize(int fw, int fh)
|
||||
{
|
||||
fixedWidth = fw;
|
||||
fixedHeight = fh;
|
||||
}
|
||||
|
||||
public int getSurfaceWidth()
|
||||
{
|
||||
return surfaceWidth;
|
||||
}
|
||||
|
||||
public int getSurfaceHeight()
|
||||
{
|
||||
return surfaceHeight;
|
||||
}
|
||||
|
||||
/**
|
||||
* Function called when app requests accelerometer events.
|
||||
* Applications need/should NOT overide this function - it will provide
|
||||
* accelerometer events into the event queue that is accessible
|
||||
* via the calls in nv_event.h
|
||||
*
|
||||
* @param values0: values[0] passed to onSensorChanged(). For accelerometer: Acceleration minus Gx on the x-axis.
|
||||
* @param values1: values[1] passed to onSensorChanged(). For accelerometer: Acceleration minus Gy on the y-axis.
|
||||
* @param values2: values[2] passed to onSensorChanged(). For accelerometer: Acceleration minus Gz on the z-axis.
|
||||
* @return True if the event was handled.
|
||||
*/
|
||||
public native boolean accelerometerEvent(float values0, float values1, float values2);
|
||||
|
||||
/**
|
||||
* The following indented function implementations are defined in libnvevent.a
|
||||
* The application does not and should not overide this; nv_event handles this internally
|
||||
* And remaps as needed into the native calls exposed by nv_event.h
|
||||
*/
|
||||
|
||||
protected native boolean onCreateNative();
|
||||
protected native boolean onStartNative();
|
||||
protected native boolean onRestartNative();
|
||||
protected native boolean onResumeNative();
|
||||
protected native boolean onSurfaceCreatedNative(int w, int h);
|
||||
protected native boolean onFocusChangedNative(boolean focused);
|
||||
protected native boolean onSurfaceChangedNative(int w, int h);
|
||||
protected native boolean onSurfaceDestroyedNative();
|
||||
protected native boolean onPauseNative();
|
||||
protected native boolean onStopNative();
|
||||
protected native boolean onDestroyNative();
|
||||
|
||||
protected native boolean postUserEvent(int u0, int u1, int u2, int u3,
|
||||
boolean blocking);
|
||||
|
||||
public native boolean touchEvent(int action, int x, int y,
|
||||
MotionEvent event);
|
||||
|
||||
public native boolean multiTouchEvent(int action,
|
||||
boolean hasFirst,
|
||||
boolean hasSecond,
|
||||
int x0, int y0, int x1, int y1,
|
||||
MotionEvent event);
|
||||
|
||||
public native boolean keyEvent(int action, int keycode, int unicodeChar, KeyEvent event);
|
||||
|
||||
/**
|
||||
* END indented block, see in comment at top of block
|
||||
*/
|
||||
|
||||
@Override
|
||||
public void onCreate(Bundle savedInstanceState)
|
||||
{
|
||||
System.out.println("**** onCreate");
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
if( wantsAccelerometer && (mSensorManager == null) )
|
||||
mSensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
|
||||
|
||||
// If the app provides a non-null view3d, then we assume that the
|
||||
// app is doing its own custom layout, and we do not create a view
|
||||
// We use the supplied one. Otherwise, we create it for them.
|
||||
if (view3d == null)
|
||||
{
|
||||
System.out.println("**** onCreate: Creating default view");
|
||||
view3d = new SurfaceView(this);
|
||||
setContentView(view3d);
|
||||
}
|
||||
else
|
||||
System.out.println("**** onCreate: App specified custom view");
|
||||
|
||||
SurfaceHolder holder = view3d.getHolder();
|
||||
holder.setType(SurfaceHolder.SURFACE_TYPE_GPU);
|
||||
|
||||
holder.addCallback(new Callback()
|
||||
{
|
||||
// @Override
|
||||
public void surfaceCreated(SurfaceHolder holder)
|
||||
{
|
||||
System.out.println("systemInit.surfaceCreated");
|
||||
cachedSurfaceHolder = holder;
|
||||
|
||||
if (fixedWidth!=0 && fixedHeight!=0)
|
||||
{
|
||||
System.out.println("Setting fixed window size");
|
||||
holder.setFixedSize(fixedWidth, fixedHeight);
|
||||
}
|
||||
|
||||
onSurfaceCreatedNative(surfaceWidth, surfaceHeight);
|
||||
}
|
||||
|
||||
// @Override
|
||||
public void surfaceChanged(SurfaceHolder holder, int format,
|
||||
int width, int height)
|
||||
{
|
||||
cachedSurfaceHolder = holder;
|
||||
System.out.println("Surface changed: " + width + ", " + height);
|
||||
surfaceWidth = width;
|
||||
surfaceHeight = height;
|
||||
onSurfaceChangedNative(surfaceWidth, surfaceHeight);
|
||||
}
|
||||
|
||||
// @Override
|
||||
public void surfaceDestroyed(SurfaceHolder holder)
|
||||
{
|
||||
cachedSurfaceHolder = null;
|
||||
System.out.println("systemInit.surfaceDestroyed");
|
||||
onSurfaceDestroyedNative();
|
||||
}
|
||||
});
|
||||
|
||||
nativeLaunched = true;
|
||||
onCreateNative();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onStart()
|
||||
{
|
||||
System.out.println("**** onStart");
|
||||
super.onStart();
|
||||
|
||||
if (nativeLaunched)
|
||||
onStartNative();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onRestart()
|
||||
{
|
||||
System.out.println("**** onRestart");
|
||||
super.onRestart();
|
||||
|
||||
if (nativeLaunched)
|
||||
onRestartNative();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onResume()
|
||||
{
|
||||
System.out.println("**** onResume");
|
||||
super.onResume();
|
||||
if (nativeLaunched)
|
||||
{
|
||||
if(mSensorManager != null)
|
||||
mSensorManager.registerListener(this,
|
||||
mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER),
|
||||
mSensorDelay);
|
||||
onResumeNative();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onLowMemory ()
|
||||
{
|
||||
System.out.println("**** onLowMemory");
|
||||
super.onLowMemory();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onWindowFocusChanged (boolean hasFocus)
|
||||
{
|
||||
System.out.println("**** onWindowFocusChanged (" + ((hasFocus == true) ? "TRUE" : "FALSE") + ")");
|
||||
if (nativeLaunched)
|
||||
onFocusChangedNative(hasFocus);
|
||||
super.onWindowFocusChanged(hasFocus);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onSaveInstanceState (Bundle outState)
|
||||
{
|
||||
System.out.println("**** onSaveInstanceState");
|
||||
super.onSaveInstanceState(outState);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPause()
|
||||
{
|
||||
System.out.println("**** onPause");
|
||||
super.onPause();
|
||||
if (nativeLaunched)
|
||||
onPauseNative();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onStop()
|
||||
{
|
||||
System.out.println("**** onStop");
|
||||
super.onStop();
|
||||
|
||||
if (nativeLaunched)
|
||||
{
|
||||
if(mSensorManager != null)
|
||||
mSensorManager.unregisterListener(this);
|
||||
|
||||
onStopNative();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDestroy()
|
||||
{
|
||||
System.out.println("**** onDestroy");
|
||||
super.onDestroy();
|
||||
|
||||
if (nativeLaunched)
|
||||
{
|
||||
onDestroyNative();
|
||||
|
||||
CleanupEGL();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation function: defined in libnvevent.a
|
||||
* The application does not and should not overide this; nv_event handles this internally
|
||||
* And remaps as needed into the native calls exposed by nv_event.h
|
||||
*/
|
||||
public void onAccuracyChanged(Sensor sensor, int accuracy) {
|
||||
// Auto-generated method stub
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation function: defined in libnvevent.a
|
||||
* The application does not and should not overide this; nv_event handles this internally
|
||||
* And remaps as needed into the native calls exposed by nv_event.h
|
||||
*/
|
||||
public void onSensorChanged(SensorEvent event) {
|
||||
// Auto-generated method stub
|
||||
if (nativeLaunched && (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER))
|
||||
accelerometerEvent(event.values[0], event.values[1], event.values[2]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation function: defined in libnvevent.a
|
||||
* The application does not and should not overide this; nv_event handles this internally
|
||||
* And remaps as needed into the native calls exposed by nv_event.h
|
||||
*/
|
||||
@Override
|
||||
public boolean onTouchEvent(MotionEvent event)
|
||||
{
|
||||
boolean ret = super.onTouchEvent(event);
|
||||
if (nativeLaunched && !ret)
|
||||
{
|
||||
if (wantsMultitouch)
|
||||
{
|
||||
boolean hasFirst = false;
|
||||
boolean hasSecond = false;
|
||||
|
||||
int x1 = 0, y1 = 0, x2 = 0, y2 = 0;
|
||||
// marshal up the data.
|
||||
int numEvents = event.getPointerCount();
|
||||
for (int i=0; i<numEvents; i++)
|
||||
{
|
||||
// only use pointers 0 and 1...
|
||||
int index = event.getPointerId(i);
|
||||
if (index < 2)
|
||||
{
|
||||
if (index == 0)
|
||||
{
|
||||
hasFirst = true;
|
||||
x1 = (int)event.getX(i);
|
||||
y1 = (int)event.getY(i);
|
||||
}
|
||||
else if (index == 1)
|
||||
{
|
||||
hasSecond = true;
|
||||
x2 = (int)event.getX(i);
|
||||
y2 = (int)event.getY(i);
|
||||
}
|
||||
/* if (count == 0)
|
||||
{
|
||||
x1 = (int)event.getX(i);
|
||||
y1 = (int)event.getY(i);
|
||||
count++;
|
||||
}
|
||||
else if (count == 1)
|
||||
{
|
||||
x2 = (int)event.getX(i);
|
||||
y2 = (int)event.getY(i);
|
||||
count++;
|
||||
}*/
|
||||
}
|
||||
}
|
||||
ret = multiTouchEvent(event.getAction(), hasFirst, hasSecond, x1, y1, x2, y2, event);
|
||||
}
|
||||
else // old style input.*/
|
||||
{
|
||||
ret = touchEvent(event.getAction(), (int)event.getX(), (int)event.getY(), event);
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation function: defined in libnvevent.a
|
||||
* The application does not and should not overide this; nv_event handles this internally
|
||||
* And remaps as needed into the native calls exposed by nv_event.h
|
||||
*/
|
||||
@Override
|
||||
public boolean onKeyDown(int keyCode, KeyEvent event)
|
||||
{
|
||||
if (nativeLaunched && keyEvent(event.getAction(), keyCode, event.getUnicodeChar(), event))
|
||||
return true;
|
||||
return super.onKeyDown(keyCode, event);
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation function: defined in libnvevent.a
|
||||
* The application does not and should not overide this; nv_event handles this internally
|
||||
* And remaps as needed into the native calls exposed by nv_event.h
|
||||
*/
|
||||
@Override
|
||||
public boolean onKeyUp(int keyCode, KeyEvent event)
|
||||
{
|
||||
if (nativeLaunched && keyEvent(event.getAction(), keyCode, event.getUnicodeChar(), event))
|
||||
return true;
|
||||
return super.onKeyUp(keyCode, event);
|
||||
}
|
||||
|
||||
/** The number of bits requested for the red component */
|
||||
protected int redSize = 8;
|
||||
/** The number of bits requested for the green component */
|
||||
protected int greenSize = 8;
|
||||
/** The number of bits requested for the blue component */
|
||||
protected int blueSize = 8;
|
||||
/** The number of bits requested for the alpha component */
|
||||
protected int alphaSize = 8;
|
||||
/** The number of bits requested for the stencil component */
|
||||
protected int stencilSize = 8;
|
||||
/** The number of bits requested for the depth component */
|
||||
protected int depthSize = 24;
|
||||
|
||||
/** Attributes used when selecting the EGLConfig */
|
||||
protected int[] configAttrs = null;
|
||||
/** Attributes used when creating the context */
|
||||
protected int[] contextAttrs = null;
|
||||
|
||||
/**
|
||||
* Called to initialize EGL. This function should not be called by the inheriting
|
||||
* activity, but can be overridden if needed.
|
||||
*
|
||||
* @return True if successful
|
||||
*/
|
||||
protected boolean InitEGL()
|
||||
{
|
||||
if (configAttrs == null)
|
||||
configAttrs = new int[] {EGL10.EGL_NONE};
|
||||
int[] oldConf = configAttrs;
|
||||
|
||||
configAttrs = new int[3 + oldConf.length-1];
|
||||
int i = 0;
|
||||
for (i = 0; i < oldConf.length-1; i++)
|
||||
configAttrs[i] = oldConf[i];
|
||||
configAttrs[i++] = EGL10.EGL_RENDERABLE_TYPE;
|
||||
configAttrs[i++] = 1;
|
||||
configAttrs[i++] = EGL10.EGL_NONE;
|
||||
|
||||
contextAttrs = new int[]
|
||||
{
|
||||
EGL_CONTEXT_CLIENT_VERSION, 1,
|
||||
EGL10.EGL_NONE
|
||||
};
|
||||
|
||||
int[] oldConfES = configAttrs;
|
||||
|
||||
configAttrs = new int[13 + oldConfES.length-1];
|
||||
|
||||
for (i = 0; i < oldConfES.length-1; i++)
|
||||
configAttrs[i] = oldConfES[i];
|
||||
configAttrs[i++] = EGL10.EGL_RED_SIZE;
|
||||
configAttrs[i++] = redSize;
|
||||
configAttrs[i++] = EGL10.EGL_GREEN_SIZE;
|
||||
configAttrs[i++] = greenSize;
|
||||
configAttrs[i++] = EGL10.EGL_BLUE_SIZE;
|
||||
configAttrs[i++] = blueSize;
|
||||
configAttrs[i++] = EGL10.EGL_ALPHA_SIZE;
|
||||
configAttrs[i++] = alphaSize;
|
||||
configAttrs[i++] = EGL10.EGL_STENCIL_SIZE;
|
||||
configAttrs[i++] = stencilSize;
|
||||
configAttrs[i++] = EGL10.EGL_DEPTH_SIZE;
|
||||
configAttrs[i++] = depthSize;
|
||||
configAttrs[i++] = EGL10.EGL_NONE;
|
||||
|
||||
egl = (EGL10) EGLContext.getEGL();
|
||||
System.out.println("getEGL:" + egl.eglGetError());
|
||||
|
||||
eglDisplay = egl.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);
|
||||
System.out.println("eglDisplay: " + eglDisplay + ", err: " + egl.eglGetError());
|
||||
|
||||
int[] version = new int[2];
|
||||
boolean ret = egl.eglInitialize(eglDisplay, version);
|
||||
|
||||
System.out.println("eglInitialize returned: " + ret);
|
||||
if (!ret)
|
||||
return false;
|
||||
|
||||
int eglErr = egl.eglGetError();
|
||||
if (eglErr != EGL10.EGL_SUCCESS)
|
||||
return false;
|
||||
|
||||
System.out.println("eglInitialize err: " + eglErr);
|
||||
|
||||
final EGLConfig[] config = new EGLConfig[20];
|
||||
int num_configs[] = new int[1];
|
||||
egl.eglChooseConfig(eglDisplay, configAttrs, config, config.length, num_configs);
|
||||
System.out.println("eglChooseConfig err: " + egl.eglGetError());
|
||||
|
||||
int score = 1<<24; // to make sure even worst score is better than this, like 8888 when request 565...
|
||||
int val[] = new int[1];
|
||||
for (i = 0; i < num_configs[0]; i++)
|
||||
{
|
||||
boolean cont = true;
|
||||
int currScore = 0;
|
||||
int r, g, b, a, d, s;
|
||||
for (int j = 0; j < (oldConf.length-1)>>1; j++)
|
||||
{
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], configAttrs[j*2], val);
|
||||
if ((val[0] & configAttrs[j*2+1]) != configAttrs[j*2+1])
|
||||
{
|
||||
cont = false; // Doesn't match the "must have" configs
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!cont)
|
||||
continue;
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], EGL10.EGL_RED_SIZE, val); r = val[0];
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], EGL10.EGL_GREEN_SIZE, val); g = val[0];
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], EGL10.EGL_BLUE_SIZE, val); b = val[0];
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], EGL10.EGL_ALPHA_SIZE, val); a = val[0];
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], EGL10.EGL_DEPTH_SIZE, val); d = val[0];
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], EGL10.EGL_STENCIL_SIZE, val); s = val[0];
|
||||
|
||||
System.out.println(">>> EGL Config ["+i+"] R"+r+"G"+g+"B"+b+"A"+a+" D"+d+"S"+s);
|
||||
|
||||
currScore = (Math.abs(r - redSize) + Math.abs(g - greenSize) + Math.abs(b - blueSize) + Math.abs(a - alphaSize)) << 16;
|
||||
currScore += Math.abs(d - depthSize) << 8;
|
||||
currScore += Math.abs(s - stencilSize);
|
||||
|
||||
if (currScore < score)
|
||||
{
|
||||
System.out.println("--------------------------");
|
||||
System.out.println("New config chosen: " + i);
|
||||
for (int j = 0; j < (configAttrs.length-1)>>1; j++)
|
||||
{
|
||||
egl.eglGetConfigAttrib(eglDisplay, config[i], configAttrs[j*2], val);
|
||||
if (val[0] >= configAttrs[j*2+1])
|
||||
System.out.println("setting " + j + ", matches: " + val[0]);
|
||||
}
|
||||
|
||||
score = currScore;
|
||||
eglConfig = config[i];
|
||||
}
|
||||
}
|
||||
|
||||
eglContext = egl.eglCreateContext(eglDisplay, eglConfig, EGL10.EGL_NO_CONTEXT, contextAttrs);
|
||||
System.out.println("eglCreateContext: " + egl.eglGetError());
|
||||
|
||||
gl = (GL11) eglContext.getGL();
|
||||
|
||||
eglInitialized = true;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Called to clean up egl. This function should not be called by the inheriting
|
||||
* activity, but can be overridden if needed.
|
||||
*/
|
||||
protected boolean CleanupEGL()
|
||||
{
|
||||
System.out.println("cleanupEGL");
|
||||
|
||||
if (!eglInitialized)
|
||||
return false;
|
||||
|
||||
if (!DestroySurfaceEGL())
|
||||
return false;
|
||||
|
||||
if (eglDisplay != null)
|
||||
egl.eglMakeCurrent(eglDisplay, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_CONTEXT);
|
||||
if (eglContext != null) {
|
||||
System.out.println("Destroy Context");
|
||||
egl.eglDestroyContext(eglDisplay, eglContext);
|
||||
}
|
||||
if (eglDisplay != null)
|
||||
egl.eglTerminate(eglDisplay);
|
||||
|
||||
eglDisplay = null;
|
||||
eglContext = null;
|
||||
eglSurface = null;
|
||||
|
||||
eglConfig = null;
|
||||
|
||||
surfaceWidth = 0;
|
||||
surfaceHeight = 0;
|
||||
|
||||
eglInitialized = false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Called to create the EGLSurface to be used for rendering. This function should not be called by the inheriting
|
||||
* activity, but can be overridden if needed.
|
||||
*
|
||||
* @param surface The SurfaceHolder that holds the surface that we are going to render to.
|
||||
* @return True if successful
|
||||
*/
|
||||
protected boolean CreateSurfaceEGL()
|
||||
{
|
||||
if (cachedSurfaceHolder == null)
|
||||
{
|
||||
System.out.println("createEGLSurface failed, cachedSurfaceHolder is null");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!eglInitialized && (eglInitialized = InitEGL()))
|
||||
{
|
||||
System.out.println("createEGLSurface failed, cannot initialize EGL");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (eglDisplay == null)
|
||||
{
|
||||
System.out.println("createEGLSurface: display is null");
|
||||
return false;
|
||||
}
|
||||
else if (eglConfig == null)
|
||||
{
|
||||
System.out.println("createEGLSurface: config is null");
|
||||
return false;
|
||||
}
|
||||
eglSurface = egl.eglCreateWindowSurface(eglDisplay, eglConfig, cachedSurfaceHolder, null);
|
||||
System.out.println("eglSurface: " + eglSurface + ", err: " + egl.eglGetError());
|
||||
int sizes[] = new int[1];
|
||||
|
||||
egl.eglQuerySurface(eglDisplay, eglSurface, EGL10.EGL_WIDTH, sizes);
|
||||
surfaceWidth = sizes[0];
|
||||
egl.eglQuerySurface(eglDisplay, eglSurface, EGL10.EGL_HEIGHT, sizes);
|
||||
surfaceHeight = sizes[0];
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Destroys the EGLSurface used for rendering. This function should not be called by the inheriting
|
||||
* activity, but can be overridden if needed.
|
||||
*/
|
||||
protected boolean DestroySurfaceEGL()
|
||||
{
|
||||
if (eglDisplay != null && eglSurface != null)
|
||||
egl.eglMakeCurrent(eglDisplay, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_SURFACE, eglContext);
|
||||
if (eglSurface != null)
|
||||
egl.eglDestroySurface(eglDisplay, eglSurface);
|
||||
eglSurface = null;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean BindSurfaceAndContextEGL()
|
||||
{
|
||||
if (eglContext == null)
|
||||
{
|
||||
System.out.println("eglContext is NULL");
|
||||
return false;
|
||||
}
|
||||
else if (eglSurface == null)
|
||||
{
|
||||
System.out.println("eglSurface is NULL");
|
||||
return false;
|
||||
}
|
||||
else if (!egl.eglMakeCurrent(eglDisplay, eglSurface, eglSurface, eglContext))
|
||||
{
|
||||
System.out.println("eglMakeCurrent err: " + egl.eglGetError());
|
||||
return false;
|
||||
}
|
||||
|
||||
// nvAcquireTimeExtension();
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Implementation function:
|
||||
* The application does not and should not overide or call this directly
|
||||
* Instead, the application should call NVEventEGLUnmakeCurrent(),
|
||||
* which is declared in nv_event.h
|
||||
*/
|
||||
public boolean UnbindSurfaceAndContextEGL()
|
||||
{
|
||||
System.out.println("UnbindSurfaceAndContextEGL");
|
||||
if (eglDisplay == null)
|
||||
{
|
||||
System.out.println("UnbindSurfaceAndContextEGL: display is null");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!egl.eglMakeCurrent(eglDisplay, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_CONTEXT))
|
||||
{
|
||||
System.out.println("egl(Un)MakeCurrent err: " + egl.eglGetError());
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public boolean SwapBuffersEGL()
|
||||
{
|
||||
//long stopTime;
|
||||
//long startTime = nvGetSystemTime();
|
||||
if (eglSurface == null)
|
||||
{
|
||||
System.out.println("eglSurface is NULL");
|
||||
return false;
|
||||
}
|
||||
else if (!egl.eglSwapBuffers(eglDisplay, eglSurface))
|
||||
{
|
||||
System.out.println("eglSwapBufferrr: " + egl.eglGetError());
|
||||
return false;
|
||||
}
|
||||
//stopTime = nvGetSystemTime();
|
||||
//String s = String.format("%d ms in eglSwapBuffers", (int)(stopTime - startTime));
|
||||
//Log.v("EventAccelerometer", s);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public int GetErrorEGL()
|
||||
{
|
||||
return egl.eglGetError();
|
||||
}
|
||||
}
|
|
@ -408,6 +408,7 @@ void Framework::StartDrag(DragEvent const & e)
|
|||
|
||||
m_navigator.StartDrag(pt, 0./*m_timer.ElapsedSeconds()*/);
|
||||
m_renderPolicy->StartDrag();
|
||||
// LOG(LINFO, ("StartDrag", e.Pos()));
|
||||
}
|
||||
|
||||
void Framework::DoDrag(DragEvent const & e)
|
||||
|
@ -422,6 +423,7 @@ void Framework::DoDrag(DragEvent const & e)
|
|||
|
||||
m_navigator.DoDrag(pt, 0./*m_timer.ElapsedSeconds()*/);
|
||||
m_renderPolicy->DoDrag();
|
||||
// LOG(LINFO, ("DoDrag", e.Pos()));
|
||||
}
|
||||
|
||||
void Framework::StopDrag(DragEvent const & e)
|
||||
|
@ -435,6 +437,8 @@ void Framework::StopDrag(DragEvent const & e)
|
|||
#endif
|
||||
|
||||
m_renderPolicy->StopDrag();
|
||||
|
||||
// LOG(LINFO, ("StopDrag", e.Pos()));
|
||||
}
|
||||
|
||||
void Framework::StartRotate(RotateEvent const & e)
|
||||
|
@ -518,6 +522,8 @@ void Framework::StartScale(ScaleEvent const & e)
|
|||
|
||||
m_navigator.StartScale(pt1, pt2, 0./*m_timer.ElapsedSeconds()*/);
|
||||
m_renderPolicy->StartScale();
|
||||
|
||||
// LOG(LINFO, ("StartScale", e.Pt1(), e.Pt2()));
|
||||
}
|
||||
|
||||
void Framework::DoScale(ScaleEvent const & e)
|
||||
|
@ -540,6 +546,7 @@ void Framework::DoScale(ScaleEvent const & e)
|
|||
|
||||
m_navigator.DoScale(pt1, pt2, 0./*m_timer.ElapsedSeconds()*/);
|
||||
m_renderPolicy->DoScale();
|
||||
// LOG(LINFO, ("DoScale", e.Pt1(), e.Pt2()));
|
||||
}
|
||||
|
||||
void Framework::StopScale(ScaleEvent const & e)
|
||||
|
@ -562,6 +569,7 @@ void Framework::StopScale(ScaleEvent const & e)
|
|||
|
||||
m_navigator.StopScale(pt1, pt2, 0./*m_timer.ElapsedSeconds()*/);
|
||||
m_renderPolicy->StopScale();
|
||||
// LOG(LINFO, ("StopScale", e.Pt1(), e.Pt2()));
|
||||
}
|
||||
|
||||
search::Engine * Framework::GetSearchEngine()
|
||||
|
@ -593,25 +601,33 @@ void Framework::Search(string const & text, SearchCallbackT callback)
|
|||
|
||||
void Framework::SetRenderPolicy(RenderPolicy * renderPolicy)
|
||||
{
|
||||
bool isVisualLogEnabled = false;
|
||||
Settings::Get("VisualLog", isVisualLogEnabled);
|
||||
m_informationDisplay.enableLog(isVisualLogEnabled, renderPolicy->GetWindowHandle().get());
|
||||
m_informationDisplay.setVisualScale(GetPlatform().VisualScale());
|
||||
if (renderPolicy)
|
||||
{
|
||||
bool isVisualLogEnabled = false;
|
||||
Settings::Get("VisualLog", isVisualLogEnabled);
|
||||
m_informationDisplay.enableLog(isVisualLogEnabled, renderPolicy->GetWindowHandle().get());
|
||||
m_informationDisplay.setVisualScale(GetPlatform().VisualScale());
|
||||
}
|
||||
|
||||
yg::gl::RenderContext::initParams();
|
||||
|
||||
m_renderPolicy.reset();
|
||||
m_renderPolicy.reset(renderPolicy);
|
||||
|
||||
m_renderPolicy->SetRenderFn(DrawModelFn());
|
||||
|
||||
m_navigator.SetSupportRotation(m_renderPolicy->DoSupportRotation());
|
||||
|
||||
if ((m_width != 0) && (m_height != 0))
|
||||
OnSize(m_width, m_height);
|
||||
|
||||
if (m_hasPendingInvalidate)
|
||||
if (m_renderPolicy.get())
|
||||
{
|
||||
m_renderPolicy->GetWindowHandle()->invalidate();
|
||||
m_hasPendingInvalidate = false;
|
||||
m_renderPolicy->SetRenderFn(DrawModelFn());
|
||||
|
||||
m_navigator.SetSupportRotation(m_renderPolicy->DoSupportRotation());
|
||||
|
||||
if ((m_width != 0) && (m_height != 0))
|
||||
OnSize(m_width, m_height);
|
||||
|
||||
if (m_hasPendingInvalidate)
|
||||
{
|
||||
m_renderPolicy->GetWindowHandle()->invalidate();
|
||||
m_hasPendingInvalidate = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -129,6 +129,8 @@ namespace yg
|
|||
if (!fin)
|
||||
break;
|
||||
|
||||
LOG(LINFO, ("whitelisting ", fontName, " for ", ubName));
|
||||
|
||||
if (ubName == "*")
|
||||
for (unicode_blocks_t::iterator it = m_unicodeBlocks.begin(); it != m_unicodeBlocks.end(); ++it)
|
||||
it->m_whitelist.push_back(fontName);
|
||||
|
@ -162,6 +164,8 @@ namespace yg
|
|||
if (!fin)
|
||||
break;
|
||||
|
||||
LOG(LINFO, ("blacklisting ", fontName, " for ", ubName));
|
||||
|
||||
if (ubName == "*")
|
||||
for (unicode_blocks_t::iterator it = m_unicodeBlocks.begin(); it != m_unicodeBlocks.end(); ++it)
|
||||
it->m_blacklist.push_back(fontName);
|
||||
|
|
Loading…
Add table
Reference in a new issue