istereo2

changeset 23:7d795dade0bc

first pass at android port... compiles!
author John Tsiombikas <nuclear@member.fsf.org>
date Sat, 03 Oct 2015 06:10:30 +0300 (2015-10-03)
parents c6971ff4795e
children 9d53a4938ce8
files .hgignore android/Makefile android/manifest.xml.in libs/Makefile src/android/amain.c src/android/android_native_app_glue.c src/android/android_native_app_glue.h src/android/assman.c src/android/logger.c src/android/logger.h src/opengl.h
diffstat 11 files changed, 1290 insertions(+), 12 deletions(-) [+]
line diff
     1.1 --- a/.hgignore	Fri Oct 02 07:10:19 2015 +0300
     1.2 +++ b/.hgignore	Sat Oct 03 06:10:30 2015 +0300
     1.3 @@ -11,3 +11,11 @@
     1.4  ^data/
     1.5  \.DS_Store
     1.6  ^test$
     1.7 +^android/bin/
     1.8 +^android/libs/
     1.9 +^android/res/
    1.10 +^android/src/
    1.11 +^android/assets/
    1.12 +\.xml$
    1.13 +\.properties$
    1.14 +proguard-project\.txt$
     2.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     2.2 +++ b/android/Makefile	Sat Oct 03 06:10:30 2015 +0300
     2.3 @@ -0,0 +1,106 @@
     2.4 +bin = stereotunnel
     2.5 +root = ..
     2.6 +src = $(wildcard $(root)/src/*.c)
     2.7 +ccsrc = $(wildcard $(root)/src/*.cc)
     2.8 +
     2.9 +include $(root)/libs/Makefile
    2.10 +
    2.11 +# --- android settings ---
    2.12 +android_platform = android-19
    2.13 +name = $(bin)
    2.14 +pkgprefix = com.mutantstargoat
    2.15 +# ------------------------
    2.16 +
    2.17 +src += $(wildcard $(root)/src/android/*.c)
    2.18 +
    2.19 +obj = $(src:.c=.o) $(ccsrc:.cc=.o)
    2.20 +lib = libs/armeabi/lib$(name).so
    2.21 +apk-release = bin/$(name).apk
    2.22 +apk-debug = bin/$(name)-debug.apk
    2.23 +
    2.24 +pkg = $(pkgprefix).$(name)
    2.25 +act = android.app.NativeActivity
    2.26 +
    2.27 +CC = arm-linux-androideabi-gcc
    2.28 +CXX = arm-linux-androideabi-g++
    2.29 +
    2.30 +android_libs = -lstdc++ -llog -landroid -lEGL -lGLESv2
    2.31 +
    2.32 +defs += -DANDROID -DAPP_NAME=\"$(name)\" -DNO_FREETYPE -DHAVE_OPENGL_H
    2.33 +CXXFLAGS = -Wall -g $(defs) -march=armv7-a -mfloat-abi=softfp -mfpu=vfpv3-d16 \
    2.34 +		 $(android_inc) -I$(root)/src -I$(root)/src/android $(incdir)
    2.35 +CFLAGS = -std=c99 $(CXXFLAGS)
    2.36 +LDFLAGS = -Wl,--fix-cortex-a8 -Wl,-z,defs $(android_libs) $(libpaths) -lm
    2.37 +
    2.38 +.PHONY: debug
    2.39 +debug: $(apk-debug)
    2.40 +
    2.41 +.PHONY: release
    2.42 +release: $(apk-release)
    2.43 +
    2.44 +$(apk-debug): $(lib) $(jsrc) AndroidManifest.xml assets/data assets/sdr
    2.45 +	ant debug
    2.46 +
    2.47 +$(apk-release): $(lib) $(jsrc) AndroidManifest.xml assets/data assets/sdr
    2.48 +	ant release
    2.49 +
    2.50 +assets/data:
    2.51 +	mkdir -p assets
    2.52 +	cd assets && rm -f data && ln -s ../../data .
    2.53 +
    2.54 +assets/sdr:
    2.55 +	mkdir -p assets
    2.56 +	cd assets && rm -f sdr && ln -s ../../sdr .
    2.57 +
    2.58 +.PHONY: lib
    2.59 +lib: $(lib)
    2.60 +
    2.61 +$(lib): $(obj)
    2.62 +	@mkdir -p libs/armeabi
    2.63 +	$(CXX) -o $@ -shared $(obj) $(LDFLAGS)
    2.64 +
    2.65 +ant_files = *.xml *.properties proguard-project.txt
    2.66 +ant_dirs = bin libs res obj src gen
    2.67 +
    2.68 +.PHONY: clean
    2.69 +clean:
    2.70 +	rm -f $(obj) $(lib) $(apk-release) $(apk-debug) $(ant_files)
    2.71 +	rm -rf $(ant_dirs)
    2.72 +	rm -f assets/data assets/sdr
    2.73 +	[ -d assets ] && rmdir assets || true
    2.74 +
    2.75 +.PHONY: install
    2.76 +install: install-debug
    2.77 +
    2.78 +.PHONY: install-debug
    2.79 +install-debug:
    2.80 +	adb install -r $(apk-debug)
    2.81 +
    2.82 +.PHONY: install-release
    2.83 +install-release:
    2.84 +	adb install -r $(apk-release)
    2.85 +
    2.86 +.PHONY: uninstall
    2.87 +uninstall:
    2.88 +	adb uninstall $(pkg)
    2.89 +
    2.90 +.PHONY: run
    2.91 +run:
    2.92 +	adb shell am start -n $(pkg)/$(act)
    2.93 +
    2.94 +.PHONY: stop
    2.95 +stop:
    2.96 +	adb shell am force-stop $(pkg)
    2.97 +
    2.98 +AndroidManifest.xml: manifest.xml.in
    2.99 +	android create project -p . -t $(android_platform) -k $(pkg) -a NativeActivity -n $(name)
   2.100 +	cat manifest.xml.in | sed 's/$$APPNAME/$(name)/g' | sed 's/$$APPTITLE/$(title)/g' >$@
   2.101 +#	cd src && rm -f *.java && ln -s ../../src/android/*.java .
   2.102 +
   2.103 +.PHONY: update-project
   2.104 +update-project: build.xml
   2.105 +	android update project -p . -t $(android_platform)
   2.106 +
   2.107 +.PHONY: logcat
   2.108 +logcat:
   2.109 +	adb logcat $(name):V AndroidRuntime:V DEBUG:V '*:S'
     3.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     3.2 +++ b/android/manifest.xml.in	Sat Oct 03 06:10:30 2015 +0300
     3.3 @@ -0,0 +1,25 @@
     3.4 +<?xml version="1.0" encoding="utf-8"?>
     3.5 +<manifest xmlns:android="http://schemas.android.com/apk/res/android"
     3.6 +	package="com.mutantstargoat.$APPNAME"
     3.7 +	android:versionCode="1"
     3.8 +	android:versionName="1.0">
     3.9 +
    3.10 +	<uses-sdk android:minSdkVersion="9"/>
    3.11 +
    3.12 +	<application android:label="$APPNAME"
    3.13 +			android:hasCode="false"
    3.14 +			android:debuggable="true">
    3.15 +			<!-- android:icon="@drawable/ic_launcher" -->
    3.16 +
    3.17 +		<activity android:name="android.app.NativeActivity"
    3.18 +			android:label="$APPTITLE">
    3.19 +
    3.20 +			<meta-data android:name="android.app.lib_name" android:value="$APPNAME"/>
    3.21 +
    3.22 +			<intent-filter>
    3.23 +				<action android:name="android.intent.action.MAIN" />
    3.24 +				<category android:name="android.intent.category.LAUNCHER" />
    3.25 +			</intent-filter>
    3.26 +		</activity>
    3.27 +	</application>
    3.28 +</manifest>
     4.1 --- a/libs/Makefile	Fri Oct 02 07:10:19 2015 +0300
     4.2 +++ b/libs/Makefile	Sat Oct 03 06:10:30 2015 +0300
     4.3 @@ -1,11 +1,12 @@
     4.4 -src += $(wildcard libs/zlib/*.c) \
     4.5 -	   $(wildcard libs/libpng/*.c) \
     4.6 -	   $(wildcard libs/libjpeg/*.c) \
     4.7 -	   $(wildcard libs/imago2/*.c) \
     4.8 -	   $(wildcard libs/vmath/*.c) \
     4.9 -	   $(wildcard libs/drawtext/*.c)
    4.10 +src += $(wildcard $(root)/libs/zlib/*.c) \
    4.11 +	   $(wildcard $(root)/libs/libpng/*.c) \
    4.12 +	   $(wildcard $(root)/libs/libjpeg/*.c) \
    4.13 +	   $(wildcard $(root)/libs/imago2/*.c) \
    4.14 +	   $(wildcard $(root)/libs/vmath/*.c) \
    4.15 +	   $(wildcard $(root)/libs/drawtext/*.c)
    4.16  
    4.17 -ccsrc += $(wildcard libs/goatkit/*.cc)
    4.18 +ccsrc += $(wildcard $(root)/libs/goatkit/*.cc)
    4.19  
    4.20 -incdir += -Ilibs -Ilibs/imago2 -Ilibs/zlib -Ilibs/libpng -Ilibs/libjpeg \
    4.21 -		  -Ilibs/vmath -Ilibs/drawtext
    4.22 +incdir += -I$(root)/libs -I$(root)/libs/imago2 -I$(root)/libs/zlib \
    4.23 +		  -I$(root)/libs/libpng -I$(root)/libs/libjpeg \
    4.24 +		  -I$(root)/libs/vmath -I$(root)/libs/drawtext
     5.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     5.2 +++ b/src/android/amain.c	Sat Oct 03 06:10:30 2015 +0300
     5.3 @@ -0,0 +1,275 @@
     5.4 +#include <stdio.h>
     5.5 +#include <stdlib.h>
     5.6 +#include <assert.h>
     5.7 +#include <EGL/egl.h>
     5.8 +#include <jni.h>
     5.9 +#include "android_native_app_glue.h"
    5.10 +#include <android/sensor.h>
    5.11 +#include "logger.h"
    5.12 +#include "istereo.h"
    5.13 +
    5.14 +struct android_app *app;
    5.15 +
    5.16 +static void handle_command(struct android_app *app, int32_t cmd);
    5.17 +static int handle_input(struct android_app *app, AInputEvent *ev);
    5.18 +static int handle_touch_input(struct android_app *app, AInputEvent *ev);
    5.19 +static int init_gl(void);
    5.20 +static void destroy_gl(void);
    5.21 +
    5.22 +static EGLDisplay dpy;
    5.23 +static EGLSurface surf;
    5.24 +static EGLContext ctx;
    5.25 +
    5.26 +static int redisp_pending = 1;	/* TODO stop busy-looping */
    5.27 +
    5.28 +static int width, height;
    5.29 +static int init_done;
    5.30 +
    5.31 +void android_main(struct android_app *app_ptr)
    5.32 +{
    5.33 +	app_dummy();
    5.34 +	app = app_ptr;
    5.35 +
    5.36 +	app->onAppCmd = handle_command;
    5.37 +	app->onInputEvent = handle_input;
    5.38 +
    5.39 +	start_logger();
    5.40 +
    5.41 +	for(;;) {
    5.42 +		int num_events;
    5.43 +		struct android_poll_source *pollsrc;
    5.44 +
    5.45 +		while(ALooper_pollAll(0, 0, &num_events, (void**)&pollsrc) >= 0) {
    5.46 +			if(pollsrc) {
    5.47 +				pollsrc->process(app, pollsrc);
    5.48 +			}
    5.49 +		}
    5.50 +
    5.51 +		if(app->destroyRequested) {
    5.52 +			return;
    5.53 +		}
    5.54 +
    5.55 +		if(init_done && redisp_pending) {
    5.56 +			redraw();
    5.57 +			eglSwapBuffers(dpy, surf);
    5.58 +		}
    5.59 +	}
    5.60 +}
    5.61 +
    5.62 +void set_mouse_pos(int x, int y)
    5.63 +{
    5.64 +}
    5.65 +
    5.66 +void set_mouse_cursor(int enable)
    5.67 +{
    5.68 +}
    5.69 +
    5.70 +/* TODO */
    5.71 +void ad_banner_show(void)
    5.72 +{
    5.73 +}
    5.74 +
    5.75 +void ad_banner_hide(void)
    5.76 +{
    5.77 +}
    5.78 +
    5.79 +static void handle_command(struct android_app *app, int32_t cmd)
    5.80 +{
    5.81 +	switch(cmd) {
    5.82 +	case APP_CMD_SAVE_STATE:
    5.83 +		/* save the application state to be reloaded on restart if needed */
    5.84 +		break;
    5.85 +
    5.86 +	case APP_CMD_INIT_WINDOW:
    5.87 +		if(init_gl() == -1) {
    5.88 +			exit(1);
    5.89 +		}
    5.90 +		/* initialize the application */
    5.91 +		if(init() == -1) {
    5.92 +			exit(1);	/* initialization failed, quit */
    5.93 +		}
    5.94 +		init_done = 1;
    5.95 +		break;
    5.96 +
    5.97 +	case APP_CMD_TERM_WINDOW:
    5.98 +		/* cleanup */
    5.99 +		init_done = 0;
   5.100 +		cleanup();
   5.101 +		destroy_gl();
   5.102 +		break;
   5.103 +
   5.104 +	case APP_CMD_GAINED_FOCUS:
   5.105 +		/* app focused */
   5.106 +		break;
   5.107 +
   5.108 +	case APP_CMD_LOST_FOCUS:
   5.109 +		/* app lost focus */
   5.110 +		break;
   5.111 +
   5.112 +	case APP_CMD_WINDOW_RESIZED:
   5.113 +	case APP_CMD_CONFIG_CHANGED:
   5.114 +		{
   5.115 +			int nx = ANativeWindow_getWidth(app->window);
   5.116 +			int ny = ANativeWindow_getHeight(app->window);
   5.117 +			if(nx != width || ny != height) {
   5.118 +				reshape(nx, ny);
   5.119 +				width = nx;
   5.120 +				height = ny;
   5.121 +			}
   5.122 +		}
   5.123 +		break;
   5.124 +
   5.125 +	default:
   5.126 +		break;
   5.127 +	}
   5.128 +}
   5.129 +
   5.130 +static int handle_input(struct android_app *app, AInputEvent *ev)
   5.131 +{
   5.132 +	int evtype = AInputEvent_getType(ev);
   5.133 +
   5.134 +	switch(evtype) {
   5.135 +	case AINPUT_EVENT_TYPE_MOTION:
   5.136 +		return handle_touch_input(app, ev);
   5.137 +
   5.138 +	default:
   5.139 +		break;
   5.140 +	}
   5.141 +	return 0;
   5.142 +}
   5.143 +
   5.144 +#define MAX_TOUCH_IDS	32
   5.145 +
   5.146 +static int handle_touch_input(struct android_app *app, AInputEvent *ev)
   5.147 +{
   5.148 +	int x, y, idx, touch_id;
   5.149 +	unsigned int action;
   5.150 +	static int prev_pos[MAX_TOUCH_IDS][2];
   5.151 +
   5.152 +	action = AMotionEvent_getAction(ev);
   5.153 +
   5.154 +	idx = (action & AMOTION_EVENT_ACTION_POINTER_INDEX_MASK) >>
   5.155 +		AMOTION_EVENT_ACTION_POINTER_INDEX_SHIFT;
   5.156 +	touch_id = AMotionEvent_getPointerId(ev, idx);
   5.157 +
   5.158 +	x = AMotionEvent_getX(ev, idx);
   5.159 +	y = AMotionEvent_getY(ev, idx);
   5.160 +
   5.161 +	switch(action & AMOTION_EVENT_ACTION_MASK) {
   5.162 +	case AMOTION_EVENT_ACTION_DOWN:
   5.163 +	case AMOTION_EVENT_ACTION_POINTER_DOWN:
   5.164 +		if(touch_id == 0) {
   5.165 +			mouse_button(0, 1, x, y);
   5.166 +		}
   5.167 +		if(touch_id < MAX_TOUCH_IDS) {
   5.168 +			prev_pos[touch_id][0] = x;
   5.169 +			prev_pos[touch_id][1] = y;
   5.170 +		}
   5.171 +		break;
   5.172 +
   5.173 +	case AMOTION_EVENT_ACTION_UP:
   5.174 +	case AMOTION_EVENT_ACTION_POINTER_UP:
   5.175 +		if(touch_id == 0) {
   5.176 +			mouse_button(0, 0, x, y);
   5.177 +		}
   5.178 +		if(touch_id < MAX_TOUCH_IDS) {
   5.179 +			prev_pos[touch_id][0] = x;
   5.180 +			prev_pos[touch_id][1] = y;
   5.181 +		}
   5.182 +		break;
   5.183 +
   5.184 +	case AMOTION_EVENT_ACTION_MOVE:
   5.185 +		{
   5.186 +			int i, pcount = AMotionEvent_getPointerCount(ev);
   5.187 +			for(i=0; i<pcount; i++) {
   5.188 +				int id = AMotionEvent_getPointerId(ev, i);
   5.189 +				if(id < MAX_TOUCH_IDS && x != prev_pos[id][0] && y != prev_pos[id][1]) {
   5.190 +					if(id == 0) {
   5.191 +						mouse_motion(x, y);
   5.192 +					}
   5.193 +					prev_pos[id][0] = x;
   5.194 +					prev_pos[id][1] = y;
   5.195 +				}
   5.196 +			}
   5.197 +		}
   5.198 +		break;
   5.199 +
   5.200 +	default:
   5.201 +		break;
   5.202 +	}
   5.203 +
   5.204 +	return 1;
   5.205 +}
   5.206 +
   5.207 +
   5.208 +static int init_gl(void)
   5.209 +{
   5.210 +	static const int eglattr[] = {
   5.211 +		EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
   5.212 +		EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
   5.213 +		EGL_RED_SIZE, 5,
   5.214 +		EGL_GREEN_SIZE, 5,
   5.215 +		EGL_BLUE_SIZE, 5,
   5.216 +		EGL_DEPTH_SIZE, 16,
   5.217 +		EGL_NONE
   5.218 +	};
   5.219 +	static const int ctxattr[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE };
   5.220 +
   5.221 +	EGLConfig eglcfg;
   5.222 +	int count, vis;
   5.223 +
   5.224 +	dpy = eglGetDisplay(EGL_DEFAULT_DISPLAY);
   5.225 +	if(!dpy || !eglInitialize(dpy, 0, 0)) {
   5.226 +		fprintf(stderr, "failed to initialize EGL\n");
   5.227 +		destroy_gl();
   5.228 +		return -1;
   5.229 +	}
   5.230 +
   5.231 +	if(!eglChooseConfig(dpy, eglattr, &eglcfg, 1, &count)) {
   5.232 +		fprintf(stderr, "no matching EGL config found\n");
   5.233 +		destroy_gl();
   5.234 +		return -1;
   5.235 +	}
   5.236 +
   5.237 +	/* configure the native window visual according to the chosen EGL config */
   5.238 +	eglGetConfigAttrib(dpy, &eglcfg, EGL_NATIVE_VISUAL_ID, &vis);
   5.239 +	ANativeWindow_setBuffersGeometry(app->window, 0, 0, vis);
   5.240 +
   5.241 +	if(!(surf = eglCreateWindowSurface(dpy, eglcfg, app->window, 0))) {
   5.242 +		fprintf(stderr, "failed to create window\n");
   5.243 +		destroy_gl();
   5.244 +		return -1;
   5.245 +	}
   5.246 +
   5.247 +	if(!(ctx = eglCreateContext(dpy, eglcfg, EGL_NO_CONTEXT, ctxattr))) {
   5.248 +		fprintf(stderr, "failed to create OpenGL ES context\n");
   5.249 +		destroy_gl();
   5.250 +		return -1;
   5.251 +	}
   5.252 +	eglMakeCurrent(dpy, surf, surf, ctx);
   5.253 +
   5.254 +	eglQuerySurface(dpy, surf, EGL_WIDTH, &width);
   5.255 +	eglQuerySurface(dpy, surf, EGL_HEIGHT, &height);
   5.256 +	reshape(width, height);
   5.257 +
   5.258 +	return 0;
   5.259 +}
   5.260 +
   5.261 +static void destroy_gl(void)
   5.262 +{
   5.263 +	if(!dpy) return;
   5.264 +
   5.265 +	eglMakeCurrent(dpy, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
   5.266 +
   5.267 +	if(ctx) {
   5.268 +		eglDestroyContext(dpy, ctx);
   5.269 +		ctx = 0;
   5.270 +	}
   5.271 +	if(surf) {
   5.272 +		eglDestroySurface(dpy, surf);
   5.273 +		surf = 0;
   5.274 +	}
   5.275 +
   5.276 +	eglTerminate(dpy);
   5.277 +	dpy = 0;
   5.278 +}
     6.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     6.2 +++ b/src/android/android_native_app_glue.c	Sat Oct 03 06:10:30 2015 +0300
     6.3 @@ -0,0 +1,440 @@
     6.4 +/*
     6.5 + * Copyright (C) 2010 The Android Open Source Project
     6.6 + *
     6.7 + * Licensed under the Apache License, Version 2.0 (the "License");
     6.8 + * you may not use this file except in compliance with the License.
     6.9 + * You may obtain a copy of the License at
    6.10 + *
    6.11 + *      http://www.apache.org/licenses/LICENSE-2.0
    6.12 + *
    6.13 + * Unless required by applicable law or agreed to in writing, software
    6.14 + * distributed under the License is distributed on an "AS IS" BASIS,
    6.15 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    6.16 + * See the License for the specific language governing permissions and
    6.17 + * limitations under the License.
    6.18 + *
    6.19 + */
    6.20 +
    6.21 +#include <jni.h>
    6.22 +
    6.23 +#include <errno.h>
    6.24 +#include <string.h>
    6.25 +#include <unistd.h>
    6.26 +#include <sys/resource.h>
    6.27 +
    6.28 +#include "android_native_app_glue.h"
    6.29 +#include <android/log.h>
    6.30 +
    6.31 +#define LOGI(...) ((void)__android_log_print(ANDROID_LOG_INFO, "threaded_app", __VA_ARGS__))
    6.32 +#define LOGE(...) ((void)__android_log_print(ANDROID_LOG_ERROR, "threaded_app", __VA_ARGS__))
    6.33 +
    6.34 +/* For debug builds, always enable the debug traces in this library */
    6.35 +#ifndef NDEBUG
    6.36 +#  define LOGV(...)  ((void)__android_log_print(ANDROID_LOG_VERBOSE, "threaded_app", __VA_ARGS__))
    6.37 +#else
    6.38 +#  define LOGV(...)  ((void)0)
    6.39 +#endif
    6.40 +
    6.41 +static void free_saved_state(struct android_app* android_app) {
    6.42 +    pthread_mutex_lock(&android_app->mutex);
    6.43 +    if (android_app->savedState != NULL) {
    6.44 +        free(android_app->savedState);
    6.45 +        android_app->savedState = NULL;
    6.46 +        android_app->savedStateSize = 0;
    6.47 +    }
    6.48 +    pthread_mutex_unlock(&android_app->mutex);
    6.49 +}
    6.50 +
    6.51 +int8_t android_app_read_cmd(struct android_app* android_app) {
    6.52 +    int8_t cmd;
    6.53 +    if (read(android_app->msgread, &cmd, sizeof(cmd)) == sizeof(cmd)) {
    6.54 +        switch (cmd) {
    6.55 +            case APP_CMD_SAVE_STATE:
    6.56 +                free_saved_state(android_app);
    6.57 +                break;
    6.58 +        }
    6.59 +        return cmd;
    6.60 +    } else {
    6.61 +        LOGE("No data on command pipe!");
    6.62 +    }
    6.63 +    return -1;
    6.64 +}
    6.65 +
    6.66 +static void print_cur_config(struct android_app* android_app) {
    6.67 +    char lang[2], country[2];
    6.68 +    AConfiguration_getLanguage(android_app->config, lang);
    6.69 +    AConfiguration_getCountry(android_app->config, country);
    6.70 +
    6.71 +    LOGV("Config: mcc=%d mnc=%d lang=%c%c cnt=%c%c orien=%d touch=%d dens=%d "
    6.72 +            "keys=%d nav=%d keysHid=%d navHid=%d sdk=%d size=%d long=%d "
    6.73 +            "modetype=%d modenight=%d",
    6.74 +            AConfiguration_getMcc(android_app->config),
    6.75 +            AConfiguration_getMnc(android_app->config),
    6.76 +            lang[0], lang[1], country[0], country[1],
    6.77 +            AConfiguration_getOrientation(android_app->config),
    6.78 +            AConfiguration_getTouchscreen(android_app->config),
    6.79 +            AConfiguration_getDensity(android_app->config),
    6.80 +            AConfiguration_getKeyboard(android_app->config),
    6.81 +            AConfiguration_getNavigation(android_app->config),
    6.82 +            AConfiguration_getKeysHidden(android_app->config),
    6.83 +            AConfiguration_getNavHidden(android_app->config),
    6.84 +            AConfiguration_getSdkVersion(android_app->config),
    6.85 +            AConfiguration_getScreenSize(android_app->config),
    6.86 +            AConfiguration_getScreenLong(android_app->config),
    6.87 +            AConfiguration_getUiModeType(android_app->config),
    6.88 +            AConfiguration_getUiModeNight(android_app->config));
    6.89 +}
    6.90 +
    6.91 +void android_app_pre_exec_cmd(struct android_app* android_app, int8_t cmd) {
    6.92 +    switch (cmd) {
    6.93 +        case APP_CMD_INPUT_CHANGED:
    6.94 +            LOGV("APP_CMD_INPUT_CHANGED\n");
    6.95 +            pthread_mutex_lock(&android_app->mutex);
    6.96 +            if (android_app->inputQueue != NULL) {
    6.97 +                AInputQueue_detachLooper(android_app->inputQueue);
    6.98 +            }
    6.99 +            android_app->inputQueue = android_app->pendingInputQueue;
   6.100 +            if (android_app->inputQueue != NULL) {
   6.101 +                LOGV("Attaching input queue to looper");
   6.102 +                AInputQueue_attachLooper(android_app->inputQueue,
   6.103 +                        android_app->looper, LOOPER_ID_INPUT, NULL,
   6.104 +                        &android_app->inputPollSource);
   6.105 +            }
   6.106 +            pthread_cond_broadcast(&android_app->cond);
   6.107 +            pthread_mutex_unlock(&android_app->mutex);
   6.108 +            break;
   6.109 +
   6.110 +        case APP_CMD_INIT_WINDOW:
   6.111 +            LOGV("APP_CMD_INIT_WINDOW\n");
   6.112 +            pthread_mutex_lock(&android_app->mutex);
   6.113 +            android_app->window = android_app->pendingWindow;
   6.114 +            pthread_cond_broadcast(&android_app->cond);
   6.115 +            pthread_mutex_unlock(&android_app->mutex);
   6.116 +            break;
   6.117 +
   6.118 +        case APP_CMD_TERM_WINDOW:
   6.119 +            LOGV("APP_CMD_TERM_WINDOW\n");
   6.120 +            pthread_cond_broadcast(&android_app->cond);
   6.121 +            break;
   6.122 +
   6.123 +        case APP_CMD_RESUME:
   6.124 +        case APP_CMD_START:
   6.125 +        case APP_CMD_PAUSE:
   6.126 +        case APP_CMD_STOP:
   6.127 +            LOGV("activityState=%d\n", cmd);
   6.128 +            pthread_mutex_lock(&android_app->mutex);
   6.129 +            android_app->activityState = cmd;
   6.130 +            pthread_cond_broadcast(&android_app->cond);
   6.131 +            pthread_mutex_unlock(&android_app->mutex);
   6.132 +            break;
   6.133 +
   6.134 +        case APP_CMD_CONFIG_CHANGED:
   6.135 +            LOGV("APP_CMD_CONFIG_CHANGED\n");
   6.136 +            AConfiguration_fromAssetManager(android_app->config,
   6.137 +                    android_app->activity->assetManager);
   6.138 +            print_cur_config(android_app);
   6.139 +            break;
   6.140 +
   6.141 +        case APP_CMD_DESTROY:
   6.142 +            LOGV("APP_CMD_DESTROY\n");
   6.143 +            android_app->destroyRequested = 1;
   6.144 +            break;
   6.145 +    }
   6.146 +}
   6.147 +
   6.148 +void android_app_post_exec_cmd(struct android_app* android_app, int8_t cmd) {
   6.149 +    switch (cmd) {
   6.150 +        case APP_CMD_TERM_WINDOW:
   6.151 +            LOGV("APP_CMD_TERM_WINDOW\n");
   6.152 +            pthread_mutex_lock(&android_app->mutex);
   6.153 +            android_app->window = NULL;
   6.154 +            pthread_cond_broadcast(&android_app->cond);
   6.155 +            pthread_mutex_unlock(&android_app->mutex);
   6.156 +            break;
   6.157 +
   6.158 +        case APP_CMD_SAVE_STATE:
   6.159 +            LOGV("APP_CMD_SAVE_STATE\n");
   6.160 +            pthread_mutex_lock(&android_app->mutex);
   6.161 +            android_app->stateSaved = 1;
   6.162 +            pthread_cond_broadcast(&android_app->cond);
   6.163 +            pthread_mutex_unlock(&android_app->mutex);
   6.164 +            break;
   6.165 +
   6.166 +        case APP_CMD_RESUME:
   6.167 +            free_saved_state(android_app);
   6.168 +            break;
   6.169 +    }
   6.170 +}
   6.171 +
   6.172 +void app_dummy() {
   6.173 +
   6.174 +}
   6.175 +
   6.176 +static void android_app_destroy(struct android_app* android_app) {
   6.177 +    LOGV("android_app_destroy!");
   6.178 +    free_saved_state(android_app);
   6.179 +    pthread_mutex_lock(&android_app->mutex);
   6.180 +    if (android_app->inputQueue != NULL) {
   6.181 +        AInputQueue_detachLooper(android_app->inputQueue);
   6.182 +    }
   6.183 +    AConfiguration_delete(android_app->config);
   6.184 +    android_app->destroyed = 1;
   6.185 +    pthread_cond_broadcast(&android_app->cond);
   6.186 +    pthread_mutex_unlock(&android_app->mutex);
   6.187 +    // Can't touch android_app object after this.
   6.188 +}
   6.189 +
   6.190 +static void process_input(struct android_app* app, struct android_poll_source* source) {
   6.191 +    AInputEvent* event = NULL;
   6.192 +    while (AInputQueue_getEvent(app->inputQueue, &event) >= 0) {
   6.193 +        LOGV("New input event: type=%d\n", AInputEvent_getType(event));
   6.194 +        if (AInputQueue_preDispatchEvent(app->inputQueue, event)) {
   6.195 +            continue;
   6.196 +        }
   6.197 +        int32_t handled = 0;
   6.198 +        if (app->onInputEvent != NULL) handled = app->onInputEvent(app, event);
   6.199 +        AInputQueue_finishEvent(app->inputQueue, event, handled);
   6.200 +    }
   6.201 +}
   6.202 +
   6.203 +static void process_cmd(struct android_app* app, struct android_poll_source* source) {
   6.204 +    int8_t cmd = android_app_read_cmd(app);
   6.205 +    android_app_pre_exec_cmd(app, cmd);
   6.206 +    if (app->onAppCmd != NULL) app->onAppCmd(app, cmd);
   6.207 +    android_app_post_exec_cmd(app, cmd);
   6.208 +}
   6.209 +
   6.210 +static void* android_app_entry(void* param) {
   6.211 +    struct android_app* android_app = (struct android_app*)param;
   6.212 +
   6.213 +    android_app->config = AConfiguration_new();
   6.214 +    AConfiguration_fromAssetManager(android_app->config, android_app->activity->assetManager);
   6.215 +
   6.216 +    print_cur_config(android_app);
   6.217 +
   6.218 +    android_app->cmdPollSource.id = LOOPER_ID_MAIN;
   6.219 +    android_app->cmdPollSource.app = android_app;
   6.220 +    android_app->cmdPollSource.process = process_cmd;
   6.221 +    android_app->inputPollSource.id = LOOPER_ID_INPUT;
   6.222 +    android_app->inputPollSource.app = android_app;
   6.223 +    android_app->inputPollSource.process = process_input;
   6.224 +
   6.225 +    ALooper* looper = ALooper_prepare(ALOOPER_PREPARE_ALLOW_NON_CALLBACKS);
   6.226 +    ALooper_addFd(looper, android_app->msgread, LOOPER_ID_MAIN, ALOOPER_EVENT_INPUT, NULL,
   6.227 +            &android_app->cmdPollSource);
   6.228 +    android_app->looper = looper;
   6.229 +
   6.230 +    pthread_mutex_lock(&android_app->mutex);
   6.231 +    android_app->running = 1;
   6.232 +    pthread_cond_broadcast(&android_app->cond);
   6.233 +    pthread_mutex_unlock(&android_app->mutex);
   6.234 +
   6.235 +    android_main(android_app);
   6.236 +
   6.237 +    android_app_destroy(android_app);
   6.238 +    return NULL;
   6.239 +}
   6.240 +
   6.241 +// --------------------------------------------------------------------
   6.242 +// Native activity interaction (called from main thread)
   6.243 +// --------------------------------------------------------------------
   6.244 +
   6.245 +static struct android_app* android_app_create(ANativeActivity* activity,
   6.246 +        void* savedState, size_t savedStateSize) {
   6.247 +    struct android_app* android_app = (struct android_app*)malloc(sizeof(struct android_app));
   6.248 +    memset(android_app, 0, sizeof(struct android_app));
   6.249 +    android_app->activity = activity;
   6.250 +
   6.251 +    pthread_mutex_init(&android_app->mutex, NULL);
   6.252 +    pthread_cond_init(&android_app->cond, NULL);
   6.253 +
   6.254 +    if (savedState != NULL) {
   6.255 +        android_app->savedState = malloc(savedStateSize);
   6.256 +        android_app->savedStateSize = savedStateSize;
   6.257 +        memcpy(android_app->savedState, savedState, savedStateSize);
   6.258 +    }
   6.259 +
   6.260 +    int msgpipe[2];
   6.261 +    if (pipe(msgpipe)) {
   6.262 +        LOGE("could not create pipe: %s", strerror(errno));
   6.263 +        return NULL;
   6.264 +    }
   6.265 +    android_app->msgread = msgpipe[0];
   6.266 +    android_app->msgwrite = msgpipe[1];
   6.267 +
   6.268 +    pthread_attr_t attr; 
   6.269 +    pthread_attr_init(&attr);
   6.270 +    pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
   6.271 +    pthread_create(&android_app->thread, &attr, android_app_entry, android_app);
   6.272 +
   6.273 +    // Wait for thread to start.
   6.274 +    pthread_mutex_lock(&android_app->mutex);
   6.275 +    while (!android_app->running) {
   6.276 +        pthread_cond_wait(&android_app->cond, &android_app->mutex);
   6.277 +    }
   6.278 +    pthread_mutex_unlock(&android_app->mutex);
   6.279 +
   6.280 +    return android_app;
   6.281 +}
   6.282 +
   6.283 +static void android_app_write_cmd(struct android_app* android_app, int8_t cmd) {
   6.284 +    if (write(android_app->msgwrite, &cmd, sizeof(cmd)) != sizeof(cmd)) {
   6.285 +        LOGE("Failure writing android_app cmd: %s\n", strerror(errno));
   6.286 +    }
   6.287 +}
   6.288 +
   6.289 +static void android_app_set_input(struct android_app* android_app, AInputQueue* inputQueue) {
   6.290 +    pthread_mutex_lock(&android_app->mutex);
   6.291 +    android_app->pendingInputQueue = inputQueue;
   6.292 +    android_app_write_cmd(android_app, APP_CMD_INPUT_CHANGED);
   6.293 +    while (android_app->inputQueue != android_app->pendingInputQueue) {
   6.294 +        pthread_cond_wait(&android_app->cond, &android_app->mutex);
   6.295 +    }
   6.296 +    pthread_mutex_unlock(&android_app->mutex);
   6.297 +}
   6.298 +
   6.299 +static void android_app_set_window(struct android_app* android_app, ANativeWindow* window) {
   6.300 +    pthread_mutex_lock(&android_app->mutex);
   6.301 +    if (android_app->pendingWindow != NULL) {
   6.302 +        android_app_write_cmd(android_app, APP_CMD_TERM_WINDOW);
   6.303 +    }
   6.304 +    android_app->pendingWindow = window;
   6.305 +    if (window != NULL) {
   6.306 +        android_app_write_cmd(android_app, APP_CMD_INIT_WINDOW);
   6.307 +    }
   6.308 +    while (android_app->window != android_app->pendingWindow) {
   6.309 +        pthread_cond_wait(&android_app->cond, &android_app->mutex);
   6.310 +    }
   6.311 +    pthread_mutex_unlock(&android_app->mutex);
   6.312 +}
   6.313 +
   6.314 +static void android_app_set_activity_state(struct android_app* android_app, int8_t cmd) {
   6.315 +    pthread_mutex_lock(&android_app->mutex);
   6.316 +    android_app_write_cmd(android_app, cmd);
   6.317 +    while (android_app->activityState != cmd) {
   6.318 +        pthread_cond_wait(&android_app->cond, &android_app->mutex);
   6.319 +    }
   6.320 +    pthread_mutex_unlock(&android_app->mutex);
   6.321 +}
   6.322 +
   6.323 +static void android_app_free(struct android_app* android_app) {
   6.324 +    pthread_mutex_lock(&android_app->mutex);
   6.325 +    android_app_write_cmd(android_app, APP_CMD_DESTROY);
   6.326 +    while (!android_app->destroyed) {
   6.327 +        pthread_cond_wait(&android_app->cond, &android_app->mutex);
   6.328 +    }
   6.329 +    pthread_mutex_unlock(&android_app->mutex);
   6.330 +
   6.331 +    close(android_app->msgread);
   6.332 +    close(android_app->msgwrite);
   6.333 +    pthread_cond_destroy(&android_app->cond);
   6.334 +    pthread_mutex_destroy(&android_app->mutex);
   6.335 +    free(android_app);
   6.336 +}
   6.337 +
   6.338 +static void onDestroy(ANativeActivity* activity) {
   6.339 +    LOGV("Destroy: %p\n", activity);
   6.340 +    android_app_free((struct android_app*)activity->instance);
   6.341 +}
   6.342 +
   6.343 +static void onStart(ANativeActivity* activity) {
   6.344 +    LOGV("Start: %p\n", activity);
   6.345 +    android_app_set_activity_state((struct android_app*)activity->instance, APP_CMD_START);
   6.346 +}
   6.347 +
   6.348 +static void onResume(ANativeActivity* activity) {
   6.349 +    LOGV("Resume: %p\n", activity);
   6.350 +    android_app_set_activity_state((struct android_app*)activity->instance, APP_CMD_RESUME);
   6.351 +}
   6.352 +
   6.353 +static void* onSaveInstanceState(ANativeActivity* activity, size_t* outLen) {
   6.354 +    struct android_app* android_app = (struct android_app*)activity->instance;
   6.355 +    void* savedState = NULL;
   6.356 +
   6.357 +    LOGV("SaveInstanceState: %p\n", activity);
   6.358 +    pthread_mutex_lock(&android_app->mutex);
   6.359 +    android_app->stateSaved = 0;
   6.360 +    android_app_write_cmd(android_app, APP_CMD_SAVE_STATE);
   6.361 +    while (!android_app->stateSaved) {
   6.362 +        pthread_cond_wait(&android_app->cond, &android_app->mutex);
   6.363 +    }
   6.364 +
   6.365 +    if (android_app->savedState != NULL) {
   6.366 +        savedState = android_app->savedState;
   6.367 +        *outLen = android_app->savedStateSize;
   6.368 +        android_app->savedState = NULL;
   6.369 +        android_app->savedStateSize = 0;
   6.370 +    }
   6.371 +
   6.372 +    pthread_mutex_unlock(&android_app->mutex);
   6.373 +
   6.374 +    return savedState;
   6.375 +}
   6.376 +
   6.377 +static void onPause(ANativeActivity* activity) {
   6.378 +    LOGV("Pause: %p\n", activity);
   6.379 +    android_app_set_activity_state((struct android_app*)activity->instance, APP_CMD_PAUSE);
   6.380 +}
   6.381 +
   6.382 +static void onStop(ANativeActivity* activity) {
   6.383 +    LOGV("Stop: %p\n", activity);
   6.384 +    android_app_set_activity_state((struct android_app*)activity->instance, APP_CMD_STOP);
   6.385 +}
   6.386 +
   6.387 +static void onConfigurationChanged(ANativeActivity* activity) {
   6.388 +    struct android_app* android_app = (struct android_app*)activity->instance;
   6.389 +    LOGV("ConfigurationChanged: %p\n", activity);
   6.390 +    android_app_write_cmd(android_app, APP_CMD_CONFIG_CHANGED);
   6.391 +}
   6.392 +
   6.393 +static void onLowMemory(ANativeActivity* activity) {
   6.394 +    struct android_app* android_app = (struct android_app*)activity->instance;
   6.395 +    LOGV("LowMemory: %p\n", activity);
   6.396 +    android_app_write_cmd(android_app, APP_CMD_LOW_MEMORY);
   6.397 +}
   6.398 +
   6.399 +static void onWindowFocusChanged(ANativeActivity* activity, int focused) {
   6.400 +    LOGV("WindowFocusChanged: %p -- %d\n", activity, focused);
   6.401 +    android_app_write_cmd((struct android_app*)activity->instance,
   6.402 +            focused ? APP_CMD_GAINED_FOCUS : APP_CMD_LOST_FOCUS);
   6.403 +}
   6.404 +
   6.405 +static void onNativeWindowCreated(ANativeActivity* activity, ANativeWindow* window) {
   6.406 +    LOGV("NativeWindowCreated: %p -- %p\n", activity, window);
   6.407 +    android_app_set_window((struct android_app*)activity->instance, window);
   6.408 +}
   6.409 +
   6.410 +static void onNativeWindowDestroyed(ANativeActivity* activity, ANativeWindow* window) {
   6.411 +    LOGV("NativeWindowDestroyed: %p -- %p\n", activity, window);
   6.412 +    android_app_set_window((struct android_app*)activity->instance, NULL);
   6.413 +}
   6.414 +
   6.415 +static void onInputQueueCreated(ANativeActivity* activity, AInputQueue* queue) {
   6.416 +    LOGV("InputQueueCreated: %p -- %p\n", activity, queue);
   6.417 +    android_app_set_input((struct android_app*)activity->instance, queue);
   6.418 +}
   6.419 +
   6.420 +static void onInputQueueDestroyed(ANativeActivity* activity, AInputQueue* queue) {
   6.421 +    LOGV("InputQueueDestroyed: %p -- %p\n", activity, queue);
   6.422 +    android_app_set_input((struct android_app*)activity->instance, NULL);
   6.423 +}
   6.424 +
   6.425 +void ANativeActivity_onCreate(ANativeActivity* activity,
   6.426 +        void* savedState, size_t savedStateSize) {
   6.427 +    LOGV("Creating: %p\n", activity);
   6.428 +    activity->callbacks->onDestroy = onDestroy;
   6.429 +    activity->callbacks->onStart = onStart;
   6.430 +    activity->callbacks->onResume = onResume;
   6.431 +    activity->callbacks->onSaveInstanceState = onSaveInstanceState;
   6.432 +    activity->callbacks->onPause = onPause;
   6.433 +    activity->callbacks->onStop = onStop;
   6.434 +    activity->callbacks->onConfigurationChanged = onConfigurationChanged;
   6.435 +    activity->callbacks->onLowMemory = onLowMemory;
   6.436 +    activity->callbacks->onWindowFocusChanged = onWindowFocusChanged;
   6.437 +    activity->callbacks->onNativeWindowCreated = onNativeWindowCreated;
   6.438 +    activity->callbacks->onNativeWindowDestroyed = onNativeWindowDestroyed;
   6.439 +    activity->callbacks->onInputQueueCreated = onInputQueueCreated;
   6.440 +    activity->callbacks->onInputQueueDestroyed = onInputQueueDestroyed;
   6.441 +
   6.442 +    activity->instance = android_app_create(activity, savedState, savedStateSize);
   6.443 +}
     7.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     7.2 +++ b/src/android/android_native_app_glue.h	Sat Oct 03 06:10:30 2015 +0300
     7.3 @@ -0,0 +1,349 @@
     7.4 +/*
     7.5 + * Copyright (C) 2010 The Android Open Source Project
     7.6 + *
     7.7 + * Licensed under the Apache License, Version 2.0 (the "License");
     7.8 + * you may not use this file except in compliance with the License.
     7.9 + * You may obtain a copy of the License at
    7.10 + *
    7.11 + *      http://www.apache.org/licenses/LICENSE-2.0
    7.12 + *
    7.13 + * Unless required by applicable law or agreed to in writing, software
    7.14 + * distributed under the License is distributed on an "AS IS" BASIS,
    7.15 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    7.16 + * See the License for the specific language governing permissions and
    7.17 + * limitations under the License.
    7.18 + *
    7.19 + */
    7.20 +
    7.21 +#ifndef _ANDROID_NATIVE_APP_GLUE_H
    7.22 +#define _ANDROID_NATIVE_APP_GLUE_H
    7.23 +
    7.24 +#include <poll.h>
    7.25 +#include <pthread.h>
    7.26 +#include <sched.h>
    7.27 +
    7.28 +#include <android/configuration.h>
    7.29 +#include <android/looper.h>
    7.30 +#include <android/native_activity.h>
    7.31 +
    7.32 +#ifdef __cplusplus
    7.33 +extern "C" {
    7.34 +#endif
    7.35 +
    7.36 +/**
    7.37 + * The native activity interface provided by <android/native_activity.h>
    7.38 + * is based on a set of application-provided callbacks that will be called
    7.39 + * by the Activity's main thread when certain events occur.
    7.40 + *
    7.41 + * This means that each one of this callbacks _should_ _not_ block, or they
    7.42 + * risk having the system force-close the application. This programming
    7.43 + * model is direct, lightweight, but constraining.
    7.44 + *
    7.45 + * The 'android_native_app_glue' static library is used to provide a different
    7.46 + * execution model where the application can implement its own main event
    7.47 + * loop in a different thread instead. Here's how it works:
    7.48 + *
    7.49 + * 1/ The application must provide a function named "android_main()" that
    7.50 + *    will be called when the activity is created, in a new thread that is
    7.51 + *    distinct from the activity's main thread.
    7.52 + *
    7.53 + * 2/ android_main() receives a pointer to a valid "android_app" structure
    7.54 + *    that contains references to other important objects, e.g. the
    7.55 + *    ANativeActivity obejct instance the application is running in.
    7.56 + *
    7.57 + * 3/ the "android_app" object holds an ALooper instance that already
    7.58 + *    listens to two important things:
    7.59 + *
    7.60 + *      - activity lifecycle events (e.g. "pause", "resume"). See APP_CMD_XXX
    7.61 + *        declarations below.
    7.62 + *
    7.63 + *      - input events coming from the AInputQueue attached to the activity.
    7.64 + *
    7.65 + *    Each of these correspond to an ALooper identifier returned by
    7.66 + *    ALooper_pollOnce with values of LOOPER_ID_MAIN and LOOPER_ID_INPUT,
    7.67 + *    respectively.
    7.68 + *
    7.69 + *    Your application can use the same ALooper to listen to additional
    7.70 + *    file-descriptors.  They can either be callback based, or with return
    7.71 + *    identifiers starting with LOOPER_ID_USER.
    7.72 + *
    7.73 + * 4/ Whenever you receive a LOOPER_ID_MAIN or LOOPER_ID_INPUT event,
    7.74 + *    the returned data will point to an android_poll_source structure.  You
    7.75 + *    can call the process() function on it, and fill in android_app->onAppCmd
    7.76 + *    and android_app->onInputEvent to be called for your own processing
    7.77 + *    of the event.
    7.78 + *
    7.79 + *    Alternatively, you can call the low-level functions to read and process
    7.80 + *    the data directly...  look at the process_cmd() and process_input()
    7.81 + *    implementations in the glue to see how to do this.
    7.82 + *
    7.83 + * See the sample named "native-activity" that comes with the NDK with a
    7.84 + * full usage example.  Also look at the JavaDoc of NativeActivity.
    7.85 + */
    7.86 +
    7.87 +struct android_app;
    7.88 +
    7.89 +/**
    7.90 + * Data associated with an ALooper fd that will be returned as the "outData"
    7.91 + * when that source has data ready.
    7.92 + */
    7.93 +struct android_poll_source {
    7.94 +    // The identifier of this source.  May be LOOPER_ID_MAIN or
    7.95 +    // LOOPER_ID_INPUT.
    7.96 +    int32_t id;
    7.97 +
    7.98 +    // The android_app this ident is associated with.
    7.99 +    struct android_app* app;
   7.100 +
   7.101 +    // Function to call to perform the standard processing of data from
   7.102 +    // this source.
   7.103 +    void (*process)(struct android_app* app, struct android_poll_source* source);
   7.104 +};
   7.105 +
   7.106 +/**
   7.107 + * This is the interface for the standard glue code of a threaded
   7.108 + * application.  In this model, the application's code is running
   7.109 + * in its own thread separate from the main thread of the process.
   7.110 + * It is not required that this thread be associated with the Java
   7.111 + * VM, although it will need to be in order to make JNI calls any
   7.112 + * Java objects.
   7.113 + */
   7.114 +struct android_app {
   7.115 +    // The application can place a pointer to its own state object
   7.116 +    // here if it likes.
   7.117 +    void* userData;
   7.118 +
   7.119 +    // Fill this in with the function to process main app commands (APP_CMD_*)
   7.120 +    void (*onAppCmd)(struct android_app* app, int32_t cmd);
   7.121 +
   7.122 +    // Fill this in with the function to process input events.  At this point
   7.123 +    // the event has already been pre-dispatched, and it will be finished upon
   7.124 +    // return.  Return 1 if you have handled the event, 0 for any default
   7.125 +    // dispatching.
   7.126 +    int32_t (*onInputEvent)(struct android_app* app, AInputEvent* event);
   7.127 +
   7.128 +    // The ANativeActivity object instance that this app is running in.
   7.129 +    ANativeActivity* activity;
   7.130 +
   7.131 +    // The current configuration the app is running in.
   7.132 +    AConfiguration* config;
   7.133 +
   7.134 +    // This is the last instance's saved state, as provided at creation time.
   7.135 +    // It is NULL if there was no state.  You can use this as you need; the
   7.136 +    // memory will remain around until you call android_app_exec_cmd() for
   7.137 +    // APP_CMD_RESUME, at which point it will be freed and savedState set to NULL.
   7.138 +    // These variables should only be changed when processing a APP_CMD_SAVE_STATE,
   7.139 +    // at which point they will be initialized to NULL and you can malloc your
   7.140 +    // state and place the information here.  In that case the memory will be
   7.141 +    // freed for you later.
   7.142 +    void* savedState;
   7.143 +    size_t savedStateSize;
   7.144 +
   7.145 +    // The ALooper associated with the app's thread.
   7.146 +    ALooper* looper;
   7.147 +
   7.148 +    // When non-NULL, this is the input queue from which the app will
   7.149 +    // receive user input events.
   7.150 +    AInputQueue* inputQueue;
   7.151 +
   7.152 +    // When non-NULL, this is the window surface that the app can draw in.
   7.153 +    ANativeWindow* window;
   7.154 +
   7.155 +    // Current content rectangle of the window; this is the area where the
   7.156 +    // window's content should be placed to be seen by the user.
   7.157 +    ARect contentRect;
   7.158 +
   7.159 +    // Current state of the app's activity.  May be either APP_CMD_START,
   7.160 +    // APP_CMD_RESUME, APP_CMD_PAUSE, or APP_CMD_STOP; see below.
   7.161 +    int activityState;
   7.162 +
   7.163 +    // This is non-zero when the application's NativeActivity is being
   7.164 +    // destroyed and waiting for the app thread to complete.
   7.165 +    int destroyRequested;
   7.166 +
   7.167 +    // -------------------------------------------------
   7.168 +    // Below are "private" implementation of the glue code.
   7.169 +
   7.170 +    pthread_mutex_t mutex;
   7.171 +    pthread_cond_t cond;
   7.172 +
   7.173 +    int msgread;
   7.174 +    int msgwrite;
   7.175 +
   7.176 +    pthread_t thread;
   7.177 +
   7.178 +    struct android_poll_source cmdPollSource;
   7.179 +    struct android_poll_source inputPollSource;
   7.180 +
   7.181 +    int running;
   7.182 +    int stateSaved;
   7.183 +    int destroyed;
   7.184 +    int redrawNeeded;
   7.185 +    AInputQueue* pendingInputQueue;
   7.186 +    ANativeWindow* pendingWindow;
   7.187 +    ARect pendingContentRect;
   7.188 +};
   7.189 +
   7.190 +enum {
   7.191 +    /**
   7.192 +     * Looper data ID of commands coming from the app's main thread, which
   7.193 +     * is returned as an identifier from ALooper_pollOnce().  The data for this
   7.194 +     * identifier is a pointer to an android_poll_source structure.
   7.195 +     * These can be retrieved and processed with android_app_read_cmd()
   7.196 +     * and android_app_exec_cmd().
   7.197 +     */
   7.198 +    LOOPER_ID_MAIN = 1,
   7.199 +
   7.200 +    /**
   7.201 +     * Looper data ID of events coming from the AInputQueue of the
   7.202 +     * application's window, which is returned as an identifier from
   7.203 +     * ALooper_pollOnce().  The data for this identifier is a pointer to an
   7.204 +     * android_poll_source structure.  These can be read via the inputQueue
   7.205 +     * object of android_app.
   7.206 +     */
   7.207 +    LOOPER_ID_INPUT = 2,
   7.208 +
   7.209 +    /**
   7.210 +     * Start of user-defined ALooper identifiers.
   7.211 +     */
   7.212 +    LOOPER_ID_USER = 3,
   7.213 +};
   7.214 +
   7.215 +enum {
   7.216 +    /**
   7.217 +     * Command from main thread: the AInputQueue has changed.  Upon processing
   7.218 +     * this command, android_app->inputQueue will be updated to the new queue
   7.219 +     * (or NULL).
   7.220 +     */
   7.221 +    APP_CMD_INPUT_CHANGED,
   7.222 +
   7.223 +    /**
   7.224 +     * Command from main thread: a new ANativeWindow is ready for use.  Upon
   7.225 +     * receiving this command, android_app->window will contain the new window
   7.226 +     * surface.
   7.227 +     */
   7.228 +    APP_CMD_INIT_WINDOW,
   7.229 +
   7.230 +    /**
   7.231 +     * Command from main thread: the existing ANativeWindow needs to be
   7.232 +     * terminated.  Upon receiving this command, android_app->window still
   7.233 +     * contains the existing window; after calling android_app_exec_cmd
   7.234 +     * it will be set to NULL.
   7.235 +     */
   7.236 +    APP_CMD_TERM_WINDOW,
   7.237 +
   7.238 +    /**
   7.239 +     * Command from main thread: the current ANativeWindow has been resized.
   7.240 +     * Please redraw with its new size.
   7.241 +     */
   7.242 +    APP_CMD_WINDOW_RESIZED,
   7.243 +
   7.244 +    /**
   7.245 +     * Command from main thread: the system needs that the current ANativeWindow
   7.246 +     * be redrawn.  You should redraw the window before handing this to
   7.247 +     * android_app_exec_cmd() in order to avoid transient drawing glitches.
   7.248 +     */
   7.249 +    APP_CMD_WINDOW_REDRAW_NEEDED,
   7.250 +
   7.251 +    /**
   7.252 +     * Command from main thread: the content area of the window has changed,
   7.253 +     * such as from the soft input window being shown or hidden.  You can
   7.254 +     * find the new content rect in android_app::contentRect.
   7.255 +     */
   7.256 +    APP_CMD_CONTENT_RECT_CHANGED,
   7.257 +
   7.258 +    /**
   7.259 +     * Command from main thread: the app's activity window has gained
   7.260 +     * input focus.
   7.261 +     */
   7.262 +    APP_CMD_GAINED_FOCUS,
   7.263 +
   7.264 +    /**
   7.265 +     * Command from main thread: the app's activity window has lost
   7.266 +     * input focus.
   7.267 +     */
   7.268 +    APP_CMD_LOST_FOCUS,
   7.269 +
   7.270 +    /**
   7.271 +     * Command from main thread: the current device configuration has changed.
   7.272 +     */
   7.273 +    APP_CMD_CONFIG_CHANGED,
   7.274 +
   7.275 +    /**
   7.276 +     * Command from main thread: the system is running low on memory.
   7.277 +     * Try to reduce your memory use.
   7.278 +     */
   7.279 +    APP_CMD_LOW_MEMORY,
   7.280 +
   7.281 +    /**
   7.282 +     * Command from main thread: the app's activity has been started.
   7.283 +     */
   7.284 +    APP_CMD_START,
   7.285 +
   7.286 +    /**
   7.287 +     * Command from main thread: the app's activity has been resumed.
   7.288 +     */
   7.289 +    APP_CMD_RESUME,
   7.290 +
   7.291 +    /**
   7.292 +     * Command from main thread: the app should generate a new saved state
   7.293 +     * for itself, to restore from later if needed.  If you have saved state,
   7.294 +     * allocate it with malloc and place it in android_app.savedState with
   7.295 +     * the size in android_app.savedStateSize.  The will be freed for you
   7.296 +     * later.
   7.297 +     */
   7.298 +    APP_CMD_SAVE_STATE,
   7.299 +
   7.300 +    /**
   7.301 +     * Command from main thread: the app's activity has been paused.
   7.302 +     */
   7.303 +    APP_CMD_PAUSE,
   7.304 +
   7.305 +    /**
   7.306 +     * Command from main thread: the app's activity has been stopped.
   7.307 +     */
   7.308 +    APP_CMD_STOP,
   7.309 +
   7.310 +    /**
   7.311 +     * Command from main thread: the app's activity is being destroyed,
   7.312 +     * and waiting for the app thread to clean up and exit before proceeding.
   7.313 +     */
   7.314 +    APP_CMD_DESTROY,
   7.315 +};
   7.316 +
   7.317 +/**
   7.318 + * Call when ALooper_pollAll() returns LOOPER_ID_MAIN, reading the next
   7.319 + * app command message.
   7.320 + */
   7.321 +int8_t android_app_read_cmd(struct android_app* android_app);
   7.322 +
   7.323 +/**
   7.324 + * Call with the command returned by android_app_read_cmd() to do the
   7.325 + * initial pre-processing of the given command.  You can perform your own
   7.326 + * actions for the command after calling this function.
   7.327 + */
   7.328 +void android_app_pre_exec_cmd(struct android_app* android_app, int8_t cmd);
   7.329 +
   7.330 +/**
   7.331 + * Call with the command returned by android_app_read_cmd() to do the
   7.332 + * final post-processing of the given command.  You must have done your own
   7.333 + * actions for the command before calling this function.
   7.334 + */
   7.335 +void android_app_post_exec_cmd(struct android_app* android_app, int8_t cmd);
   7.336 +
   7.337 +/**
   7.338 + * Dummy function you can call to ensure glue code isn't stripped.
   7.339 + */
   7.340 +void app_dummy();
   7.341 +
   7.342 +/**
   7.343 + * This is the function that application code must implement, representing
   7.344 + * the main entry to the app.
   7.345 + */
   7.346 +extern void android_main(struct android_app* app);
   7.347 +
   7.348 +#ifdef __cplusplus
   7.349 +}
   7.350 +#endif
   7.351 +
   7.352 +#endif /* _ANDROID_NATIVE_APP_GLUE_H */
     8.1 --- a/src/android/assman.c	Fri Oct 02 07:10:19 2015 +0300
     8.2 +++ b/src/android/assman.c	Sat Oct 03 06:10:30 2015 +0300
     8.3 @@ -1,8 +1,9 @@
     8.4  #include <fcntl.h>
     8.5  #include <android/asset_manager.h>
     8.6  #include "assman.h"
     8.7 -#include "native_glue.h"
     8.8 -#include "amain.h"
     8.9 +#include "android_native_app_glue.h"
    8.10 +
    8.11 +struct android_app *app;	/* defined in android/amain.c */
    8.12  
    8.13  ass_file *ass_fopen(const char *fname, const char *mode)
    8.14  {
     9.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     9.2 +++ b/src/android/logger.c	Sat Oct 03 06:10:30 2015 +0300
     9.3 @@ -0,0 +1,60 @@
     9.4 +#include <stdio.h>
     9.5 +#include <assert.h>
     9.6 +#include <unistd.h>
     9.7 +#include <pthread.h>
     9.8 +#include <android/log.h>
     9.9 +#include "logger.h"
    9.10 +
    9.11 +static void *thread_func(void *arg);
    9.12 +
    9.13 +static int pfd[2];
    9.14 +static pthread_t thr;
    9.15 +static int initialized;
    9.16 +
    9.17 +int start_logger(void)
    9.18 +{
    9.19 +	if(initialized) {
    9.20 +		return 1;
    9.21 +	}
    9.22 +
    9.23 +	/* set stdout to line-buffered, and stderr to unbuffered */
    9.24 +	setvbuf(stdout, 0, _IOLBF, 0);
    9.25 +	setvbuf(stderr, 0, _IONBF, 0);
    9.26 +
    9.27 +	if(pipe(pfd) == -1) {
    9.28 +		perror("failed to create logging pipe");
    9.29 +		return -1;
    9.30 +	}
    9.31 +	assert(pfd[0] > 2 && pfd[1] > 2);
    9.32 +
    9.33 +	/* redirect stdout & stderr to the write-end of the pipe */
    9.34 +	dup2(pfd[1], 1);
    9.35 +	dup2(pfd[1], 2);
    9.36 +
    9.37 +	/* start the logging thread */
    9.38 +	if(pthread_create(&thr, 0, thread_func, 0) == -1) {
    9.39 +		perror("failed to spawn logging thread");
    9.40 +		return -1;
    9.41 +	}
    9.42 +	pthread_detach(thr);
    9.43 +	return 0;
    9.44 +}
    9.45 +
    9.46 +static void *thread_func(void *arg)
    9.47 +{
    9.48 +	ssize_t rdsz;
    9.49 +	char buf[257];
    9.50 +
    9.51 +	__android_log_print(ANDROID_LOG_DEBUG, APP_NAME, "logger starting up...");
    9.52 +
    9.53 +	while((rdsz = read(pfd[0], buf, sizeof buf - 1)) > 0) {
    9.54 +		if(buf[rdsz - 1] == '\n') {
    9.55 +			--rdsz;
    9.56 +		}
    9.57 +		buf[rdsz] = 0;
    9.58 +		__android_log_write(ANDROID_LOG_DEBUG, APP_NAME, buf);
    9.59 +	}
    9.60 +
    9.61 +	__android_log_print(ANDROID_LOG_DEBUG, APP_NAME, "logger shutting down...");
    9.62 +	return 0;
    9.63 +}
    10.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    10.2 +++ b/src/android/logger.h	Sat Oct 03 06:10:30 2015 +0300
    10.3 @@ -0,0 +1,6 @@
    10.4 +#ifndef LOGGER_H_
    10.5 +#define LOGGER_H_
    10.6 +
    10.7 +int start_logger(void);
    10.8 +
    10.9 +#endif	/* LOGGER_H_ */
    11.1 --- a/src/opengl.h	Fri Oct 02 07:10:19 2015 +0300
    11.2 +++ b/src/opengl.h	Sat Oct 03 06:10:30 2015 +0300
    11.3 @@ -1,6 +1,6 @@
    11.4  /*
    11.5  Stereoscopic tunnel for iOS.
    11.6 -Copyright (C) 2011  John Tsiombikas <nuclear@member.fsf.org>
    11.7 +Copyright (C) 2011-2015 John Tsiombikas <nuclear@member.fsf.org>
    11.8  
    11.9  This program is free software: you can redistribute it and/or modify
   11.10  it under the terms of the GNU General Public License as published by
   11.11 @@ -28,6 +28,13 @@
   11.12  #define glClearDepth	glClearDepthf
   11.13  #define GLDEF
   11.14  #include "sanegl.h"
   11.15 +
   11.16 +#elif defined(ANDROID)
   11.17 +#include <GLES2/gl2.h>
   11.18 +#include <GLES2/gl2ext.h>
   11.19 +#define GLDEF
   11.20 +#include "sanegl.h"
   11.21 +
   11.22  #else
   11.23  
   11.24  #include <GL/glew.h>