Compare commits

...

320 Commits

Author SHA1 Message Date
prideout
a82d93914e Bump to v1.4.2 2019-10-30 13:49:36 -07:00
Philip Rideout
389349198f gltfio-jni: Fix double-init panic caused by build issues.
Filament keeps a global static list of engines for verification purposes
which was being recreated multiple times, causing a panic. After a lot
of experimentation with the build, I was not able to fix this while
keeping the current scheme of dynamically loading both libraries.

As a "solution", we now staticly link filament into gltfio-jni, and ask
clients to load only that.

Fixes #1811.
2019-10-30 09:51:14 -07:00
Philip Rideout
225a844e0f Minor improvements to Engine validation.
This enhances the assert added in 2548da9 in the following ways:

- Adds console output at startup to help detect "double load" bugs.
- Adds PRETTY_FUNCTION to show where the invalid usage is coming from.
- Tweaks the wording since use-after-destroy is not always the cause.
- Consolidates usage of static globals into a helper class.

Motivated by #1811.
2019-10-29 15:48:29 -07:00
Mathias Agopian
f018c719a6 a test for the skinning math 2019-10-29 15:38:40 -07:00
Philip Rideout
2a229571ad Minor fixes to gltf-bloom demo. 2019-10-29 10:20:28 -07:00
Romain Guy
35959b547c Use unnormalized vectors to compute bump mapping. (#1813)
If we assume the tangents and bitangents were generated using mikktspace,
we should use the unnormalized vectors, as indicated at mikktspace.com
or in http://www.metalliandy.com/mikktspace/downloads/mikktspace/mikktspace.h
(see last comment at the end of the file).
2019-10-29 09:41:15 +00:00
Philip Rideout
796007377d Revert the Engine validation commit.
This needs to wait until code review and was pushed by accident.

This reverts commit bb90103e74.
2019-10-28 16:14:07 -07:00
Philip Rideout
5c73d0ae75 Update package.json for npm release. 2019-10-28 16:05:45 -07:00
Philip Rideout
bb90103e74 Minor improvements to Engine validation.
This enhances the assert added in 2548da9 in the following ways:

- Adds console output at startup to help detect "double load" bugs.
- Adds PRETTY_FUNCTION to show where the invalid usage is coming from.
- Tweak the wording since use-after-destroy is not always the cause.

Motivated by #1811.
2019-10-28 14:08:54 -07:00
Ben Doherty
0149bc504d Fix Windows GitHub Action (#1814) 2019-10-28 12:22:42 -07:00
Philip Rideout
3db035520b gltf-bloom now invokes the gltfio animator.
Tested with CesiumMan.
2019-10-28 09:32:39 -07:00
Philip Rideout
618b59932c gltfio: Add Java / Kotlin bindings for Animator.
Fixes #1529.
2019-10-28 09:32:39 -07:00
Mathias Agopian
9ef05ad456 better fix for NoopDriver on iOS 2019-10-24 14:17:29 -07:00
Adrian Perez
c8a0dbb59a Fix OpenGL ES 3.0 support on iOS (#1803) 2019-10-24 12:43:23 -07:00
dsternfeld7
35a0b045f8 Adds missing include in NoopDriver that prevented the correct shader model from being picked when using the Noop backend. 2019-10-24 10:58:16 -07:00
Ben Doherty
a4369c833d Add README section on Filament CMake options (#1801) 2019-10-23 13:27:08 -07:00
Philip Rideout
354f1da708 Enhance KHR_debug output + code cleanup. 2019-10-22 12:41:34 -07:00
Philip Rideout
7e47d257d1 OpenGL Driver: add support for KHR_debug.
This extension dumps out some interesting vendor-specific information
to the log, which could be useful for debugging. Interestingly, some
of Android demos produce this output on Pixel 3:

  KHR_debug: EsxBufferObject::Map - Ignoring EsxBufferMapUnsyncedBit

Enabled for debug builds only.
2019-10-22 12:41:34 -07:00
Ben Doherty
c3d2b3578d Add Javadoc for Colors, Box, and MathUtils (#1796) 2019-10-21 17:59:05 -07:00
tpsiaki
783aa06a46 Fix typo in Noop on mobile. (#1799) 2019-10-21 17:41:32 -07:00
Mathias Agopian
faa82ff35a implement diag() in TMatSquareFunctions
this requires using 'auto' as the return type.
2019-10-21 14:27:40 -07:00
Ben Doherty
3008123d6b Update RELEASE_NOTES for v1.4.1 (#1793) 2019-10-18 13:36:34 -07:00
Ben Doherty
2d2987066d Implement user-controlled frame presentation in Metal (#1760) 2019-10-17 17:11:34 -07:00
Mathias Agopian
c84ec04efa remove unneeded loop in TransformManager
during a transaction, it makes no sense iterating through all
children just so that we can iterate through their children...
and do nothing else.
2019-10-17 16:17:32 -07:00
Mathias Agopian
197b45c1d1 fix a double-free when exiting the FroxelData test
We were destroying the engine manually (by calling delete) which is
not allowed.

Also made FEngine::shutdown() private.
2019-10-17 16:17:12 -07:00
Mathias Agopian
af9ecf0a61 minor cleanup based on clion suggestions 2019-10-17 16:16:51 -07:00
Ben Doherty
530fb58337 Add Javadoc to IndexBuffer and VertexBuffer (#1788) 2019-10-16 17:59:53 -07:00
Philip Rideout
6a891021b1 MSVC fixes for Vulkan.
Should fix #1772, hopefully.
2019-10-16 17:37:33 -07:00
Ben Doherty
0e7c68b50d Fix, assertion using bones with Metal backend (#1787) 2019-10-16 15:54:08 -07:00
Ben Doherty
32a32d3d21 Remove iOS OpenGL ES 3.1 hack (#1781) 2019-10-16 10:51:55 -07:00
Ben Doherty
fca71813d4 Use CMAKE_COMMAND variable in place of cmake (#1783) 2019-10-16 10:50:55 -07:00
Philip Rideout
3618f14531 Add Android sample for Camera Stream.
This simple Kotlin app projects a live feed on the side of a cube.
It looks very similar to the `lit-cube` demo but there is a new Kotlin
class called `CameraHelper` that creates Camera2 objects, starts
the capture session, creates the Filament texture, etc.

For now, this app uses SurfaceTexture. This is a starting point and
provides a testing ground for tentative improvements to our Stream API.
2019-10-16 09:05:24 -07:00
Romain Guy
9de3e76a53 Remove the use of reflection where it's not necessary (#1782)
Fixes #1510
2019-10-15 17:54:02 -07:00
Romain Guy
e472fdbce4 Remove Kokoro Windows presubmit 2019-10-15 17:40:24 -07:00
Mathias Agopian
c9fcedb40b some optimizations to the math code (#1779)
- we use by-value parameters in some places, which helps the compiler
  (not sure why, but assuming references can be aliased and the
   compiler is cautious about that).
  This helps matrix multiplies (which is almost never inlined) and
  transpose (but that's almost always inlined and merged with
  something else, so the gain here is not completely real)

- remove support for matrices-of-matrices because this wasn't complete
  anyways.

- improve matrix * scalar by inlining the loop manually instead of
  going through *=, this saves some extra writes.
2019-10-15 17:22:54 -07:00
Mathias Agopian
bd8c3e9181 don't limit the near plane to ~1mm (#1778)
We were not checking Camera's preconditions properly -- we assumed that
near < 1/1024 meant "near == 0".

Now we check preconditions properly and revert to the same hardcoded
default if they're not met -- doesn't matter what the default is,
it's there only to avoid NaNs and Infs elsewhere -- rendering will
be bogus eitherway.

Fixes #1766
2019-10-15 17:22:38 -07:00
Gregory Popovitch
26d0aa76f8 README.md: add build steps for Visual Studio 2019 / MSVC (#1775) 2019-10-15 12:02:08 -07:00
Ben Doherty
20f4ec6576 Use platform-specific temporary directories for CircularBuffer (#1769) 2019-10-15 12:01:03 -07:00
Ben Doherty
d0bc23473c Fix crash when using Filamat lite library (#1780) 2019-10-15 10:49:59 -07:00
Ben Doherty
c33f6adf62 Remove BUILD_DEMO CMake option (#1776) 2019-10-14 09:09:46 -07:00
Ben Doherty
d4efa105b7 Set up Windows build through GitHub actions (#1767) 2019-10-11 09:28:57 -07:00
Ben Doherty
e8641ce257 Additional fixes for MSVC (#1765) 2019-10-10 16:45:06 -07:00
Ben Doherty
d7a4bdc3f1 Combine matdbg dependencies into single shared library (#1764) 2019-10-10 12:45:51 -07:00
Philip Rideout
e66215295c gltfio: use the new sparse support in cgltf. 2019-10-09 18:45:27 -07:00
Philip Rideout
552bb3e6db Update cgltf with sparse functionality. 2019-10-09 18:45:27 -07:00
Mathias Agopian
a96da05ab8 fix CircularBuffer ashmem fallback (#1763) 2019-10-09 16:48:22 -07:00
Ben Doherty
fb5ee4a9d0 Add Javadoc comments to Material and MaterialInstance (#1696) 2019-10-08 15:56:26 -07:00
Philip Rideout
da13067b57 libmath: prevent NaN's in slerp due to acos.
Fixes #1749.
2019-10-08 15:37:48 -07:00
Ben Doherty
d48d69c765 Use custom Git checkout action (#1756) 2019-10-08 15:22:09 -07:00
Mathias Agopian
5098b2af77 SwapChain javadoc 2019-10-08 10:31:32 -07:00
Mathias Agopian
5576d6b014 TextureSampler javadoc 2019-10-08 10:31:14 -07:00
Mathias Agopian
b072399229 TransformManager javadoc 2019-10-08 10:30:54 -07:00
Mathias Agopian
4f0ab43d58 Stream Javadoc 2019-10-08 10:27:53 -07:00
Philip Rideout
3f742880b5 gltfio: do not abort for sparse data, warn instead.
Sparse accessors are easy to punt gracefully, we can emit a warning
and instead use the dense "base" data.

Also, after cgltf support for this lands, we will need to apply sparse
data in ResourceLoader, so this adds a bit of plumbing to prep for that.

Relates to #1727.
2019-10-08 08:18:46 -07:00
Mathias Agopian
0704194460 javadoc for Renderer
also made resetUserTime and getUserTime public
2019-10-07 16:13:56 -07:00
Mathias Agopian
56e747f558 Scene javadoc 2019-10-07 16:13:38 -07:00
Ben Doherty
61c22f2295 Fix culling in MaterialInstance (#1750) 2019-10-07 16:08:50 -07:00
Mathias Agopian
c4bbaeda86 improve CircularBuffer error message 2019-10-04 13:18:09 -07:00
Philip Rideout
678ffc7503 gltfio: do not loop animation samplers, fixes #1730. 2019-10-04 08:48:35 -07:00
Philip Rideout
a080a47268 Add RenderTarget javadoc. 2019-10-04 08:47:55 -07:00
Philip Rideout
82e8104539 Fix javadoc warnings. 2019-10-04 08:47:55 -07:00
Mathias Agopian
11bd8e72ef harden lispsm code against div-by-zero 2019-10-03 22:58:26 -07:00
Mathias Agopian
03c2a90206 noexcept-ize libmath a bit move
this gets rid of a few ide-warnings about static initialization of
arrays with math types.
2019-10-03 17:12:35 -07:00
Mathias Agopian
c38ab64b6a Update Engine.cpp 2019-10-03 17:12:18 -07:00
Mathias Agopian
2548da9aed validate Engine in all Builder::build() methods
We now assert if the Engine given is invalid,
which would happen if it was used after being
destroyed.

Builder::build() is a reasonable place for this since it's generally
not in the critical path.
2019-10-03 17:12:18 -07:00
Ben Doherty
2fa08123a0 Add an iOS runner for backend-test (#1720) 2019-10-03 17:02:28 -07:00
Mathias Agopian
20c3d45b9c Engine.flushAndWait() is not needed in most places 2019-10-03 16:04:22 -07:00
Philip Rideout
c8af3226de OpenGL driver: fix error with uvec4 attribs, clean up Vulkan/Metal. (#1735)
The vertex shader in the material variant for skinning-and-morphing
declares a `uvec4` attribute that can be unused and disabled, but still
needs to be typed correctly.

The fix is motivated by WebGL, but the issue might occur with other
OpenGL implementations that have strict drivers.

Fixes #1726
2019-10-03 15:42:07 -07:00
Romain Guy
4b669fb1c7 Fix formatting 2019-10-03 14:25:39 -07:00
Mathias Agopian
340f6d721a Texture javadoc 2019-10-03 13:25:06 -07:00
Philip Rideout
489f3ff3c9 Add javadoc for View and Viewport. (#1734)
This CL also fixes warnings in IndirectLight and adds a new C++ getter
to View for consistency with Java.
2019-10-03 08:38:10 -07:00
Mathias Agopian
566d3613d1 Skybox javadoc 2019-10-02 15:58:57 -07:00
Mathias Agopian
e691cfa9e3 javadoc fixes for IndirectLight 2019-10-02 15:57:31 -07:00
Romain Guy
a4a1bf07aa Use our own constants instead of M_PI, M_*, etc. (#1733)
These constants are not part of the standard. We instead use our own
constexpr definitions in the filament::math namespace, as part of
the scalar.h include.
2019-10-02 15:31:38 -07:00
Ben Doherty
7ea7d1d5f2 Fix, matinfo doesn't work on Windows (#1732) 2019-10-02 13:20:24 -07:00
Philip Rideout
686c68c627 View API: minor doc fixes, add getRenderTarget. (#1723)
Since getRenderTarget() already exists in Java, we should also provide
it for C++ clients.
2019-10-02 10:13:42 -07:00
Romain Guy
c652dcfd2a Fix warning 2019-10-02 08:56:39 -07:00
Ben Doherty
01fab5a6e4 Update gtest to latest master (#1718) 2019-10-01 16:48:10 -07:00
Philip Rideout
16a5adcfde Fix quat slerp so it takes shortest path.
Fixes #1716.

This regressed in c65e561c59.
2019-10-01 16:22:00 -07:00
Philip Rideout
d550d36637 Add javadoc for RenderableManager, fix javadoc warnings. 2019-10-01 16:09:05 -07:00
Mathias Agopian
3857c1a9b2 javadoc for IndirectLight 2019-10-01 13:32:54 -07:00
Philip Rideout
2f8bc9239b Add JS binding to unary Camera::setExposure. 2019-10-01 12:39:40 -07:00
Philip Rideout
9de934e2bd Add Java annotations to setMorphWeights.
Noticed this while working on javadoc stuff.
2019-10-01 11:16:15 -07:00
Philip Rideout
769f1db011 Fix ambiguous atomic-to-bool conversion. 2019-10-01 10:43:15 -07:00
Philip Rideout
da72c58146 mac_runner: disable warning. 2019-10-01 10:43:15 -07:00
Ben Doherty
ef34dd2461 Fix operator overrides for Clang / MSVC on Windows (#1715) 2019-09-30 18:39:38 -07:00
prideout
90fdb95429 Update release notes for v1.4.0 2019-09-30 14:55:39 -07:00
Gregory Popovitch
d5c0d11404 Final changes for building with msvc 2019 on Windows (#1681) 2019-09-30 14:18:06 -07:00
Romain Guy
b0d116632c Add the ability to modify clip space coordinates in the vertex shader (#1704)
* Add the ability to modify clip space coordinates in the vertex shader

This introduces MaterialVertexInputs.clipSpaceTransform, a mat4 that
is applied to gl_Position before exiting the vertex stage.

* Address code review comments
2019-09-28 14:15:32 +03:00
Romain Guy
a67d92c2ca Lower limit from API 21 to API 19 (#1701)
* Lower limit from API 21 to API 19

This was requested by an internal application. API 19 is when OpenGL
ES 3.0 support was added so there is no good reason for us to not
support this API level. The only trick is to avoid referring to the
glTexStorage2DMultisample symbol directly as it only exists in 3.1.

* Compile out code we never use

* Use reflection to handle shared EGL contexts pre-API 21.

* Remove comment

* More fixes required to run on API level 19

- dlym() fails for ashmem on API 19, so we only try on API 26+ instead.
- Older emulation for OpenGL ES 3.0 returns error states for valid API calls so we need to clear the GL error bit before we create the GL driver.
- Activity lifecycle changes since API 19 would cause animations to keep running and to reference destroyed objects.
- EGLContext.getNativeHandle() is new in API 21, we need to use reflection on API 19. The new code path uses the class loading trick to avoid a bytecode verification error on API 19.

* Filament now runs properly on API level 19

This commit adds a new api_level() API to libutils which can be used to
query the platform's API level. On Android it works as expected, other
platforms currently return 0.
2019-09-28 14:13:06 +03:00
Philip Rideout
21b208398b Renderable doxygen now uses \see etc. 2019-09-27 15:26:08 -07:00
Philip Rideout
65b3e956df Add doxygen for RenderableManager and NameComponentMananger. 2019-09-27 15:26:08 -07:00
Mathias Agopian
c8d0715b16 SamplerParams is not default-initialized anymore
This is to allow designated aggregate initialization in C++20.
We do this because we have been relying on C99 designated
initialization, which is only supported by clang, other compilers need
c++20 to get a similar functionality. Therefore, we now try to limit
Ourselves to C++20 rules.

This could break existing code, since now SamplerParams needs to be
zero-initialized:

SamplerParams p{};

This is generally not an issue because the intended use is:

doSomethingWithSamplerParams({
      .filterMag = LINEAR,
   });
2019-09-27 13:53:11 -07:00
Mathias Agopian
16d307e543 enforce that quaternions are made of arithmetic types
I believe this might fix some msvc compiler issues too.
2019-09-26 19:05:33 -07:00
Mathias Agopian
5ed07639af workaround an MSVC bug with brace initialization
the C++ standard says:

   if T is a class type with a default constructor that is neither
   user-provided nor deleted (that is, it may be a class with an
   implicitly-defined or defaulted default constructor), the object
   is zero-initialized and then it is default-initialized if it has a
   non-trivial default constructor

Unfortunately, MSVC always calls the default constructor, even if it
is trivial, which breaks constexpr-ness.
To workaround this, we're always zero-initializing TVecN<>

Also removed constexpr from default constructors, since they never can
be constexpr as they're not initializing the vector.
2019-09-26 19:05:33 -07:00
Ben Doherty
e2655e93bc Add test framework for libbackend (#1697) 2019-09-26 12:30:30 -07:00
Philip Rideout
3a20be1d1a gltfio javadoc: add links, fix formatting. 2019-09-26 14:07:13 -04:00
Philip Rideout
8244f8e9cb Add javadoc comments for gltfio. 2019-09-26 14:07:13 -04:00
Romain Guy
c575fd7395 Add new Camera::setExposure(float) API (#1699)
* Add new setExposure(float) API on Camera

This API can be used to set a specific exposure value. For instance:

camera.setExposure(1.0)

Will set the exposure to 1.0 (or ISO 100, apeture 1.0 and shutter speed 1.2s).
This can be useful to match the lighting setup of other engines/tools. Setting
the exposure to 1.0 can for instance be used to treat lights as being unitless
(a directional light could have a strength of 2.0 for instance).

Fixes #1667

* Update release notes

* Fix Java build error

* Fix Web bindings
2019-09-26 20:08:46 +03:00
Philip Rideout
6a67cf0009 Enhance doxygen annotations for gltfio. 2019-09-26 12:18:56 -04:00
Romain Guy
79292481b3 Update documentation 2019-09-26 16:31:55 +03:00
Mathias Agopian
829f341de3 Engine and Camera javadoc (#1698) 2019-09-26 11:53:14 +03:00
Mathias Agopian
b509ef4dad handle scalar op vector directly to help some compilers 2019-09-25 22:39:13 -07:00
Mathias Agopian
422c8d9a64 Make DriverAPI.inc work with MSVC19 2019-09-25 13:52:41 -07:00
Romain Guy
d0b5c5cee8 Update compile & targetSdk from 28 to 29 (#1691) 2019-09-25 22:25:43 +03:00
Romain Guy
9b9548306d Fix Android samples (#1688)
Fixes #1687
2019-09-25 15:48:30 +03:00
Ben Doherty
b28e535540 Combine license file generation functions for MSVC compatability (#1684) 2019-09-24 12:45:38 -07:00
Mathias Agopian
1f7584ace2 Improve backend documentation (#1674)
We need to document the backend headers that are public.
Added documentation for:
- DriverEnums.h
- BufferDescriptor.h
- PixelBufferDescriptor.h

Disabled doxygen for public headers of backend that don't need to
be public for filament's API.

Also improved documentation for:
- LightManager.h
- RenderTarget.h
- TextureSampler.h
2019-09-24 11:18:15 -07:00
Ben Doherty
21512e8228 Update MaterialBuilder header comments for Doxygen (#1673) 2019-09-24 09:11:35 -07:00
Gregory Popovitch
a34903aa33 Update CMake configuration to support building with MSVC 2019-09-23 10:15:34 -07:00
Philip Rideout
4eba402adb Enhance the deprecation note. 2019-09-23 12:31:24 -04:00
Philip Rideout
64c5f5aee7 Deprecate populateTangentQuaternions. 2019-09-23 12:09:37 -04:00
Philip Rideout
1bda37776b Rename KtxUtility namespace to image::ktx. 2019-09-23 12:08:57 -04:00
Philip Rideout
1c5d7d6cc3 Build OpenImageDenoise only in host builds.
Fixes #1672
2019-09-21 14:31:54 -04:00
Mathias Agopian
b81ff060ae Remove HARD Fence public API
This API will be hard to implement on all backends (e.g. Vulkan) and
isn't that useful other than for timing.
2019-09-20 17:57:48 -07:00
Ben Doherty
0cebbfca23 Fix STB linking issues on Windows (#1669) 2019-09-19 15:31:54 -07:00
prideout
b9c7814dfd Remove unused variable. 2019-09-19 11:05:33 -07:00
Philip Rideout
ba9fd6719a FrameGraph, OpenGL, Vulkan: Clean up RenderPassFlags. 2019-09-19 10:52:29 -07:00
Mathias Agopian
f0ad12ce2e Fix matrix operations when using mixed precision
This is a similar fix to the previous vector fix. Commutative 
operations now always return the same type and the operations are
carried out in the precision resulting from following traditional
C++ rules.
2019-09-18 18:22:05 -07:00
Mathias Agopian
21acf53d3f improve vector operations when using implicit conversion
It used to be that operations e.g. like:

 float3{} + double{} would be computed as
 float3{} + float3{double{}} instead of
 float3{} + double3{double{}}

I other words, when an implicit conversion was involved on the right
it would be converted to the left side’s type, possibly losing 
precision.

Another problem was that swiping the operands could produce different
Results, e.g.:

   float3{1} * 5.0 -> float3{5.0f}
   5.0 * float3{1} -> double3{5.0}


This is no longer the case, now both expressions would return a double3. 

Note:

float3 r{};
r *= 5;

Is now equivalent to:

r[0] *= 5;
r[1] *= 5;
r[2] *= 5;

Instead of before:

r[0] *= 5.0f;
r[1] *= 5.0f;
r[2] *= 5.0f;
2019-09-18 18:22:05 -07:00
Mathias Agopian
2f5927d531 Minor code clean-up 2019-09-18 18:22:05 -07:00
Mathias Agopian
cf950a8d6f improve mixed-precision vector operations
We now follow the same rules than for basic types, when performing
mixed-precision operations, e.g.:

  float3 + double3 -> double3
  double3 + float3 -> double3
  dot(float3, double3) -> double

In particular, e.g. swapping the arguments to arithmetic operations
now always yields to the same result type (before, float3 + double3
would not return the same type than double3 + float3).
2019-09-18 18:22:05 -07:00
Philip Rideout
48b73ad998 Vulkan: use a dummy buffer for missing attributes.
To prevent an excess of variants, sometimes our vertex shaders declare
inputs that they never read from. For example, our skin-and-morph
variant might never read from the skinning attribute, but still declares
the input.

This PR is a sister to #1663, which is a fix for #1525.

See also #1279.
2019-09-18 18:14:45 -07:00
Philip Rideout
8fc94f127b Add language bindings for setCullingMode. 2019-09-18 18:12:02 -07:00
Philip Rideout
ce33f42e62 Fix warning in WebGL builds 2019-09-18 18:12:02 -07:00
Philip Rideout
61032e55b5 Update release notes. 2019-09-18 18:12:02 -07:00
Gregory Popovitch
6160d3f51e more changes for building with vs2019/msvc (#1506) 2019-09-18 17:25:22 -07:00
Ben Doherty
5d725f6de9 Fix, allow missing bone indices vertex attribute in Metal driver (#1663) 2019-09-18 16:59:43 -07:00
Romain Guy
020e9e59ed Don't use shallow fetches
This can cause issues with some PRs.
2019-09-18 16:39:40 -07:00
Mathias Agopian
bd777c1e56 Update View.cpp 2019-09-18 15:25:51 -07:00
Mathias Agopian
087b12c802 Fix minor time calculation imprecision
use double for the shader's 'second' time calculation, since 1 billion cannot be stored
in a float accurately.
2019-09-18 15:25:51 -07:00
prideout
42f2642d72 Vulkan: fix layout and miplevel for depth attachments.
We now use GENERAL layout for depth-sampled images in order to support
the SSAO use case of simultaneous binding + sampling.

We may be able to optimize this in the future by enhancing DriverAPI
so that it feeds more information into the render pass.

Fixes #1627, #1649.
2019-09-18 07:41:04 -07:00
Mathias Agopian
9c8d0ed5a1 handle basis inverting transforms (e.g. planar symmetries)
We now inverse the face winding order based on the determinant of
each render primitive world's transform. This prevents front/back faces
to be reversed when using a left-handed basis, after transform.

Fix #1520
2019-09-17 18:38:52 -07:00
Philip Rideout
9fd1705acf Add support for dynamic backface culling.
Inspired by our recent scissor change (#1639) but motivated by gltfio,
which was forced to create a ton of ubershaders (see #1562).

Tested with:

  gltf_viewer -u ../glTF-Sample-Models/2.0/TextureSettingsTest/glTF/TextureSettingsTest.gltf
2019-09-17 08:22:01 -07:00
Mathias Agopian
c981b52832 use enable_if<> instead of static_assert()
This is nicer because this way the methods that don't match don't
even exist. Also make it better when editing code with a C++ aware
ide.

Also use the less verbose std::enable_if_t<>
2019-09-16 18:13:21 -07:00
prideout
d074b6fd1a Remove unused field from backend::SamplerParams. 2019-09-16 16:04:26 -07:00
Mathias Agopian
7da5a25e2f Split OpenGLDriver in two files
We now have OpenGLContext which only tracks the OpenGL state.
It exposes an API very similar to OpenGL and does all the state
tracking. Currently, it doesn't expose non state-changing APIs 
(e.g. draw methods) and only implements the state-changing API we
need (to track the state of).
2019-09-16 15:53:02 -07:00
Romain Guy
b96b8eb787 Encode sRGB images as sRGB, not linear (#1646)
Fixes #1608
2019-09-16 10:51:17 -07:00
Philip Rideout
a25be5ee6e civetweb: do not build unless matdbg is built 2019-09-16 09:43:49 -07:00
Philip Rideout
bf9d1f1967 civetweb: fix warnings 2019-09-16 09:43:49 -07:00
Philip Rideout
561221fdc3 Rename PostProcessVertexInputs uv to normalizedUV. 2019-09-16 09:04:14 -07:00
Philip Rideout
8548afa231 Fix full-screen triangle winding and tex coords.
If you tried removing `culling: none` from any post-process materials,
everything would be fine in GL but Vulkan would be black because
our full-screen triangle was back-facing.

Continue reading only if you thrive in absurd situations.

- Metal and OpenGL define clip space as Y-up whereas Vulkan is Y-down.
- Metal and Vulkan define tex coords as Y-down whereas OpenGL is Y-up.
2019-09-16 09:04:14 -07:00
Philip Rideout
803e695d5c Add "uvToRenderTargetUV" to public shading API.
Fixes #1626.
2019-09-16 09:03:30 -07:00
Philip Rideout
5c58776313 Repair WebGL builds. 2019-09-16 08:39:20 -07:00
Romain Guy
0db39ea08a Add Android presubmit (#1644)
* Add Android presubmit

* Build Android on Linux, macOS targets have NDK 18 only

* Consolidate workflows and jobs

* Try to build Android on macOS

* Fix typo

* Cleanup scripts

* Try NDK side by side if NDK bundle has the wrong version

* Simplify logic

* Fix NDK logic

* Only update the NDK when necessary

* Fix typo

* Update required NDK version

* Favor NDK side-by-side

* Remove support for obsolete NDK bundle, require NDK side-by-side

* Install the NDK side by side when missing
2019-09-13 19:28:55 -07:00
Romain Guy
d76fb5d7e5 Fix matc dependencies and compilation messages 2019-09-13 18:41:38 -07:00
Romain Guy
f37ab8158c Name build jobs (#1643)
* Name build jobs

* Use build/ prefix instead of presubmit prefix

* Rename desktop workflow for consistency

* Forgot a git add...
2019-09-13 17:01:45 -07:00
Romain Guy
33db5acd13 Add web presubmit (#1642)
* Add web presubmit

* Use macOS not Linux
2019-09-13 16:40:57 -07:00
Mathias Agopian
e068204876 scissor is now part of PipelineState
setViewportScissor is gone, which will avoid a runtime error if
called outside of begin/endRenderPass, making the driver API more
robust.
2019-09-13 13:58:29 -07:00
Ben Doherty
d849231caf Remove old post-process shader pipeline (#1631) 2019-09-13 09:29:28 -07:00
Ben Doherty
996bd58fcb Use bitmask enum utility for MaterialBuilder::TargetApi (#1634) 2019-09-13 09:28:58 -07:00
Mathias Agopian
9b70dc5253 Fix linux/android DEBUG builds 2019-09-12 19:43:27 -07:00
Benjamin Doherty
31af9ca80f Automatically download iOS toolchain for Kokoro / GitHub 2019-09-12 17:41:33 -07:00
Benjamin Doherty
08aa2ec4ca Fix, iOS builds still need to work with Kokoro 2019-09-12 17:15:01 -07:00
Ben Doherty
95be2e24ce Fix iOS CI build (#1635) 2019-09-12 16:59:21 -07:00
Mathias Agopian
7b200ec85e make TextureUsage and TargetBufferFlags enum class
This prevents values like NONE or COLOR which are very generic
names to collide with other classic enums. It also forces us to be
specific about what we're doing when converting to bitfields.

Also added a utility to easily enable any enum or enum class to
support binary operation (i.e. Bitmask contract), e.g.:

template<> struct utils::EnableBitMaskOperators<Foo> : public std::true_type{};

gives all binary operators to 'Foo'. Foo must be an enum.

Also added any() and none() convenience to convert an enum to a boolean.
2019-09-12 11:19:54 -07:00
Mathias Agopian
cf1b45c052 Make SamplerParam a C struct
This simplifies its usage at call sites at lot.

The C++ ctor was needed before to create uninitialized objects, which
was needed in SamplerGroup. Instead we use a custom 'static_vector' 
(instead of std::array), this actually makes SamplerGroup's
implementation clearer.

static_vector<> is just a fixed-capacity, fixed-storage dynamic
vector. we currently don't expose it and implement only what we
need.
2019-09-12 11:18:47 -07:00
Ben Doherty
57e7280ac9 Remove post-process uniform / sampler blocks (#1624) 2019-09-11 16:57:48 -07:00
prideout
64d432cb34 PostProcessManager: another Vulkan-related re-ordering.
My fault, I did not catch this while reviewing #1621.
2019-09-11 11:33:43 -07:00
Ben Doherty
fde23c89e2 Use post-process material domain for FXAA (#1621) 2019-09-11 09:35:40 -07:00
Ben Doherty
0e3bc8a443 Only install iOS builds if requested (#1614) 2019-09-11 09:35:01 -07:00
Philip Rideout
88bcd133f2 filamat: add Backend-to-TargetApi utility. 2019-09-11 08:38:47 -07:00
Philip Rideout
2072b67e04 PostProcessManager: repair Vulkan by re-ordering commands.
The suzanne Vulkan demo was asserting because `setViewportScissor` can
only be called within a render pass, which means that
`MaterialInstance::use` can only be called within a render pass.

We probably need a better way to enforce these rules...
2019-09-11 08:38:19 -07:00
Philip Rideout
2a2974580c resgen: remove unused / incorrect constant. 2019-09-10 10:19:37 -07:00
Ben Doherty
65f72a48fb Terminate tonemapping material (#1619) 2019-09-10 10:15:15 -07:00
Philip Rideout
65674c7c51 filagui: disambiguate the resources header. 2019-09-10 09:08:34 -07:00
Philip Rideout
8d1b630b24 resgen: repair wasm builds. 2019-09-10 08:02:35 -07:00
Philip Rideout
1235a66f83 resgen: avoid overwriting the header if nothing changed. 2019-09-10 08:02:35 -07:00
Philip Rideout
411e7f4511 resgen: use extern int instead of #define. 2019-09-10 08:02:35 -07:00
Philip Rideout
cf095cbdd2 filagui: use resgen for the UI material.
This makes building samples faster when combined with PR #1613.
2019-09-10 08:02:23 -07:00
Philip Rideout
394db90ec9 CMake: Use configure_file for licenses.inc
Every time ninja invoked CMake, we were unconditionally regenerating
all of our "licenses.inc" files which caused a ton of re-compiles.

Before the change, building after touching `ubershader.mat.in` took over
a minute, now it takes 20 seconds.

The CMake "file" commands are bad. The CMake documentation says:

    If the file is a build input, use the configure_file() command
    to update the file only when its content changes.
2019-09-10 08:02:12 -07:00
Philip Rideout
72644f1416 mipgen: add --quiet argument.
Note that some encodings like astc will still be noisy due to
third-party code but this works fine for the s3tc encoder.
2019-09-10 08:01:49 -07:00
Philip Rideout
ed2fc0e391 matdbg: opt-in via environment variable. 2019-09-10 08:01:29 -07:00
Ben Doherty
c9f4d983a0 Convert SSAO shaders to use post-process materials (#1610) 2019-09-09 16:41:49 -07:00
Philip Rideout
694db54d9d Minor cleanup. 2019-09-09 11:10:39 -07:00
Philip Rideout
15b663ba0b matdbg client: show inactive variants in gray.
This makes it easier to find the variants that are actually being used.
2019-09-09 11:04:02 -07:00
Philip Rideout
23ee5e276c matdbg server: add query for active programs.
This uses the program cache to determine the set of variants that
are actually being used.
2019-09-09 11:04:02 -07:00
Ben Doherty
7c24a1bbaa Add custom vertex shaders for post-process materials (#1606) 2019-09-09 09:17:35 -07:00
Ben Doherty
32eee2cdf1 Tungsten: upgrade Gradle, fix IBL (#1604) 2019-09-09 09:13:06 -07:00
Ben Doherty
7dbad8928f Ensure GitHub actions build presubmit (#1605) 2019-09-06 19:43:42 -07:00
Philip Rideout
0e81207758 WebGL: add demo for glTF animation.
Fixes #1583.
2019-09-06 16:44:03 -07:00
Philip Rideout
c67266833a gltfio: rename getResourceUrl => getResourceUri 2019-09-06 16:43:53 -07:00
Philip Rideout
d6b0b6ff52 gltfio: expose all resource URI strings.
Some buffers (like animation data) do not contain any vertex data
but web-based clients still need to know about this.

This interface is much simpler than the bindings lists, and in fact
we might remove the binding lists in the future to simplify the API.

Fixes #1593.
2019-09-06 16:43:53 -07:00
Ben Doherty
9d0ad3586d Update tonemapping to use new post-process material domain (#1428) 2019-09-06 15:04:52 -07:00
Philip Rideout
7393d5d98c WebGL: add LightManager getter methods to fix #1599. 2019-09-06 14:12:33 -07:00
Mathias Agopian
7a819a60ae RGB32F is never supported as a render target 2019-09-06 11:09:20 -07:00
Mathias Agopian
c735659592 better handle float rendertargets. fixes #1533
some driver don't support all float buffers as rendertarget, so we
improve the backends (opengl in this PR) to return what it actually
supports and we update the client (filament) to take that into
account when making decisions.
2019-09-06 11:09:20 -07:00
Mathias Agopian
c36745db97 simplify DriverAPI.inc macros a bit further
we now handle zero argument declaration in some cases.
2019-09-06 11:08:59 -07:00
Philip Rideout
a5284af2b8 Use original glTF filename for FlightHelmet. 2019-09-06 10:47:47 -07:00
Philip Rideout
4989c3dc48 Remove unused material. 2019-09-06 10:47:47 -07:00
Philip Rideout
3925b0a8ec matdbg: Add detailed README describing the design. 2019-09-06 08:17:02 -07:00
Mathias Agopian
356610ba1d simplify DriverAPI.inc macros
by using variadic macros we can greatly simplify the macros
definitions for declaring the backend API.

in particular we don't need a macro per number-of-argument, this is
now done automatically.

Unfortunately, we still can't handle zero-argument declarations, so
we keep the _0 variants.
2019-09-05 15:07:25 -07:00
Philip Rideout
9eae72f99d matdbg: allow editing for GLSL shaders.
This performs surgical modification of the IFF chunks rather than
invoking filamat from within matdbg. Low-level direct manipulation
(bypassing optimization passes etc) allows us to diagnose issues with
the shading pipeline.

This CL also adds keystroke bindings to the Monaco editor. You can
press Cmd+S to rebuild the current materials, or use Ctrl+Arrow to
navigate between shader variants and materials.

In a subsequent CL I will add a README that describes how this works in
detail, it will include a list of limitations and a feature wishlist.
2019-09-05 13:03:49 -07:00
Philip Rideout
1d12fc1aaa Fix default IBL in lucy_bloom. 2019-09-04 13:35:34 -07:00
Philip Rideout
f9bd085757 OpenGL: shader errors should not cause termination.
We now check for FILAMENT_ENABLE_MATDBG and simply skip the draw call
when the program is invalid. The initial error message is still dumped
to the console, but we will not terminate or dump an infinite cascade of
error messages.
2019-09-04 13:11:11 -07:00
Philip Rideout
1003926665 Engine: add support for material edits.
This only adds Engine-side support for edits. The debugger client will
be enhanced in another PR. This works by simply clearing the program
cache and swapping out the MaterialParser. Neat!

However, there are some gotchas:

- The DebugServer callback is triggered on another thread so to be safe
  we defer the swap until the subsequent call to getProgram().

- The getProgram() method is const but in reality it can insert entries
  into a mutable cache. This CL makes the constness of it even more
  misleading by applying edits.

- The edit operation will leak the old HwProgram but I feel this is
  fine because simplicity is crucial for injected behavior and this is
  a developer-only feature.
2019-09-04 13:10:44 -07:00
Mathias Agopian
709180932e use Builder.sample() instead Builder.read() for textures
Texture are special resources that can be read in different ways; either as
attachment (read) or sampled (sample), the user must now declare explicitly if she intends
to sample from a texture.

sample() implies read() so there is no need to call both.

This was already the case, as read() used to take a boolean, essentially meaning the same
thing. However, the bool didn't make sense for non-texture resources.

This also simplifies the implementation.
2019-09-04 11:40:59 -07:00
Mathias Agopian
132326f8f2 getDescriptor() needn't be const 2019-09-04 11:40:59 -07:00
Mathias Agopian
92e86de630 No need to specify the sample count for attachments
the framegraph will figure it out automatically from the 
rendertarget sample count. this is done during resolve(), and after
moveResource is taken into account (so this works with imported and
moved resources).
2019-09-04 11:40:59 -07:00
Mathias Agopian
3e8a145fce importRenderTarget() now returns a FrameGraphRenderTargerHandle
This makes the API more symetric/logical.
2019-09-04 11:40:59 -07:00
Mathias Agopian
a5db6a84b6 add name to rendertarget creation 2019-09-04 11:40:59 -07:00
Mathias Agopian
f192b89ef9 sample count for rendertargets doesn't need to be specified
when creating a rendertarget, if the sample count is not specified,
any existing target with the same attachment will match.
this simplifies both the user and implementation.

getRenderTargetDescriptor() now takes a RenderTargetHandle instead
of a texture handle.
2019-09-04 11:40:59 -07:00
Mathias Agopian
637395b075 declaring a rendertarget now returns a FrameGraphRenderTargetHandle
this makes managing render targets more excplicit and simplify the
code a lot. before, render targets were referenced through one of
their attachment.
2019-09-04 11:40:59 -07:00
Philip Rideout
b03172f4a2 Java bindings: fix CUSTOM attribute mismatch.
Fix #1578
2019-09-03 15:49:22 -07:00
Romain Guy
24225315fa Add iOS presubmit (#1589)
* Add iOS presubmit

* Use a separate workflow for iOS
2019-09-03 11:56:32 -07:00
Romain Guy
2bc9256a7a Fix compilation error when DEBUG_COMMAND_STREAM is true (#1588)
Fixes issue #1585
2019-09-03 10:00:11 -07:00
Philip Rideout
264cf020b6 Minor TypeScript annotations fix. 2019-09-03 08:44:16 -07:00
Romain Guy
ceafc7835b Rename CI workflow to Presubmit 2019-08-31 11:53:20 -07:00
Romain Guy
db8d4ee076 Setup Linux for GitHub CI (#1581)
* Setup Linux for GitHub CI

* Pass env var properly
2019-08-30 15:24:29 -07:00
Romain Guy
26f7907b17 Fix punctuation (#1580) 2019-08-30 14:59:25 -07:00
Romain Guy
c35f1e0b29 More fixes 2019-08-30 14:53:02 -07:00
Romain Guy
e2ee00e36e Fix Linux CI on GitHub 2019-08-30 14:50:40 -07:00
Romain Guy
d87f77c41a Don't set $TARGET 2019-08-30 14:45:03 -07:00
Romain Guy
13fcce2d9a Fix workflow 2019-08-30 14:44:47 -07:00
Romain Guy
216761656f Fix CI 2019-08-30 14:38:37 -07:00
Romain Guy
05e9d708ed Update build scripts for workflows 2019-08-30 14:29:44 -07:00
Romain Guy
cc321385cd Run CI on Linux and macOS 2019-08-30 14:29:18 -07:00
Romain Guy
038d39f294 Fix path 2019-08-30 14:07:01 -07:00
Romain Guy
3555a01541 Support for workflows 2019-08-30 14:03:55 -07:00
Romain Guy
5d52266eac Add workflow 2019-08-30 14:03:37 -07:00
Romain Guy
5368756983 Remove workflow 2019-08-30 13:59:34 -07:00
Romain Guy
f74e136f46 Add GitHub Action for CI duty 2019-08-30 12:02:39 -07:00
Romain Guy
040c1c686a Update build script to support GitHub Actions 2019-08-30 12:02:21 -07:00
Philip Rideout
e74c33ce04 Java bindings now define CUSTOM / MORPH attribs.
Fixes #1578.
2019-08-29 17:13:28 -07:00
Philip Rideout
ab24634637 Add matdbg to READMEs. 2019-08-29 10:19:18 -07:00
Philip Rideout
8f3bdc20ba matdbg: add "install" directive.
Fixes #1576.
2019-08-29 10:19:18 -07:00
Romain Guy
74aaaed4db Code cleanup 2019-08-29 09:34:12 -07:00
Philip Rideout
9c64c20ceb matdbg: fail gracefully, emit logs, fewer threads. 2019-08-29 09:13:20 -07:00
Mathias Agopian
b92c748fcc Use allocator Debug policy in more places 2019-08-28 19:06:54 -07:00
Mathias Agopian
844dde4e6d simple allocation debugger
This can be enabled by using the Tracking::Debug policy,
currently this just fills allocation on alloc() and free(),
which is useful to help detect access to uninitialized memory
and use after free.

Now enabled by default in libfilament allocators on debug builds.

This caught uninitialized access in the froxelizer.
2019-08-28 19:06:54 -07:00
Philip Rideout
c0de0ca36d gltfio: fix transforms for nodes with non-uniform scale.
I tested this with the problem model and a couple other models.

This bug is inherited from cgltf_node_transform_local, so I will
upstream the fix. There are two ways to see that the new math is
correct:

1) Look at decomposeMatrix and note that scales belong to rows,
   not columns.

2) Look at the Brandon Jones implementation:
   http://glmatrix.net/docs/mat4.js.html#line1079

Fixes #1519.
2019-08-28 15:31:33 -07:00
Mathias Agopian
ad3bc4a6f7 ran "optimize imports" on all files
This removed a few unneeded imports.
2019-08-28 15:19:50 -07:00
Mathias Agopian
572b2cff57 fix use after free
We were storing pointer to RenderTarget objects that were allocated
in a vector<>, which obviously is not safe.
This didn't cause any issue most of the time because our allocator is
a linear allocator.
2019-08-28 15:19:26 -07:00
Philip Rideout
fb9cf43768 WebGL: fix JPEG decoding.
We've been using STB for a while now and we already include a JPEG
decoder in our wasm bundle. This removes some vestigial code that was
getting in the way.

Fixes #1565.
2019-08-28 13:15:15 -07:00
Philip Rideout
f41db8bb91 gltfio: add lazy loading to ubershader mode. 2019-08-28 13:01:05 -07:00
Philip Rideout
89c8e7c51a gltfio: support doubleSided in ubershader mode.
Fixes #1562
2019-08-28 13:01:05 -07:00
Jordan Rupprecht
0ad21cd47a Avoid undefined behavior when doing pointer calculation.
Performing `base + offset` pointer arithmetic is only allowed when `base` itself is not nullptr. In other words, the compiler is assumed to allow that `base + offset` is always non-null, which an upcoming compiler release will do in this case. The result is that CommandStream.cpp, which calls this in a loop until the result is nullptr, will never terminate (until it runs junk data and crashes).

Avoid this by using intptr_t instead of actual pointers (i.e. char*), which seems to avoid this failure.
2019-08-28 12:13:09 -07:00
Mathias Agopian
7dbf81fe3d FrameGraphPassResources should use typed FrameGraphId<> 2019-08-28 11:43:27 -07:00
Mathias Agopian
3072309574 rename FrameGraphResource to FrameGraphHandle
also FrameFrapheResourceId to FrameGraphId.
2019-08-28 11:43:27 -07:00
Mathias Agopian
573e9220a2 Improve import API and code 2019-08-28 11:43:27 -07:00
Mathias Agopian
e22f02d228 public APIs only use FrameGraphResourceId<> 2019-08-28 11:43:27 -07:00
Mathias Agopian
b620b7efa1 major framegraph overhaul 2019-08-28 11:43:27 -07:00
Philip Rideout
d1593326cb gltfio: support spec-gloss in ubershader mode.
Fixes #1564.
2019-08-28 10:56:56 -07:00
Romain Guy
6a8e6d45b5 Improve materials under white furnace test (#1563)
* Improve materials under white furnace test

Two major changes:
- Mobile target now implements a cheaper variant of the off specular
  peak bias (which moves the reflected vector towards the normal).
  This greatly helps with rough surfaces that may otherwise point
  toward a bright part of the IBL.
- The indirect diffuse love is now properly attenuated to avoid adding
  the energy reflected by the specular layer. This allows dielectrics
  to be correctly energy conserving under a white furnace.
- Tweak the (hacky) clear coat layer attenuation to behave properly
  under a white furnace.

* Use the same reflected vector modification everywhere
2019-08-28 10:39:57 -07:00
Ben Doherty
11336964a1 Add startCapture and stopCapture debug driver methods (#1560) 2019-08-28 10:34:46 -07:00
Mathias Agopian
71e98d54f5 add missing includes 2019-08-27 18:03:45 -07:00
Mathias Agopian
417ba87fcb don't use hard-coded values for Builder::radiance()
Instead we use constexpr expressions of the values,
which makes it much easier to remember where they
come from.
2019-08-26 17:47:28 -07:00
prideout
f71fea4ef5 Update release notes. 2019-08-26 14:16:07 -07:00
prideout
dc062305b9 Bump to 1.3.2 2019-08-26 14:07:14 -07:00
Philip Rideout
bc94269e73 matdbg: fix a warning when building civetweb. 2019-08-26 12:37:38 -07:00
Philip Rideout
876534330a matdbg: add more info to the details panel. 2019-08-23 16:43:16 -07:00
Philip Rideout
2af8ef81b3 matdbg: add required attributes. 2019-08-23 16:43:16 -07:00
Philip Rideout
8da553db58 matdbg: fix active shader highlight. 2019-08-23 16:43:16 -07:00
Philip Rideout
83e9857b30 matdbg: remove bogus hrefs from anchors. 2019-08-23 16:43:16 -07:00
Philip Rideout
088ed2d472 matdbg: make mat list and shader list scrollable. 2019-08-23 16:43:16 -07:00
Ben Doherty
c826b03411 Break driver dependency on Texture.h (#1549) 2019-08-23 16:13:24 -07:00
Benjamin Doherty
f1257994d1 Disable DirIncluder tests 2019-08-23 13:31:20 -07:00
Benjamin Doherty
efb9e005b2 Fix build due to DirIncluder test 2019-08-23 12:40:47 -07:00
Romain Guy
6f30ff31c8 Fix warnings 2019-08-23 11:55:06 -07:00
Romain Guy
d0f38c2d5d Update Gradle, AGP and Kotlin 2019-08-23 11:52:30 -07:00
Ben Doherty
d4943cc70b Add #include preprocessing to filamat and matc (#1541)
* Add #include preprocessing to filamat and matc

* Update RELEASE_NOTES

* Fix RELEASE_NOTES

* Use final instead of virtual / override

* Clarify comments

* Use pure virtual for includer functions

* Use a callback instead of an interface

* Rename Includer.h to IncludeCallback.h

* Update comment
2019-08-23 11:10:36 -07:00
Ben Doherty
9a47ea1ef0 Fix Tungsten build (#1548) 2019-08-23 11:08:27 -07:00
Romain Guy
ac33331fd6 Update documentation to fix typos 2019-08-23 10:58:08 -07:00
Philip Rideout
a65998bbc1 Fix Linux build, hopefully. 2019-08-23 10:16:22 -07:00
Philip Rideout
5120c18977 Update third_party/cgltf to latest. 2019-08-23 09:50:08 -07:00
Philip Rideout
21347eaa88 Fix build break, re-enable DebugServer. 2019-08-23 09:49:57 -07:00
Philip Rideout
a191cc8171 Fix build break. 2019-08-22 17:07:34 -07:00
Philip Rideout
ad3d823fa3 matdbg: Use Variant enum. 2019-08-22 16:12:20 -07:00
Philip Rideout
8d7c00b0d9 matdbg: add DebugServer class.
matdbg is now linked into the Filament Engine in debug config (allowing
live inspection of GLSL / SPIRV) and into the matinfo tool (to support
the --web-server option).

In both cases, the library spins up a small web server that listens to
http://localhost:8080. You can run any Filament app and attach to it.

The web client caches all material information. This allows the user to
close an atttached Filament app, and the web app will continue to
function properly (useful for crash diagnosis). Moreover the user can
launch a second Filament app and the web client will add its materials
to the existing list (useful when comparing two Filament apps).

For now this only supports inspection, not editing. Some of the material
info such as required attributes is not yet displayed but this will be
easy to flesh out in a subsequent PR.
2019-08-22 16:12:20 -07:00
Ben Doherty
99d9ea73ac Fix potential Metal memory leak (#1542) 2019-08-22 14:15:39 -07:00
Philip Rideout
c4b0edbfe3 Introduce matdbg library, simplify matinfo.
This moves some of the matinfo functionality into a library which will
soon have an embedded web server.
2019-08-22 08:11:06 -07:00
Philip Rideout
6d74d31ecd Add civetweb to third_party (MIT).
This is a dependency of our upcoming web-based material debugger. This
CL also includes `tnt/CMakeLists.txt`, which builds a static lib in a
minimal configuration that enables WebSocket support. The entire library
is built from only 4 files:

    ${PUBLIC_HDR_DIR}/CivetServer.h
    ${PUBLIC_HDR_DIR}/civetweb.h
    ${SRC_DIR}/civetweb.c
    ${SRC_DIR}/CivetServer.cpp
2019-08-21 17:45:37 -07:00
Ben Doherty
bd22bed2fb Fix Windows test cases (#1537) 2019-08-20 15:36:22 -07:00
Ben Doherty
f8aa17c245 Preallocate uniform buffers in Metal (#1528)
* Preallocate uniform buffers in Metal

* Allocate empty uniform at draw time

* Remove unnecessary return
2019-08-19 15:37:13 -07:00
Ben Doherty
6ba20f29d4 Fix normal mapping with skinning or morphing in Metal (#1530) 2019-08-16 17:24:01 -07:00
Philip Rideout
95ef8d85db Minor fixup to suzanne Vulkan demo. 2019-08-16 17:04:32 -07:00
Mathias Agopian
eca381a5be Rename fg::Resource to fg::TextureResource 2019-08-15 16:58:58 -07:00
Mathias Agopian
ea47f0bd87 simplify how we determine if a resource is sample-able 2019-08-15 16:58:58 -07:00
Mathias Agopian
b849008ce5 rename useRenderTarget to createRenderTarget 2019-08-15 16:58:58 -07:00
Mathias Agopian
5389ac963f framegraph: useRenderTarget() doesn't set access anymore
access to resources now needs to be done explicitly before calling
useRenderTarget(). This simplifies things and is needed for more
simplifications to come.
2019-08-15 16:58:58 -07:00
Mathias Agopian
0fd16d760a refactor/cleanup of FrameGraph code
this change mostly puts all internal classes into their own file,
making FrameGraph.cpp much smaller and easier to deal with
2019-08-15 16:58:58 -07:00
Philip Rideout
6754e80c75 gltfio: Minor cleanup. 2019-08-15 14:29:07 -07:00
Benjamin Doherty
fb95d8e7a4 debug 2019-08-14 18:13:57 -07:00
Benjamin Doherty
40efd5fd23 More debug 2019-08-14 18:13:57 -07:00
Benjamin Doherty
d440775775 debug 2019-08-14 18:13:57 -07:00
Benjamin Doherty
e1d9f26dbd Try keyserver 2019-08-14 18:13:57 -07:00
Benjamin Doherty
ee048d259f Try to fix broken Linux and Android CI 2019-08-14 18:13:57 -07:00
Philip Rideout
d7e5f2c26b Web: remove more &free callbacks.
It is not safe to take the address of a stdlib function with emscripten,
this causes an intermittent crash in the WebGL backend.
2019-08-14 15:14:49 -07:00
Philip Rideout
c11a61aa15 Web: add setBoolParameter, fixes #1499. 2019-08-14 13:05:17 -07:00
Philip Rideout
b859b57797 matc, matinfo et al: improve diagnostic output
This enhances the exisiting matc option "--debug" so that it includes
debug information in the resulting SPIR-V.

Previously we tied this to the build configuration which isn't very
flexible. Moreover this needs to be orthogonal to shader opts, because
we often need to debug problems that arise from optimization.

This CL also gives meaningful names to the ubershader materials and
fixes up the new matinfo analysis so that it prints the entire SPIRV
chunk that corresponds to each potentially problematic GLSL codeline.

Motivated by #1516.
2019-08-14 13:01:43 -07:00
Mathias Agopian
76cca21c65 most StructureOfArray<> methods cannot be constexpr
This is because we can't create a constexpr SoA in the first place,
and it wouldn't be very useful if we could.

This change ripples into users of SoA.
2019-08-13 18:29:15 -07:00
Philip Rideout
a09c3b154a Fix paths in android glTF demo. 2019-08-13 12:47:08 -07:00
Philip Rideout
2460f9e721 Better Adreno workaround for #1096.
This is an improved workaround for a known limitation with Vulkan
drivers on Android P. The offending SPIR-V sequence was introduced
during shader optimization and can be avoided by using full precision
for certain variables.
2019-08-13 11:36:59 -07:00
Mathias Agopian
d30429e891 make getNativeObject() public
this allows mixed java/native apps to use filament objects on both
sides more easily
2019-08-12 21:56:49 -07:00
Ben Doherty
c23c3d48b2 Remove unusued function (#1508) 2019-08-12 15:17:05 -07:00
Philip Rideout
046809cee0 Use --quiet when invoking cmgen in build. 2019-08-12 14:59:46 -07:00
Philip Rideout
c9750dd633 Fix gradle bug that created junk folder. 2019-08-12 13:41:52 -07:00
Philip Rideout
18a0bd70fb Restore accidental removal of Adreno workaround. 2019-08-12 11:42:16 -07:00
Philip Rideout
38955a91e5 matinfo: fix ordering of analyze-spirv output. 2019-08-12 11:40:36 -07:00
Mathias Agopian
eac2ad7e05 fix some lint warnings in Hash.h 2019-08-12 11:09:24 -07:00
Philip Rideout
5df53f733e Add error message to matinfo. 2019-08-12 10:45:44 -07:00
Romain Guy
50abcf6bfa Don't check for NDK side-by-side if ndk bundle is found 2019-08-12 10:22:06 -07:00
Philip Rideout
4d41b83448 Minor fixups in TypeScript annotations. 2019-08-12 09:13:46 -07:00
Philip Rideout
d2324d253c filamat: when building debug, use verbose SPIR-V. 2019-08-12 09:12:59 -07:00
Philip Rideout
cf9e87a892 matinfo: add SPIR-V analysis functionality
This adds a new option to matinfo that does some very dumb regex-based
analysis on the disassembled SPIR-V, then dumps out transpiled GLSL
that has annotations added to the end of some codelines. For example:

```glsl
float luminance(vec3 linear) // POTENTIAL MIXED PRECISION %49 = OpDot %float %linear %48; relaxed = %49 %linear
{
    return dot(linear, vec3(0.2125999927520751953125, 0.715200006961822509765625, 0.072200000286102294921875));
}
```

The code currently tries to find potential mixed precision but does not
do the right thing yet for pointers and structs. This is just a starting
point to help diagnose problem areas in our generated SPIR-V.
2019-08-12 09:12:59 -07:00
Romain Guy
829db244f2 Add support for side-by-side NDK (#1505) 2019-08-11 17:36:49 -07:00
Gregory Popovitch
772af1e897 More fixes for building with vs2019/msvc (#1500)
* Update Froxelizer.h

Fix this error when building with msvc from vs2019
error C2926:  'filament::details::Froxelizer::FroxelEntry::<unnamed-tag>::offset': a default member initializer is not allowed for a member of an anonymous struct within a union

* Fix some compilation issues with vs2019/msvc

Program.cpp:
1>C:\greg\github\filament\filament\backend\src\Program.cpp(28,42): error C2610:  'filament::backend::Program::Program(void) noexcept': is not a special member function or comparison operator which can be defaulted
1>C:\greg\github\filament\filament\backend\src\Program.cpp(28,42): message :  exception specification does not match the implicitly declared specification.

GLUtils.h: __PRETTY_FUNCTION__ macro is clang specific. Use MSVC equivalent

Color.h: fix warning

* #1493 - inline constructor in definition as requested by @romainguy

* #1493  "move this #define inside the #else below" as requested

* #1493 revert last change which causes compilation failures on other platforms.

provide empty implementation of Program::Program() in Program.cpp

* More fixes for building with vs2019/msvc

* #1500 use consistent macro definition syntax (@bejado)

* #1500 simplify DEBUG_COMMAND macro as requested by @pixelflinger

* #1500 use `{ 0 }` which is accepted by Visual Studio  (` = 0 ` is not accepted)

* #1500 remove incorrect UTILS_RESTRICT alltogether
2019-08-09 16:38:31 -07:00
Romain Guy
52a8f50539 Fix backface issue in lit-cube sample 2019-08-09 16:10:12 -07:00
Mathias Agopian
9f1cadf77a remplace multimap by a vector for the framegraph texture cache
because we don't expect many items in the cache, using a linear
search is not a problem -- multimap generates tons of code 
by comparison.
2019-08-09 15:36:40 -07:00
Mathias Agopian
308935d706 fix framegraph test 2019-08-09 15:36:40 -07:00
Mathias Agopian
67f75f0ecf reformatting 2019-08-09 15:07:12 -07:00
Mathias Agopian
f728776714 better normal transforms
We use the cofactors to transform normals, which preserves its
direction, unlike of the inverse-transpose method.
2019-08-09 15:07:12 -07:00
Mathias Agopian
10259f80f4 Add support for det() and cof()
Respectively computing the determinant and cofactors of a matrix.
2019-08-09 15:07:12 -07:00
Mathias Agopian
c65239aec2 Implement a basic Texture cache for the framegraph
We're only caching textures (not render targets yet), and we're
evicting cache entries older than 30 allocations.

This should cut down on texture allocation / gl calls.

The cache should get in a stable state very quickly (less than half
second).
2019-08-09 15:01:04 -07:00
Philip Rideout
0bd9515b3d Add getWorldOffset() shader API, enhance docs.
This allows advanced shader authors like myself to get coordinates
in the API level world space.

Fixes #1485.
2019-08-09 08:28:28 -07:00
Ben Doherty
442315aaa8 Silence Metal shader compilation warnings (#1496) 2019-08-08 18:07:01 -07:00
Gregory Popovitch
614a7d6de8 Fix some compilation issues with vs2019/msvc (#1493)
* Update Froxelizer.h

Fix this error when building with msvc from vs2019
error C2926:  'filament::details::Froxelizer::FroxelEntry::<unnamed-tag>::offset': a default member initializer is not allowed for a member of an anonymous struct within a union

* Fix some compilation issues with vs2019/msvc

Program.cpp:
1>C:\greg\github\filament\filament\backend\src\Program.cpp(28,42): error C2610:  'filament::backend::Program::Program(void) noexcept': is not a special member function or comparison operator which can be defaulted
1>C:\greg\github\filament\filament\backend\src\Program.cpp(28,42): message :  exception specification does not match the implicitly declared specification.

GLUtils.h: __PRETTY_FUNCTION__ macro is clang specific. Use MSVC equivalent

Color.h: fix warning

* #1493 - inline constructor in definition as requested by @romainguy

* #1493  "move this #define inside the #else below" as requested

* #1493 revert last change which causes compilation failures on other platforms.

provide empty implementation of Program::Program() in Program.cpp
2019-08-08 17:03:26 -07:00
Philip Rideout
d28189c173 Repaired broken npm package. 2019-08-08 13:27:52 -07:00
Ben Doherty
dc75c10014 Update Xcode projects for KTX build step (#1495) 2019-08-08 11:39:13 -07:00
Philip Rideout
91ff1d87eb Replace prebuilt KTX with a build step.
This removes the samples/envs folder and replaces these KTX files with a
build step that is driven by CMake / gradle / bash, depending on
platform.

This makes it easier to use IBL files that are generated by the latest
and greatest version of cmake.
2019-08-08 10:41:13 -07:00
1149 changed files with 156795 additions and 30725 deletions

108
.github/workflows/presubmit.yml vendored Normal file
View File

@@ -0,0 +1,108 @@
name: Presubmit
on: [pull_request]
jobs:
build-desktop:
name: build-desktop
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [macos-latest, ubuntu-latest]
steps:
- name: Checkout Filament
run: |
git version
git init $GITHUB_WORKSPACE
cd $GITHUB_WORKSPACE
git remote add origin https://github.com/google/filament
git config gc.auto 0
git fetch --tags --prune --progress --no-recurse-submodules origin +${GITHUB_REF}:${GITHUB_REF/refs\//refs\/remote\/}
git checkout --progress --force ${GITHUB_REF/refs\//refs\/remote\/}
- name: Run build script
run: |
WORKFLOW_OS=`echo \`uname\` | sed "s/Darwin/mac/" | tr [:upper:] [:lower:]`
cd build/$WORKFLOW_OS && ./build.sh ${TARGET}
env:
TARGET: presubmit
build-windows:
name: build-windows
runs-on: windows-latest
steps:
- name: Checkout Filament
run: |
git version
git init %GITHUB_WORKSPACE%
cd %GITHUB_WORKSPACE%
git remote add origin https://github.com/google/filament
git config gc.auto 0
git fetch --tags --prune --progress --no-recurse-submodules origin +%GITHUB_REF%:%GITHUB_REF:refs/=refs/remote/%
git checkout --progress --force %GITHUB_REF:refs/=refs/remote/%
shell: cmd
- name: Run build script
run: |
build\windows\build-github.bat
build-android:
name: build-android
runs-on: macos-latest
steps:
- name: Checkout Filament
run: |
git version
git init $GITHUB_WORKSPACE
cd $GITHUB_WORKSPACE
git remote add origin https://github.com/google/filament
git config gc.auto 0
git fetch --tags --prune --progress --no-recurse-submodules origin +${GITHUB_REF}:${GITHUB_REF/refs\//refs\/remote\/}
git checkout --progress --force ${GITHUB_REF/refs\//refs\/remote\/}
- name: Run build script
run: |
cd build/android && ./build.sh ${TARGET}
env:
TARGET: presubmit
build-ios:
name: build-iOS
runs-on: macos-latest
steps:
- name: Checkout Filament
run: |
git version
git init $GITHUB_WORKSPACE
cd $GITHUB_WORKSPACE
git remote add origin https://github.com/google/filament
git config gc.auto 0
git fetch --tags --prune --progress --no-recurse-submodules origin +${GITHUB_REF}:${GITHUB_REF/refs\//refs\/remote\/}
git checkout --progress --force ${GITHUB_REF/refs\//refs\/remote\/}
- name: Run build script
run: |
cd build/ios && ./build.sh ${TARGET}
env:
TARGET: presubmit
build-web:
name: build-web
runs-on: macos-latest
steps:
- name: Checkout Filament
run: |
git version
git init $GITHUB_WORKSPACE
cd $GITHUB_WORKSPACE
git remote add origin https://github.com/google/filament
git config gc.auto 0
git fetch --tags --prune --progress --no-recurse-submodules origin +${GITHUB_REF}:${GITHUB_REF/refs\//refs\/remote\/}
git checkout --progress --force ${GITHUB_REF/refs\//refs\/remote\/}
- name: Run build script
run: |
cd build/web && ./build.sh ${TARGET}
env:
TARGET: presubmit

4
.gitignore vendored
View File

@@ -3,7 +3,7 @@
imgui.ini
cmake-*
ImportExecutables-*.cmake
out
/out*
dist
dist-*
toolchains
@@ -11,3 +11,5 @@ filament/docs/html/**
.vscode
gltf_baker.ini
*tmp*.png
civetweb.txt
/TAGS

View File

@@ -50,19 +50,22 @@ if (WIN32)
set(CRT_FLAGS_DEBUG "/MDd")
endif()
# TODO: Figure out why pdb generation messes with incremental compilaton.
# IN RELEASE_WITH_DEBUG_INFO, generate debug info in .obj, no in pdb.
set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} ${CRT_FLAGS_RELEASE} /Z7")
set(CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO} ${CRT_FLAGS_RELEASE} /Z7")
if (CMAKE_C_COMPILER_ID MATCHES "Clang")
# TODO: Figure out why pdb generation messes with incremental compilaton.
# IN RELEASE_WITH_DEBUG_INFO, generate debug info in .obj, no in pdb.
set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} ${CRT_FLAGS_RELEASE} /Z7")
set(CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO} ${CRT_FLAGS_RELEASE} /Z7")
# In DEBUG, avoid generating a PDB file which seems to mess with incremental compilation.
# Instead generate debug info directly inside obj files.
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} ${CRT_FLAGS_DEBUG} /Z7")
set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} ${CRT_FLAGS_DEBUG} /Z7")
endif()
# In RELEASE, also generate PDBs.
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} ${CRT_FLAGS_RELEASE} /Zi")
set(CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} ${CRT_FLAGS_RELEASE} /Zi")
# In DEBUG, avoid generating a PDB file which seems to mess with incremental compilation.
# Instead generate debug info directly inside obj files.
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} ${CRT_FLAGS_DEBUG} /Z7")
set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} ${CRT_FLAGS_DEBUG} /Z7")
endif()
# ==================================================================================================
@@ -71,7 +74,7 @@ endif()
# ==================================================================================================
find_package(embree 3.0 QUIET PATHS /usr/lib64/cmake)
if (embree_FOUND AND NOT ANDROID)
if (embree_FOUND AND NOT ANDROID AND NOT IOS)
message("Found embree in ${embree_DIR}")
set(MKLDNN_THREADING "TBB")
include(third_party/OpenImageDenoise/cmake/resource.cmake)
@@ -113,7 +116,7 @@ if (CMAKE_C_COMPILER_ID MATCHES "Clang")
if (CMAKE_C_COMPILER_VERSION VERSION_LESS MIN_CLANG_VERSION)
message(FATAL_ERROR "Detected C compiler Clang ${CMAKE_C_COMPILER_VERSION} < ${MIN_CLANG_VERSION}")
endif()
else()
elseif (NOT MSVC)
message(FATAL_ERROR "Detected C compiler ${CMAKE_C_COMPILER_ID} is unsupported")
endif()
@@ -121,13 +124,17 @@ if (CMAKE_CXX_COMPILER_ID MATCHES "Clang")
if (CMAKE_CXX_COMPILER_VERSION VERSION_LESS MIN_CLANG_VERSION)
message(FATAL_ERROR "Detected CXX compiler Clang ${CMAKE_CXX_COMPILER_VERSION} < ${MIN_CLANG_VERSION}")
endif()
else()
elseif (NOT MSVC)
message(FATAL_ERROR "Detected CXX compiler ${CMAKE_CXX_COMPILER_ID} is unsupported")
endif()
# Detect use of the clang-cl.exe frontend, which does not support all of clangs normal options
if ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "Clang" AND "${CMAKE_CXX_SIMULATE_ID}" STREQUAL "MSVC")
set(CLANG_CL true)
if ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "Clang")
if ("${CMAKE_CXX_SIMULATE_ID}" STREQUAL "MSVC")
set(CLANG_CL true)
endif()
elseif (MSVC)
set(MSVC_NATIVE true)
endif()
# ==================================================================================================
@@ -152,7 +159,12 @@ if (WIN32)
set(CXX_STANDARD "/std:c++14")
endif()
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${CXX_STANDARD} -fstrict-aliasing -Wno-unknown-pragmas -Wno-unused-function")
if (MSVC_NATIVE)
set(CXX_STANDARD "/std:c++latest")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${CXX_STANDARD} /W0 /Zc:__cplusplus")
else()
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${CXX_STANDARD} -fstrict-aliasing -Wno-unknown-pragmas -Wno-unused-function")
endif()
if (USE_EXTERNAL_GLES3)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -DUSE_EXTERNAL_GLES3")
@@ -173,7 +185,7 @@ if (CYGWIN)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fno-exceptions -fno-rtti")
endif()
if (CLANG_CL)
if (CLANG_CL OR MSVC)
# Since the "secure" replacements that MSVC suggests are not portable, disable
# the deprecation warnings. Also disable warnings about use of POSIX functions (i.e. "unlink").
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE")
@@ -189,7 +201,7 @@ endif()
# ==================================================================================================
# Release compiler flags
# ==================================================================================================
if (NOT CLANG_CL)
if (NOT CLANG_CL AND NOT MSVC)
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fomit-frame-pointer -ffunction-sections -fdata-sections")
endif()
@@ -207,7 +219,7 @@ endif()
# -fsanitize=address causes a crash with assimp, which we can't explain for now
#set(EXTRA_SANITIZE_OPTIONS "-fsanitize=undefined -fsanitize=address")
# clang-cl.exe on Windows does not support -fstack-protector.
if (NOT CLANG_CL)
if (NOT CLANG_CL AND NOT MSVC)
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -fstack-protector")
endif()
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} ${EXTRA_SANITIZE_OPTIONS}")
@@ -324,20 +336,47 @@ endif()
# ==================================================================================================
# Functions
# ==================================================================================================
## The MSVC compiler has a limitation on literal string length which is reached when all the
## licenses are concatenated together into a large string... so split them into multiple strings.
function(list_licenses OUTPUT MODULES)
file(WRITE ${OUTPUT} "R\"FILAMENT__(\n")
set(STR_OPENER "R\"FILAMENT__(")
set(STR_CLOSER ")FILAMENT__\"")
set(CONTENT)
set(_MODULES ${MODULES} ${ARGN})
foreach(module ${_MODULES})
set(license_path "../../third_party/${module}/LICENSE")
get_filename_component(fullname "${license_path}" ABSOLUTE)
file(APPEND ${OUTPUT} "License and copyrights for ${module}:\n\n")
file(READ ${license_path} license)
file(APPEND ${OUTPUT} ${license})
file(APPEND ${OUTPUT} "\n\n")
string(APPEND CONTENT "${STR_OPENER}License and copyrights for ${module}:\n${STR_CLOSER},\n")
file(READ ${license_path} license_long)
string(REPLACE "\n" "${STR_CLOSER},\n${STR_OPENER}" license ${license_long})
string(APPEND CONTENT ${STR_OPENER}${license}\n${STR_CLOSER},)
string(APPEND CONTENT "\n\n")
endforeach()
file(APPEND ${OUTPUT} ")FILAMENT__\"\n")
configure_file(${FILAMENT}/build/licenses.inc.in ${OUTPUT})
endfunction(list_licenses)
set(COMBINE_SCRIPT "${CMAKE_CURRENT_SOURCE_DIR}/build/linux/combine-static-libs.sh")
if (WIN32)
set(COMBINE_SCRIPT "${CMAKE_CURRENT_SOURCE_DIR}/build/windows/combine-static-libs.bat")
set(CMAKE_AR "lib.exe")
endif()
# Add a custom command to TARGET that combines the static libraries in DEPS into a single archive.
function(combine_static_libs TARGET OUTPUT DEPS)
# Loop through the dependent libraries and query their location on disk.
set(DEPS_FILES )
foreach(DEPENDENCY ${DEPS})
list(APPEND DEPS_FILES "$<TARGET_FILE:${DEPENDENCY}>")
endforeach()
add_custom_command(
TARGET ${TARGET} POST_BUILD
COMMAND "${COMBINE_SCRIPT}" "${CMAKE_AR}" "${OUTPUT}" ${DEPS_FILES}
COMMENT "Combining ${target} dependencies into single shared library"
VERBATIM
)
endfunction()
# ==================================================================================================
# Configuration for CMAKE_CROSSCOMPILING.
# ==================================================================================================
@@ -423,6 +462,7 @@ add_subdirectory(${EXTERNAL}/meshoptimizer)
add_subdirectory(${EXTERNAL}/cgltf/tnt)
add_subdirectory(${EXTERNAL}/xatlas/tnt)
add_subdirectory(${EXTERNAL}/stb/tnt)
add_subdirectory(${EXTERNAL}/getopt)
if (FILAMENT_BUILD_FILAMAT)
# spirv-tools must come before filamat, as filamat relies on the presence of the
@@ -459,14 +499,15 @@ if (NOT ANDROID AND NOT WEBGL AND NOT IOS)
add_subdirectory(${LIBRARIES}/bluegl)
add_subdirectory(${LIBRARIES}/filagui)
add_subdirectory(${LIBRARIES}/imageio)
add_subdirectory(${LIBRARIES}/matdbg)
add_subdirectory(${FILAMENT}/java/filamat)
add_subdirectory(${FILAMENT}/java/filament)
add_subdirectory(${FILAMENT}/samples)
add_subdirectory(${EXTERNAL}/astcenc/tnt)
add_subdirectory(${EXTERNAL}/civetweb/tnt)
add_subdirectory(${EXTERNAL}/etc2comp)
add_subdirectory(${EXTERNAL}/getopt)
add_subdirectory(${EXTERNAL}/imgui/tnt)
add_subdirectory(${EXTERNAL}/libassimp/tnt)
add_subdirectory(${EXTERNAL}/libpng/tnt)
@@ -474,6 +515,7 @@ if (NOT ANDROID AND NOT WEBGL AND NOT IOS)
add_subdirectory(${EXTERNAL}/libz/tnt)
add_subdirectory(${EXTERNAL}/skylight/tnt)
add_subdirectory(${EXTERNAL}/tinyexr/tnt)
add_subdirectory(${TOOLS}/cmgen)
add_subdirectory(${TOOLS}/filamesh)
add_subdirectory(${TOOLS}/glslminifier)
@@ -485,10 +527,11 @@ if (NOT ANDROID AND NOT WEBGL AND NOT IOS)
add_subdirectory(${TOOLS}/roughness-prefilter)
add_subdirectory(${TOOLS}/skygen)
add_subdirectory(${TOOLS}/specular-color)
endif()
if (DENOISE_LIBRARY)
add_subdirectory(${EXTERNAL}/OpenImageDenoise/tnt)
if (DENOISE_LIBRARY)
add_subdirectory(${EXTERNAL}/OpenImageDenoise/tnt)
endif()
endif()
# Generate exported executables for cross-compiled builds (Android, WebGL, and iOS)

View File

@@ -142,6 +142,7 @@ and tools.
- `ibl`: IBL generation tools
- `image`: Image filtering and simple transforms
- `imageio`: Image file reading / writing, only intended for internal use
- `matdbg`: DebugServer for inspecting shaders at run-time (debug builds only)
- `math`: Math library
- `rays`: Simple path tracer used for baking ambient occlusion, etc.
- `utils`: Utility library (threads, memory, data structures, etc.)
@@ -189,9 +190,9 @@ Building the `rays` library (used for light baking) is optional and requires the
To build Filament for Android you must also install the following:
- Android Studio 3.3
- Android Studio 3.5
- Android SDK
- Android NDK 19 or higher
- Android NDK "side-by-side" 20 or higher
### Environment variables
@@ -248,6 +249,28 @@ $ ./build.sh -j release
If you use CMake directly instead of the build script, pass `-DENABLE_JAVA=OFF` to CMake instead.
### Filament-specific CMake Options
The following CMake options are boolean options specific to Filament:
- `ENABLE_JAVA`: Compile Java projects: requires a JDK and the JAVA_HOME env var
- `ENABLE_LTO`: Enable link-time optimizations if supported by the compiler
- `FILAMENT_BUILD_FILAMAT`: Build filamat and JNI buildings
- `FILAMENT_SUPPORTS_METAL`: Include the Metal backend
- `FILAMENT_SUPPORTS_VULKAN`: Include the Vulkan backend
- `GENERATE_JS_DOCS`: Build WebGL documentation and tutorials
- `INSTALL_BACKEND_TEST`: Install the backend test library so it can be consumed on iOS
- `USE_EXTERNAL_GLES3`: Experimental: Compile Filament against OpenGL ES 3
To turn an option on or off:
```
$ cd <cmake-build-directory>
$ cmake . -DOPTION=ON # Relace OPTION with the option name, set to ON / OFF
```
Options can also be set with the CMake GUI.
### Linux
Make sure you've installed the following dependencies:
@@ -340,6 +363,33 @@ See [ios/samples/README.md](./ios/samples/README.md) for more information.
### Windows
#### Building on Windows with the Visual Studio 2019 compiler
Install the following components:
- [Visual Studio 2019](https://www.visualstudio.com/downloads)
- [Python 3.7](https://www.python.org/ftp/python/3.7.0/python-3.7.0.exe)
- [CMake 3.14 or later](https://github.com/Kitware/CMake/releases/download/v3.14.7/cmake-3.14.7-win64-x64.msi)
Open the `x64 Native Tools Command Prompt for VS 2019`.
Create a working directory, and run cmake in it:
```
> mkdir out
> cd out
> cmake ..
```
Then, you should be able to load the generated solution file `TNT.sln` in Visual Studio and build the `material_sandbox` project.
Run it from the `out` directory with:
```
> samples\Debug\material_sandbox.exe ..\assets\models\monkey\monkey.obj
```
#### Building on Windows with the Clang compiler
The following instructions have been tested on a machine running Windows 10. They should take you
from a machine with only the operating system to a machine able to build and run Filament.

View File

@@ -3,6 +3,61 @@
This file contains one line summaries of commits that are worthy of mentioning in release notes.
A new header is inserted each time a *tag* is created.
## v1.4.2
- Cleaned up the validation strategy in Engine (checks for use-after-destroy etc).
- OpenGL: Fixed ES 3.0 support on iOS.
- OpenGL: Added support for KHR_debug in debug builds.
- gltfio: Added Java / Kotlin bindings for Animator.
- gltfio: Fixed panic with the Android gltf-bloom demo.
- gltfio: Java clients should no longer call Filament#init.
## v1.4.1
- Added missing API documentation.
- Fixed crash for sandboxed macOS apps using Filament.
- Fixed an issue that limited the camera near plane to ~1mm.
- Added Android sample for Camera Stream.
- Fixed an Xcode assertion when rendering skinned meshes using the Metal backend.
- Added support for Core Animation / Metal frame synchronization with Metal backend.
- Fixed an issue with culling in `MaterialInstance`.
- Fix additional compatibility issues with MSVC, including the Vulkan backend.
- matdbg: fixed missing symbol issue when linking against debug builds.
- filamat: fixed crash when using the "lite" version of the library.
- matinfo: Fix a crash with on Windows.
- gltfio: fixed an animation loop bug.
- gltfio: added support for sparse accessors.
- Add JS binding to unary `Camera::setExposure`.
## v1.4.0
- API Breakage: Simplified public-facing Fence API.
- Minimum API level on Android is now API 19 instead of API 21.
- Filament can now be built with msvc 2019.
- Added the ability to modify clip space coordinates in the vertex shader.
- Added missing API documentation.
- Improved existing API documentation.
- Added `Camera::setExposure(float)` to directly control the camera's exposure.
- Backface culling can now be toggled on material instances.
- Face direction is now reversed when transforms have negative scale.
- Dielectrics now behave properly under a white furnace (energy preserving and conserving).
- Clear coat roughness now remains in the 0..1 (previously remapped to the 0..0.6 range).
- gltfio: Fixed several limitations with ubershader mode.
- gltfio: Fixed a transforms issue with non-uniform scale.
- webgl: Fixed an issue with JPEG textures.
- Windows: Fix link error in debug builds.
- matdbg: Web server must now be enabled with an environment variable.
- matdbg: Added support for editing GLSL and MSL code.
## v1.3.2
- Added optional web server for real-time inspection of shader code.
- Added basic #include support in material files.
- Fixed potential Metal memory leak.
- Fixed intermittent memory overflow in wasm builds.
- Fix bad normal mapping with skinning.
- Java clients can now call getNativeObject().
## v1.3.1
- Unified Filament Sceneform and npm releases.

View File

@@ -116,6 +116,7 @@ class MaterialCompiler extends DefaultTask {
// This task handles incremental builds
class IblGenerator extends DefaultTask {
File cmgenPath
String cmgenArgs = null;
@SuppressWarnings("GroovyUnusedDeclaration")
@InputFile
@@ -155,9 +156,13 @@ class IblGenerator extends DefaultTask {
project.exec {
standardOutput out
if (!cmgenArgs) {
cmgenArgs = '--format=rgb32f --extract-blur=0.08 --extract=' + outputDir.absolutePath
}
cmgenArgs = cmgenArgs + " " + file
errorOutput err
executable "${cmgenPath}"
args('--format=rgb32f', '--extract-blur=0.08', "--extract=${outputDir.absolutePath}", file)
args(cmgenArgs.split())
}
}

View File

@@ -9,3 +9,4 @@
/build
/captures
.externalNativeBuild
/.cxx

View File

@@ -13,7 +13,7 @@ buildscript {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:3.4.1'
classpath 'com.android.tools.build:gradle:3.5.0'
}
}
@@ -25,7 +25,7 @@ allprojects {
}
group = "com.google.android.filament"
version = "0.1"
version = "1.3"
apply plugin: 'com.android.library'
@@ -35,10 +35,10 @@ if (project.hasProperty("filament_dist_dir")) {
}
android {
compileSdkVersion 28
compileSdkVersion 29
defaultConfig {
minSdkVersion 14
targetSdkVersion 28
targetSdkVersion 29
versionCode 1
versionName "1.0"
@@ -47,7 +47,7 @@ android {
externalNativeBuild {
cmake {
arguments.add("-DANDROID_PIE=ON")
arguments.add("-DANDROID_PLATFORM=android-21")
arguments.add("-DANDROID_PLATFORM=android-19")
arguments.add("-DANDROID_STL=c++_static")
arguments.add("-DFILAMENT_DIST_DIR=${filament_path}".toString())
cppFlags.add("-std=c++14")

View File

@@ -1,5 +1,5 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4-bin.zip
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

View File

@@ -99,14 +99,15 @@ public class MaterialBuilder {
UV1, // texture coordinates (float2)
BONE_INDICES, // indices of 4 bones (uvec4)
BONE_WEIGHTS, // weights of the 4 bones (normalized float4)
CUSTOM0,
CUSTOM1,
CUSTOM2,
CUSTOM3,
CUSTOM4,
CUSTOM5,
CUSTOM6,
CUSTOM7
UNUSED, // reserved for future use
CUSTOM0, // custom or MORPH_POSITION_0
CUSTOM1, // custom or MORPH_POSITION_1
CUSTOM2, // custom or MORPH_POSITION_2
CUSTOM3, // custom or MORPH_POSITION_3
CUSTOM4, // custom or MORPH_TANGENTS_0
CUSTOM5, // custom or MORPH_TANGENTS_1
CUSTOM6, // custom or MORPH_TANGENTS_2
CUSTOM7 // custom or MORPH_TANGENTS_3
}
public enum BlendingMode {

View File

@@ -9,3 +9,4 @@
/build
/captures
.externalNativeBuild
/.cxx

View File

@@ -1,29 +1,113 @@
<component name="ProjectCodeStyleConfiguration">
<code_scheme name="Project" version="173">
<Objective-C-extensions>
<file>
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Import" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Macro" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Typedef" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Enum" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Constant" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Global" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Struct" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="FunctionPredecl" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Function" />
</file>
<class>
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Property" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Synthesize" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="InitMethod" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="StaticMethod" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="InstanceMethod" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="DeallocMethod" />
</class>
<extensions>
<pair source="cpp" header="h" fileNamingConvention="NONE" />
<pair source="c" header="h" fileNamingConvention="NONE" />
</extensions>
</Objective-C-extensions>
<codeStyleSettings language="XML">
<arrangement>
<rules>
<section>
<rule>
<match>
<AND>
<NAME>xmlns:android</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>xmlns:.*</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:id</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:name</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>name</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>style</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>ANDROID_ATTRIBUTE_ORDER</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_ATTRIBUTE />
<XML_NAMESPACE>.*</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
</rules>
</arrangement>
</codeStyleSettings>
</code_scheme>
</component>

View File

@@ -50,57 +50,66 @@ set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fomit-frame-pointer -ff
set(CMAKE_SHARED_LINKER_FLAGS" ${CMAKE_SHARED_LINKER_FLAGS} -Wl,--gc-sections")
set(CMAKE_SHARED_LINKER_FLAGS" ${CMAKE_SHARED_LINKER_FLAGS} -Wl,-Bsymbolic-functions")
set(CMAKE_SHARED_LINKER_FLAGS_RELEASE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE} -Wl,--version-script=${CMAKE_SOURCE_DIR}/libfilament-jni.map")
add_library(filament-jni SHARED
src/main/cpp/Camera.cpp
src/main/cpp/Colors.cpp
src/main/cpp/VertexBuffer.cpp
src/main/cpp/Engine.cpp
src/main/cpp/EntityManager.cpp
src/main/cpp/Fence.cpp
src/main/cpp/IndexBuffer.cpp
src/main/cpp/IndirectLight.cpp
src/main/cpp/LightManager.cpp
src/main/cpp/Material.cpp
src/main/cpp/MaterialInstance.cpp
src/main/cpp/MathUtils.cpp
src/main/cpp/RenderableManager.cpp
src/main/cpp/Renderer.cpp
src/main/cpp/RenderTarget.cpp
src/main/cpp/Scene.cpp
src/main/cpp/SkyBox.cpp
src/main/cpp/Stream.cpp
src/main/cpp/Texture.cpp
src/main/cpp/TextureSampler.cpp
src/main/cpp/TransformManager.cpp
src/main/cpp/View.cpp
# Android specific
src/main/cpp/nativewindow/Android.cpp
# Private utils
src/main/cpp/Filament.cpp
# Common utils
../common/CallbackUtils.cpp
../common/NioUtils.cpp
add_library(filament-jni-obj OBJECT
src/main/cpp/Camera.cpp
src/main/cpp/Colors.cpp
src/main/cpp/VertexBuffer.cpp
src/main/cpp/Engine.cpp
src/main/cpp/EntityManager.cpp
src/main/cpp/Fence.cpp
src/main/cpp/IndexBuffer.cpp
src/main/cpp/IndirectLight.cpp
src/main/cpp/LightManager.cpp
src/main/cpp/Material.cpp
src/main/cpp/MaterialInstance.cpp
src/main/cpp/MathUtils.cpp
src/main/cpp/RenderableManager.cpp
src/main/cpp/Renderer.cpp
src/main/cpp/RenderTarget.cpp
src/main/cpp/Scene.cpp
src/main/cpp/SkyBox.cpp
src/main/cpp/Stream.cpp
src/main/cpp/Texture.cpp
src/main/cpp/TextureSampler.cpp
src/main/cpp/TransformManager.cpp
src/main/cpp/View.cpp
# Android specific
src/main/cpp/nativewindow/Android.cpp
)
target_link_libraries(filament-jni
filament
backend
filaflat
filabridge
geometry
ibl
utils
log
GLESv3
EGL
android
jnigraphics
)
if (NOT DISABLE_FILAMENT_JNI)
option(FILAMENT_SUPPORTS_VULKAN "Enables Vulkan on Android" OFF)
set(CMAKE_SHARED_LINKER_FLAGS_RELEASE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE} -Wl,--version-script=${CMAKE_CURRENT_SOURCE_DIR}/libfilament-jni.map")
add_library(filament-jni SHARED
$<TARGET_OBJECTS:filament-jni-obj>
# Private utils
src/main/cpp/Filament.cpp
# Common utils
../common/CallbackUtils.cpp
../common/NioUtils.cpp
)
target_link_libraries(filament-jni
filament
backend
filaflat
filabridge
geometry
ibl
utils
log
GLESv3
EGL
android
jnigraphics
)
option(FILAMENT_SUPPORTS_VULKAN "Enables Vulkan on Android" OFF)
if (FILAMENT_SUPPORTS_VULKAN)
target_link_libraries(filament-jni bluevk smol-v)
endif()
if (FILAMENT_SUPPORTS_VULKAN)
target_link_libraries(filament-jni bluevk smol-v)
endif()

View File

@@ -13,7 +13,7 @@ buildscript {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:3.4.1'
classpath 'com.android.tools.build:gradle:3.5.0'
}
}
@@ -25,7 +25,7 @@ allprojects {
}
group = "com.google.android.filament"
version = "0.1"
version = "1.3"
apply plugin: 'com.android.library'
@@ -35,13 +35,13 @@ if (project.hasProperty("filament_dist_dir")) {
}
android {
compileSdkVersion 28
compileSdkVersion 29
defaultConfig {
// Our minSdkVersion is actually 21, we lie and say 14 here so apps don't have
// Our minSdkVersion is actually 19, we lie and say 14 here so apps don't have
// to increase their minSdkVersion unnecessarily. It is however up to them to
// ensure they do not initialize Filament on API levels < 21.
// ensure they do not initialize Filament on API levels < 19.
minSdkVersion 14
targetSdkVersion 28
targetSdkVersion 29
versionCode 1
versionName "1.0"
@@ -50,7 +50,7 @@ android {
externalNativeBuild {
cmake {
arguments.add("-DANDROID_PIE=ON")
arguments.add("-DANDROID_PLATFORM=android-21")
arguments.add("-DANDROID_PLATFORM=android-19")
arguments.add("-DANDROID_STL=c++_static")
arguments.add("-DFILAMENT_DIST_DIR=${filament_path}".toString())
cppFlags.add("-std=c++14")

View File

@@ -1,5 +1,5 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4-bin.zip
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

View File

@@ -154,9 +154,9 @@ Java_com_google_android_filament_Engine_nDestroyScene(JNIEnv*, jclass,
extern "C" JNIEXPORT jlong JNICALL
Java_com_google_android_filament_Engine_nCreateFence(JNIEnv*, jclass,
jlong nativeEngine, jint fenceType) {
jlong nativeEngine) {
Engine* engine = (Engine*) nativeEngine;
return (jlong) engine->createFence((Fence::Type) fenceType);
return (jlong) engine->createFence();
}
extern "C" JNIEXPORT void JNICALL
@@ -252,6 +252,13 @@ Java_com_google_android_filament_Engine_nDestroyEntity(JNIEnv*, jclass,
engine->destroy(entity);
}
extern "C" JNIEXPORT void JNICALL
Java_com_google_android_filament_Engine_nFlushAndWait(JNIEnv *env, jclass clazz,
jlong nativeEngine) {
Engine* engine = (Engine*) nativeEngine;
engine->flushAndWait();
}
// Managers...
extern "C" JNIEXPORT jlong JNICALL

View File

@@ -316,3 +316,11 @@ Java_com_google_android_filament_MaterialInstance_nSetDoubleSided(JNIEnv*,
MaterialInstance* instance = (MaterialInstance*) nativeMaterialInstance;
instance->setDoubleSided(doubleSided);
}
extern "C"
JNIEXPORT void JNICALL
Java_com_google_android_filament_MaterialInstance_nSetCullingMode(JNIEnv*,
jclass, jlong nativeMaterialInstance, jlong cullingMode) {
MaterialInstance* instance = (MaterialInstance*) nativeMaterialInstance;
instance->setCullingMode((MaterialInstance::CullingMode) cullingMode);
}

View File

@@ -47,6 +47,13 @@ Java_com_google_android_filament_Skybox_nBuilderShowSun(JNIEnv *env, jclass type
builder->showSun(show);
}
extern "C" JNIEXPORT void JNICALL
Java_com_google_android_filament_Skybox_nBuilderIntensity(JNIEnv *env, jclass clazz,
jlong nativeSkyBoxBuilder, jfloat intensity) {
Skybox::Builder *builder = (Skybox::Builder *) nativeSkyBoxBuilder;
builder->intensity(intensity);
}
extern "C" JNIEXPORT jlong JNICALL
Java_com_google_android_filament_Skybox_nBuilderBuild(JNIEnv *env, jclass type,
jlong nativeSkyBoxBuilder, jlong nativeEngine) {
@@ -68,3 +75,10 @@ Java_com_google_android_filament_Skybox_nGetLayerMask(JNIEnv *env, jclass type,
Skybox *skybox = (Skybox *) nativeSkybox;
return static_cast<jint>(skybox->getLayerMask());
}
extern "C" JNIEXPORT jfloat JNICALL
Java_com_google_android_filament_Skybox_nGetIntensity(JNIEnv *env, jclass clazz,
jlong nativeSkybox) {
Skybox *skybox = (Skybox *) nativeSkybox;
return static_cast<jint>(skybox->getIntensity());
}

View File

@@ -177,7 +177,7 @@ Java_com_google_android_filament_Texture_nSetImage(JNIEnv* env, jclass, jlong na
(Texture::Type) type, (size_t) stride, (size_t) alignment);
AutoBuffer nioBuffer(env, storage, 0);
if (sizeInBytes > (remaining << nioBuffer.getShift())) {
if (sizeInBytes > (size_t(remaining) << nioBuffer.getShift())) {
// BufferOverflowException
return -1;
}
@@ -199,8 +199,7 @@ extern "C" JNIEXPORT jint JNICALL
Java_com_google_android_filament_Texture_nSetImageCompressed(JNIEnv *env, jclass,
jlong nativeTexture, jlong nativeEngine, jint level, jint xoffset, jint yoffset,
jint width, jint height, jobject storage, jint remaining,
jint left, jint bottom, jint type, jint alignment,
jint compressedSizeInBytes, jint compressedFormat,
jint, jint, jint, jint, jint compressedSizeInBytes, jint compressedFormat,
jobject handler, jobject runnable) {
Texture *texture = (Texture *) nativeTexture;
Engine *engine = (Engine *) nativeEngine;
@@ -208,7 +207,7 @@ Java_com_google_android_filament_Texture_nSetImageCompressed(JNIEnv *env, jclass
size_t sizeInBytes = (size_t) compressedSizeInBytes;
AutoBuffer nioBuffer(env, storage, 0);
if (sizeInBytes > (remaining << nioBuffer.getShift())) {
if (sizeInBytes > (size_t(remaining) << nioBuffer.getShift())) {
// BufferOverflowException
return -1;
}
@@ -235,7 +234,7 @@ Java_com_google_android_filament_Texture_nSetImageCubemap(JNIEnv *env, jclass,
Texture *texture = (Texture *) nativeTexture;
Engine *engine = (Engine *) nativeEngine;
jint *faceOffsetsInBytes = env->GetIntArrayElements(faceOffsetsInBytes_, NULL);
jint *faceOffsetsInBytes = env->GetIntArrayElements(faceOffsetsInBytes_, nullptr);
Texture::FaceOffsets faceOffsets;
std::copy_n(faceOffsetsInBytes, 6, faceOffsets.offsets);
env->ReleaseIntArrayElements(faceOffsetsInBytes_, faceOffsetsInBytes, JNI_ABORT);
@@ -244,7 +243,7 @@ Java_com_google_android_filament_Texture_nSetImageCubemap(JNIEnv *env, jclass,
(Texture::Type) type, (size_t) stride, (size_t) alignment);
AutoBuffer nioBuffer(env, storage, 0);
if (sizeInBytes > (remaining << nioBuffer.getShift())) {
if (sizeInBytes > (size_t(remaining) << nioBuffer.getShift())) {
// BufferOverflowException
return -1;
}
@@ -271,7 +270,7 @@ Java_com_google_android_filament_Texture_nSetImageCubemapCompressed(JNIEnv *env,
Texture *texture = (Texture *) nativeTexture;
Engine *engine = (Engine *) nativeEngine;
jint *faceOffsetsInBytes = env->GetIntArrayElements(faceOffsetsInBytes_, NULL);
jint *faceOffsetsInBytes = env->GetIntArrayElements(faceOffsetsInBytes_, nullptr);
Texture::FaceOffsets faceOffsets;
std::copy_n(faceOffsetsInBytes, 6, faceOffsets.offsets);
env->ReleaseIntArrayElements(faceOffsetsInBytes_, faceOffsetsInBytes, JNI_ABORT);
@@ -279,7 +278,7 @@ Java_com_google_android_filament_Texture_nSetImageCubemapCompressed(JNIEnv *env,
size_t sizeInBytes = 6 * (size_t) compressedSizeInBytes;
AutoBuffer nioBuffer(env, storage, 0);
if (sizeInBytes > (remaining << nioBuffer.getShift())) {
if (sizeInBytes > (size_t(remaining) << nioBuffer.getShift())) {
// BufferOverflowException
return -1;
}
@@ -340,28 +339,29 @@ Java_com_google_android_filament_Texture_nGeneratePrefilterMipmap(JNIEnv *env, j
Texture *texture = (Texture *) nativeTexture;
Engine *engine = (Engine *) nativeEngine;
jint *faceOffsetsInBytes = env->GetIntArrayElements(faceOffsetsInBytes_, NULL);
jint *faceOffsetsInBytes = env->GetIntArrayElements(faceOffsetsInBytes_, nullptr);
Texture::FaceOffsets faceOffsets;
std::copy_n(faceOffsetsInBytes, 6, faceOffsets.offsets);
env->ReleaseIntArrayElements(faceOffsetsInBytes_, faceOffsetsInBytes, JNI_ABORT);
stride = stride ? stride : width;
size_t sizeInBytes = 6 *
Texture::computeTextureDataSize((Texture::Format) format, (Texture::Type) type,
(size_t) stride, (size_t) height, (size_t) alignment);
Texture::computeTextureDataSize((Texture::Format) format, (Texture::Type) type,
(size_t) stride, (size_t) height, (size_t) alignment);
AutoBuffer nioBuffer(env, storage, 0);
if (sizeInBytes > (remaining << nioBuffer.getShift())) {
if (sizeInBytes > (size_t(remaining) << nioBuffer.getShift())) {
// BufferOverflowException
return -1;
}
void *buffer = nioBuffer.getData();
auto *callback = JniBufferCallback::make(engine, env, handler, runnable, std::move(nioBuffer));
void* buffer = nioBuffer.getData();
auto* callback = JniBufferCallback::make(engine, env, handler, runnable, std::move(nioBuffer));
Texture::PixelBufferDescriptor desc(buffer, sizeInBytes, (backend::PixelDataFormat) format,
(backend::PixelDataType) type, (uint8_t) alignment, (uint32_t)0, (uint32_t)0,
(uint32_t) stride, &JniBufferCallback::invoke, callback);
(backend::PixelDataType) type, (uint8_t) alignment,
(uint32_t) left, (uint32_t) top, (uint32_t) stride,
&JniBufferCallback::invoke, callback);
Texture::PrefilterOptions options;
options.sampleCount = sampleCount;

View File

@@ -19,8 +19,10 @@ package com.google.android.filament;
import android.graphics.SurfaceTexture;
import android.opengl.EGL14;
import android.opengl.EGLContext;
import android.os.Build;
import android.util.Log;
import android.view.Surface;
import java.lang.reflect.Method;
final class AndroidPlatform extends Platform {
private static final String LOG_TAG = "Filament";
@@ -59,6 +61,20 @@ final class AndroidPlatform extends Platform {
@Override
long getSharedContextNativeHandle(Object sharedContext) {
return ((EGLContext) sharedContext).getNativeHandle();
if (Build.VERSION.SDK_INT >= 21) {
return AndroidPlatform21.getSharedContextNativeHandle(sharedContext);
} else {
try {
//noinspection JavaReflectionMemberAccess
Method method = EGLContext.class.getDeclaredMethod("getHandle");
Integer handle = (Integer) method.invoke(sharedContext);
//noinspection ConstantConditions
return handle.longValue();
} catch (Exception e) {
Log.d(LOG_TAG, "Could not access shared context's native handle", e);
}
// Should not happen
return 0;
}
}
}

View File

@@ -0,0 +1,25 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.google.android.filament;
import android.opengl.EGLContext;
final class AndroidPlatform21 {
static long getSharedContextNativeHandle(Object sharedContext) {
return ((EGLContext) sharedContext).getNativeHandle();
}
}

View File

@@ -16,7 +16,6 @@
package com.google.android.filament;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.Size;

View File

@@ -19,12 +19,25 @@ package com.google.android.filament;
import android.support.annotation.NonNull;
import android.support.annotation.Size;
/**
* An axis-aligned 3D box represented by its center and half-extent.
*
* The half-extent is a vector representing the distance from the center to the edge of the box in
* each dimension. For example, a box of size 2 units in X, 4 units in Y, and 10 units in Z would
* have a half-extent of (1, 2, 5).
*/
public class Box {
private final float[] mCenter = new float[3];
private final float[] mHalfExtent = new float[3];
/**
* Default-initializes the 3D box to have a center and half-extent of (0,0,0).
*/
public Box() { }
/**
* Initializes the 3D box from its center and half-extent.
*/
public Box(float centerX, float centerY, float centerZ,
float halfExtentX, float halfExtentY, float halfExtentZ) {
mCenter[0] = centerX;
@@ -35,6 +48,13 @@ public class Box {
mHalfExtent[2] = halfExtentZ;
}
/**
* Initializes the 3D box from its center and half-extent.
*
* @param center a float array with XYZ coordinates representing the center of the box
* @param halfExtent a float array with XYZ coordinates representing half the size of the box in
* each dimension
*/
public Box(@NonNull @Size(min = 3) float[] center, @NonNull @Size(min = 3) float[] halfExtent) {
mCenter[0] = center[0];
mCenter[1] = center[1];
@@ -44,21 +64,37 @@ public class Box {
mHalfExtent[2] = halfExtent[2];
}
/**
* Sets the center of of the 3D box.
*/
public void setCenter(float centerX, float centerY, float centerZ) {
mCenter[0] = centerX;
mCenter[1] = centerY;
mCenter[2] = centerZ;
}
/**
* Sets the half-extent of the 3D box.
*/
public void setHalfExtent(float halfExtentX, float halfExtentY, float halfExtentZ) {
mHalfExtent[0] = halfExtentX;
mHalfExtent[1] = halfExtentY;
mHalfExtent[2] = halfExtentZ;
}
/**
* Returns the center of the 3D box.
*
* @return an XYZ float array of size 3
*/
@NonNull @Size(min = 3)
public float[] getCenter() { return mCenter; }
/**
* Returns the half-extent from the center of the 3D box.
*
* @return an XYZ float array of size 3
*/
@NonNull @Size(min = 3)
public float[] getHalfExtent() { return mHalfExtent; }
}

View File

@@ -157,26 +157,30 @@ public class Camera {
* @param projection type of projection to use
*
* @param left distance in world units from the camera to the left plane,
* at the near plane. Precondition: left != right.
* at the near plane. Precondition: <code>left</code> != <code>right</code>
*
* @param right distance in world units from the camera to the right plane,
* at the near plane. Precondition: left != right.
* at the near plane. Precondition: <code>left</code> != <code>right</code>
*
* @param bottom distance in world units from the camera to the bottom plane,
* at the near plane. Precondition: bottom != top.
* at the near plane. Precondition: <code>bottom</code> != <code>top</code>
*
* @param top distance in world units from the camera to the top plane,
* at the near plane. Precondition: left != right.
* at the near plane. Precondition: <code>bottom</code> != <code>top</code>
*
* @param near distance in world units from the camera to the near plane.
* The near plane's position in view space is z = -near.
* Precondition: near > 0 for {@link Projection#PERSPECTIVE} or
* near != far for {@link Projection#ORTHO}.
* The near plane's position in view space is z = -<code>near</code>.
* Precondition:
* <code>near</code> > 0 for {@link Projection#PERSPECTIVE} or
* <code>near</code> != <code>far</code> for {@link Projection#ORTHO}.
*
* @param far distance in world units from the camera to the far plane.
* The far plane's position in view space is z = -far.
* Precondition: far > near for {@link Projection#PERSPECTIVE} or
* far != near for {@link Projection#ORTHO}.
* The far plane's position in view space is z = -<code>far</code>.
* Precondition:
* <code>far</code> > <code>near</code>
* for {@link Projection#PERSPECTIVE} or
* <code>far</code> != <code>near</code>
* for {@link Projection#ORTHO}.
*
* <p>
* These parameters are silently modified to meet the preconditions above.
@@ -189,12 +193,32 @@ public class Camera {
}
/**
* Sets the projection matrix from the field-of-view.
*
* @param fovInDegrees
* @param aspect
* @param near
* @param far
* @param direction
* @param fovInDegrees field-of-view in degrees from the camera center axis.
* 0 < <code>fovInDegrees</code> < 180
*
* @param aspect aspect ratio width/height. <code>aspect</code> > 0
*
* @param near distance in world units from the camera to the near plane.
* The near plane's position in view space is z = -<code>near</code>.
* Precondition:
* <code>near</code> > 0 for {@link Projection#PERSPECTIVE} or
* <code>near</code> != <code>far</code> for {@link Projection#ORTHO}.
*
* @param far distance in world units from the camera to the far plane.
* The far plane's position in view space is z = -<code>far</code>.
* Precondition:
* <code>far</code> > <code>near</code>
* for {@link Projection#PERSPECTIVE} or
* <code>far</code> != <code>near</code>
* for {@link Projection#ORTHO}.
*
* @param direction direction of the field-of-view parameter.
* <p>
* These parameters are silently modified to meet the preconditions above.
*
* @see Fov
*/
public void setProjection(double fovInDegrees, double aspect, double near, double far,
@NonNull Fov direction) {
@@ -202,47 +226,86 @@ public class Camera {
}
/**
* Sets the projection matrix from the focal length
*
* @param focalLength lense's focal length in millimeters. <code>focalLength</code> > 0
*
* @param near distance in world units from the camera to the near plane.
* The near plane's position in view space is z = -<code>near</code>.
* Precondition:
* <code>near</code> > 0 for {@link Projection#PERSPECTIVE} or
* <code>near</code> != <code>far</code> for {@link Projection#ORTHO}.
*
* @param far distance in world units from the camera to the far plane.
* The far plane's position in view space is z = -<code>far</code>.
* Precondition:
* <code>far</code> > <code>near</code>
* for {@link Projection#PERSPECTIVE} or
* <code>far</code> != <code>near</code>
* for {@link Projection#ORTHO}.
*
* @param focalLength
* @param near
* @param far
*/
public void setLensProjection(double focalLength, double near, double far) {
nSetLensProjection(getNativeObject(), focalLength, near, far);
}
/**
* Sets the projection matrix.
*
* @param inMatrix
* @param near
* @param far
* @param inMatrix custom projection matrix.
*
* @param near distance in world units from the camera to the near plane.
* The near plane's position in view space is z = -<code>near</code>.
* Precondition:
* <code>near</code> > 0 for {@link Projection#PERSPECTIVE} or
* <code>near</code> != <code>far</code> for {@link Projection#ORTHO}.
*
* @param far distance in world units from the camera to the far plane.
* The far plane's position in view space is z = -<code>far</code>.
* Precondition:
* <code>far</code> > <code>near</code>
* for {@link Projection#PERSPECTIVE} or
* <code>far</code> != <code>near</code>
* for {@link Projection#ORTHO}.
*/
public void setCustomProjection(@NonNull @Size(min = 16) double inMatrix[],
public void setCustomProjection(@NonNull @Size(min = 16) double[] inMatrix,
double near, double far) {
Asserts.assertMat4dIn(inMatrix);
nSetCustomProjection(getNativeObject(), inMatrix, near, far);
}
/**
* Sets the camera's view matrix.
* <p>
* Helper method to set the camera's entity transform component.
* Remember that the Camera "looks" towards its -z axis.
* <p>
* This has the same effect as calling:
*
* @param in
* <pre>
* engine.getTransformManager().setTransform(
* engine.getTransformManager().getInstance(camera->getEntity()), viewMatrix);
* </pre>
*
* @param viewMatrix The camera position and orientation provided as a <b>rigid transform</b> matrix.
*/
public void setModelMatrix(@NonNull @Size(min = 16) float in[]) {
Asserts.assertMat4fIn(in);
nSetModelMatrix(getNativeObject(), in);
public void setModelMatrix(@NonNull @Size(min = 16) float[] viewMatrix) {
Asserts.assertMat4fIn(viewMatrix);
nSetModelMatrix(getNativeObject(), viewMatrix);
}
/**
* Sets the camera's view matrix.
*
* @param eyeX
* @param eyeY
* @param eyeZ
* @param centerX
* @param centerY
* @param centerZ
* @param upX
* @param upY
* @param upZ
* @param eyeX x-axis position of the camera in world space
* @param eyeY y-axis position of the camera in world space
* @param eyeZ z-axis position of the camera in world space
* @param centerX x-axis position of the point in world space the camera is looking at
* @param centerY y-axis position of the point in world space the camera is looking at
* @param centerZ z-axis position of the point in world space the camera is looking at
* @param upX x-axis coordinate of a unit vector denoting the camera's "up" direction
* @param upY y-axis coordinate of a unit vector denoting the camera's "up" direction
* @param upZ z-axis coordinate of a unit vector denoting the camera's "up" direction
*/
public void lookAt(double eyeX, double eyeY, double eyeZ,
double centerX, double centerY, double centerZ, double upX, double upY, double upZ) {
@@ -250,16 +313,14 @@ public class Camera {
}
/**
*
* @return Distance to the near plane.
* @return Distance to the near plane
*/
public float getNear() {
return nGetNear(getNativeObject());
}
/**
*
* @return Distance to the far plane.
* @return Distance to the far plane
*/
public float getCullingFar() {
return nGetCullingFar(getNativeObject());
@@ -267,12 +328,14 @@ public class Camera {
/**
* Retrieves the camera's projection matrix.
*
* @param out A 16-float array where the projection matrix will be stored, or null in which
* case a new array is allocated.
*
* @return A 16-float array containing the camera's projection as a column-major matrix.
*/
@NonNull @Size(min = 16)
public double[] getProjectionMatrix(@Nullable @Size(min = 16) double out[]) {
public double[] getProjectionMatrix(@Nullable @Size(min = 16) double[] out) {
out = Asserts.assertMat4d(out);
nGetProjectionMatrix(getNativeObject(), out);
return out;
@@ -281,12 +344,14 @@ public class Camera {
/**
* Retrieves the camera's model matrix. The model matrix encodes the camera position and
* orientation, or pose.
*
* @param out A 16-float array where the model matrix will be stored, or null in which
* case a new array is allocated.
*
* @return A 16-float array containing the camera's pose as a column-major matrix.
*/
@NonNull @Size(min = 16)
public float[] getModelMatrix(@Nullable @Size(min = 16) float out[]) {
public float[] getModelMatrix(@Nullable @Size(min = 16) float[] out) {
out = Asserts.assertMat4f(out);
nGetModelMatrix(getNativeObject(), out);
return out;
@@ -297,10 +362,11 @@ public class Camera {
*
* @param out A 16-float array where the model view will be stored, or null in which
* case a new array is allocated.
*
* @return A 16-float array containing the camera's view as a column-major matrix.
*/
@NonNull @Size(min = 16)
public float[] getViewMatrix(@Nullable @Size(min = 16) float out[]) {
public float[] getViewMatrix(@Nullable @Size(min = 16) float[] out) {
out = Asserts.assertMat4f(out);
nGetViewMatrix(getNativeObject(), out);
return out;
@@ -308,12 +374,14 @@ public class Camera {
/**
* Retrieves the camera position in world space.
*
* @param out A 3-float array where the position will be stored, or null in which case a new
* array is allocated.
*
* @return A 3-float array containing the camera's position in world units.
*/
@NonNull @Size(min = 3)
public float[] getPosition(@Nullable @Size(min = 3) float out[]) {
public float[] getPosition(@Nullable @Size(min = 3) float[] out) {
out = Asserts.assertFloat3(out);
nGetPosition(getNativeObject(), out);
return out;
@@ -322,12 +390,14 @@ public class Camera {
/**
* Retrieves the camera left unit vector in world space, that is a unit vector that points to
* the left of the camera.
*
* @param out A 3-float array where the left vector will be stored, or null in which case a new
* array is allocated.
*
* @return A 3-float array containing the camera's left vector in world units.
*/
@NonNull @Size(min = 3)
public float[] getLeftVector(@Nullable @Size(min = 3) float out[]) {
public float[] getLeftVector(@Nullable @Size(min = 3) float[] out) {
out = Asserts.assertFloat3(out);
nGetLeftVector(getNativeObject(), out);
return out;
@@ -336,12 +406,14 @@ public class Camera {
/**
* Retrieves the camera up unit vector in world space, that is a unit vector that points up with
* respect to the camera.
*
* @param out A 3-float array where the up vector will be stored, or null in which case a new
* array is allocated.
*
* @return A 3-float array containing the camera's up vector in world units.
*/
@NonNull @Size(min = 3)
public float[] getUpVector(@Nullable @Size(min = 3) float out[]) {
public float[] getUpVector(@Nullable @Size(min = 3) float[] out) {
out = Asserts.assertFloat3(out);
nGetUpVector(getNativeObject(), out);
return out;
@@ -350,52 +422,89 @@ public class Camera {
/**
* Retrieves the camera forward unit vector in world space, that is a unit vector that points
* in the direction the camera is looking at.
*
* @param out A 3-float array where the forward vector will be stored, or null in which case a
* new array is allocated.
*
* @return A 3-float array containing the camera's forward vector in world units.
*/
@NonNull @Size(min = 3)
public float[] getForwardVector(@Nullable @Size(min = 3) float out[]) {
public float[] getForwardVector(@Nullable @Size(min = 3) float[] out) {
out = Asserts.assertFloat3(out);
nGetForwardVector(getNativeObject(), out);
return out;
}
/**
* Sets this camera's exposure (default is f/16, 1/125s, 100 ISO)
*
* @param aperture
* @param shutterSpeed
* @param sensitivity
* The exposure ultimately controls the scene's brightness, just like with a real camera.
* The default values provide adequate exposure for a camera placed outdoors on a sunny day
* with the sun at the zenith.
*
* With the default parameters, the scene must contain at least one Light of intensity
* similar to the sun (e.g.: a 100,000 lux directional light) and/or an indirect light
* of appropriate intensity (30,000).
*
* @param aperture Aperture in f-stops, clamped between 0.5 and 64.
* A lower aperture value increases the exposure, leading to
* a brighter scene. Realistic values are between 0.95 and 32.
*
* @param shutterSpeed Shutter speed in seconds, clamped between 1/25,000 and 60.
* A lower shutter speed increases the exposure. Realistic values are
* between 1/8000 and 30.
*
* @param sensitivity Sensitivity in ISO, clamped between 10 and 204,800.
* A higher sensitivity increases the exposure. Realistic values are
* between 50 and 25600.
*
* @see LightManager
* @see #setExposure(float)
*/
public void setExposure(float aperture, float shutterSpeed, float sensitivity) {
nSetExposure(getNativeObject(), aperture, shutterSpeed, sensitivity);
}
/**
* Sets this camera's exposure directly. Calling this method will set the aperture
* to 1.0, the shutter speed to 1.2 and the sensitivity will be computed to match
* the requested exposure (for a desired exposure of 1.0, the sensitivity will be
* set to 100 ISO).
*
* @return
* This method is useful when trying to match the lighting of other engines or tools.
* Many engines/tools use unit-less light intensities, which can be matched by setting
* the exposure manually. This can be typically achieved by setting the exposure to
* 1.0.
*
* @see LightManager
* @see #setExposure(float, float, float)
*/
public void setExposure(float exposure) {
setExposure(1.0f, 1.2f, 100.0f * (1.0f / exposure));
}
/**
* @return Aperture in f-stops
*/
public float getAperture() {
return nGetAperture(getNativeObject());
}
/**
*
* @return
* @return Shutter speed in seconds
*/
public float getShutterSpeed() {
return nGetShutterSpeed(getNativeObject());
}
/**
*
* @return
* @return Sensitivity in ISO
*/
public float getSensitivity() {
return nGetSensitivity(getNativeObject());
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Camera");
}

View File

@@ -28,6 +28,9 @@ import static java.lang.annotation.ElementType.METHOD;
import static java.lang.annotation.ElementType.PARAMETER;
import static java.lang.annotation.RetentionPolicy.SOURCE;
/**
* Utilities to manipulate and convert colors.
*/
public class Colors {
private Colors() {
}
@@ -37,23 +40,67 @@ public class Colors {
public @interface LinearColor {
}
/**
* Types of RGB colors.
*/
public enum RgbType {
/** The color is defined in sRGB space. */
SRGB,
/** The color is defined in linear space. */
LINEAR
}
/**
* Types of RGBA colors.
*/
public enum RgbaType {
/**
* The color is defined in sRGB space and the RGB values have not been premultiplied by the
* alpha (for instance, a 50% transparent red is <1,0,0,0.5>).
*/
SRGB,
/**
* The color is defined in linear space and the RGB values have not been premultiplied by
* the alpha (for instance, a 50% transparent red is <1,0,0,0.5>).
*/
LINEAR,
/**
* The color is defined in sRGB space and the RGB values have been premultiplied by the
* alpha (for instance, a 50% transparent red is <0.5,0,0,0.5>).
*/
PREMULTIPLIED_SRGB,
/**
* The color is defined in linear space and the RGB values have been premultiplied by the
* alpha (for instance, a 50% transparent red is <0.5,0,0,0.5>).
*/
PREMULTIPLIED_LINEAR
}
/**
* Type of color conversion to use when converting to/from sRGB and linear spaces.
*/
public enum Conversion {
/** Accurate conversion using the sRGB standard. */
ACCURATE,
/** Fast conversion using a simple gamma 2.2 curve. */
FAST
}
/**
* Converts an RGB color to linear space, the conversion depends on the specified type.
*
* @param type the color space of the RGB color values provided
* @param r the red component
* @param g the green component
* @param b the blue component
*
* @return an RGB float array of size 3 with the result of the conversion
*/
@NonNull
@Size(3)
@LinearColor
@@ -61,6 +108,14 @@ public class Colors {
return toLinear(type, new float[] { r, g, b });
}
/**
* Converts an RGB color to linear space, the conversion depends on the specified type.
*
* @param type the color space of the RGB color values provided
* @param rgb an RGB float array of size 3, will be modified
*
* @return the passed-in <code>rgb</code> array, after applying the conversion
*/
@NonNull
@Size(min = 3)
@LinearColor
@@ -68,6 +123,17 @@ public class Colors {
return (type == RgbType.LINEAR) ? rgb : toLinear(Conversion.ACCURATE, rgb);
}
/**
* Converts an RGBA color to linear space, with pre-multiplied alpha.
*
* @param type the color space and type of RGBA color values provided
* @param r the red component
* @param g the green component
* @param b the blue component
* @param a the alpha component
*
* @return an RGBA float array of size 4 with the result of the conversion
*/
@NonNull
@Size(4)
@LinearColor
@@ -75,6 +141,14 @@ public class Colors {
return toLinear(type, new float[] { r, g, b, a });
}
/**
* Converts an RGBA color to linear space, with pre-multiplied alpha.
*
* @param type the color space of the RGBA color values provided
* @param rgba an RGBA float array of size 4, will be modified
*
* @return the passed-in <code>rgba</code> array, after applying the conversion
*/
@NonNull
@Size(min = 4)
@LinearColor
@@ -97,6 +171,15 @@ public class Colors {
return rgba;
}
/**
* Converts an RGB color in sRGB space to an RGB color in linear space.
*
* @param conversion the conversion algorithm to use
* @param rgb an RGB float array of at least size 3, will be modified
*
* @return the passed-in <code>rgb</code> array, after applying the conversion. The alpha
* channel, if present, is left unmodified.
*/
@NonNull
@LinearColor
public static float[] toLinear(@NonNull Conversion conversion, @NonNull @Size(min = 3) float[] rgb) {
@@ -116,6 +199,14 @@ public class Colors {
return rgb;
}
/**
* Converts a correlated color temperature to a linear RGB color in sRGB space. The temperature
* must be expressed in Kelvin and must be in the range 1,000K to 15,000K.
*
* @param temperature the temperature, in Kelvin
*
* @return an RGB float array of size 3 with the result of the conversion
*/
@NonNull
@Size(3)
@LinearColor
@@ -125,6 +216,14 @@ public class Colors {
return color;
}
/**
* Converts a CIE standard illuminant series D to a linear RGB color in sRGB space. The
* temperature must be expressed in Kelvin and must be in the range 4,000K to 25,000K.
*
* @param temperature the temperature, in Kelvin
*
* @return an RGB float array of size 3 with the result of the conversion
*/
@NonNull
@Size(3)
@LinearColor

View File

@@ -16,21 +16,120 @@
package com.google.android.filament;
import com.google.android.filament.proguard.UsedByReflection;
import android.support.annotation.NonNull;
import com.google.android.filament.proguard.UsedByReflection;
/**
* Engine is filament's main entry-point.
* <p>
* An Engine instance main function is to keep track of all resources created by the user and
* manage the rendering thread as well as the hardware renderer.
* <p>
* To use filament, an Engine instance must be created first:
*
* <pre>
* import com.google.android.filament.*
*
* Engine engine = Engine.create();
* </pre>
* <p>
* Engine essentially represents (or is associated to) a hardware context
* (e.g. an OpenGL ES context).
* <p>
* Rendering typically happens in an operating system's window (which can be full screen), such
* window is managed by a {@link Renderer}.
* <p>
* A typical filament render loop looks like this:
*
*
* <pre>
* import com.google.android.filament.*
*
* Engin engine = Engine.create();
* SwapChain swapChain = engine.createSwapChain(nativeWindow);
* Renderer renderer = engine.createRenderer();
* Scene scene = engine.createScene();
* View view = engine.createView();
*
* view.setScene(scene);
*
* do {
* // typically we wait for VSYNC and user input events
* if (renderer.beginFrame(swapChain)) {
* renderer.render(view);
* renderer.endFrame();
* }
* } while (!quit);
*
* engine.destroyView(view);
* engine.destroyScene(scene);
* engine.destroyRenderer(renderer);
* engine.destroySwapChain(swapChain);
* engine.destroy();
* </pre>
*
* <h1><u>Resource Tracking</u></h1>
* <p>
* Each <code>Engine</code> instance keeps track of all objects created by the user, such as vertex
* and index buffers, lights, cameras, etc...
* The user is expected to free those resources, however, leaked resources are freed when the
* engine instance is destroyed and a warning is emitted in the console.
*
* <h1><u>Thread safety</u></h1>
* <p>
* An <code>Engine</code> instance is not thread-safe. The implementation makes no attempt to
* synchronize calls to an <code>Engine</code> instance methods.
* If multi-threading is needed, synchronization must be external.
*
* <h1><u>Multi-threading</u></h1>
* <p>
* When created, the <code>Engine</code> instance starts a render thread as well as multiple worker
* threads, these threads have an elevated priority appropriate for rendering, based on the
* platform's best practices. The number of worker threads depends on the platform and is
* automatically chosen for best performance.
* <p>
* On platforms with asymmetric cores (e.g. ARM's Big.Little), <code>Engine</code> makes some
* educated guesses as to which cores to use for the render thread and worker threads. For example,
* it'll try to keep an OpenGL ES thread on a Big core.
*
* <h1><u>Swap Chains</u></h1>
* <p>
* A swap chain represents an Operating System's <b>native</b> renderable surface.
* Typically it's a window or a view. Because a {@link SwapChain} is initialized from a native
* object, it is given to filament as an <code>Object</code>, which must be of the proper type for
* each platform filament is running on.
* <p>
*
* @see SwapChain
* @see Renderer
*/
public class Engine {
private long mNativeObject;
@NonNull private final TransformManager mTransformManager;
@NonNull private final LightManager mLightManager;
@NonNull private final RenderableManager mRenderableManager;
/**
* Denotes a backend
*/
public enum Backend {
DEFAULT, // Automatically selects an appropriate driver for the platform.
OPENGL, // Selects the OpenGL ES driver.
VULKAN, // Selects the experimental Vulkan driver.
NOOP, // Selects the no-op driver for testing purposes.
/**
* Automatically selects an appropriate driver for the platform.
*/
DEFAULT,
/**
* Selects the OpenGL ES driver.
*/
OPENGL,
/**
* Selects the experimental Vulkan driver.
*/
VULKAN,
/**
* Selects the no-op driver for testing purposes.
*/
NOOP,
}
private Engine(long nativeEngine) {
@@ -40,6 +139,19 @@ public class Engine {
mRenderableManager = new RenderableManager(nGetRenderableManager(nativeEngine));
}
/**
* Creates an instance of Engine using the default {@link Backend}
* <p>
* This method is one of the few thread-safe methods.
*
* @return A newly created <code>Engine</code>, or <code>null</code> if the GPU driver couldn't
* be initialized, for instance if it doesn't support the right version of OpenGL or
* OpenGL ES.
*
* @exception IllegalStateException can be thrown if there isn't enough memory to
* allocate the command buffer.
*
*/
@NonNull
public static Engine create() {
long nativeEngine = nCreateEngine(0, 0);
@@ -47,6 +159,21 @@ public class Engine {
return new Engine(nativeEngine);
}
/**
* Creates an instance of Engine using the specified {@link Backend}
* <p>
* This method is one of the few thread-safe methods.
*
* @param backend driver backend to use
*
* @return A newly created <code>Engine</code>, or <code>null</code> if the GPU driver couldn't
* be initialized, for instance if it doesn't support the right version of OpenGL or
* OpenGL ES.
*
* @exception IllegalStateException can be thrown if there isn't enough memory to
* allocate the command buffer.
*
*/
@NonNull
public static Engine create(@NonNull Backend backend) {
long nativeEngine = nCreateEngine(backend.ordinal(), 0);
@@ -55,9 +182,21 @@ public class Engine {
}
/**
* Valid shared context:
* - Android: EGLContext
* - Other: none
* Creates an instance of Engine using the {@link Backend#OPENGL} and a shared OpenGL context.
* <p>
* This method is one of the few thread-safe methods.
*
* @param sharedContext A platform-dependant OpenGL context used as a shared context
* when creating filament's internal context. On Android this parameter
* <b>must be</b> an instance of {@link android.opengl.EGLContext}.
*
* @return A newly created <code>Engine</code>, or <code>null</code> if the GPU driver couldn't
* be initialized, for instance if it doesn't support the right version of OpenGL or
* OpenGL ES.
*
* @exception IllegalStateException can be thrown if there isn't enough memory to
* allocate the command buffer.
*
*/
@NonNull
public static Engine create(@NonNull Object sharedContext) {
@@ -70,15 +209,41 @@ public class Engine {
throw new IllegalArgumentException("Invalid shared context " + sharedContext);
}
/**
* @return <code>true</code> if this <code>Engine</code> is initialized properly.
*/
public boolean isValid() {
return mNativeObject != 0;
}
/**
* Destroy the <code>Engine</code> instance and all associated resources.
* <p>
* This method is one of the few thread-safe methods.
* <p>
* {@link Engine#destroy()} should be called last and after all other resources have been
* destroyed, it ensures all filament resources are freed.
* <p>
* <code>Destroy</code> performs the following tasks:
* <li>Destroy all internal software and hardware resources.</li>
* <li>Free all user allocated resources that are not already destroyed and logs a warning.
* <p>This indicates a "leak" in the user's code.</li>
* <li>Terminate the rendering engine's thread.</li>
*
* <pre>
* Engine engine = Engine.create();
* engine.destroy();
* </pre>
*/
public void destroy() {
nDestroyEngine(getNativeObject());
clearNativeObject();
}
/**
* @return the backend used by this <code>Engine</code>
*/
@NonNull
public Backend getBackend() {
return Backend.values()[(int) nGetBackend(getNativeObject())];
}
@@ -86,25 +251,36 @@ public class Engine {
// SwapChain
/**
* Valid surface types:
* - Android: Surface
* - Other: none
* Creates an opaque {@link SwapChain} from the given OS native window handle.
*
* @param surface on Android, <b>must be</b> an instance of {@link android.view.Surface}
*
* @return a newly created {@link SwapChain} object
*
* @exception IllegalStateException can be thrown if the SwapChain couldn't be created
*/
@NonNull
public SwapChain createSwapChain(@NonNull Object surface) {
return createSwapChain(surface, SwapChain.CONFIG_DEFAULT);
}
/**
* Valid surface types:
* - Android: Surface
* - Other: none
* Creates a {@link SwapChain} from the given OS native window handle.
*
* Flags: see CONFIG flags in SwapChain.
* @param surface on Android, <b>must be</b> an instance of {@link android.view.Surface}
*
* @param flags configuration flags, see {@link SwapChain}
*
* @return a newly created {@link SwapChain} object
*
* @exception IllegalStateException can be thrown if the SwapChain couldn't be created
*
* @see SwapChain#CONFIG_DEFAULT
* @see SwapChain#CONFIG_TRANSPARENT
* @see SwapChain#CONFIG_READABLE
*
*/
@NonNull
public SwapChain createSwapChain(@NonNull Object surface, long flags) {
if (Platform.get().validateSurface(surface)) {
long nativeSwapChain = nCreateSwapChain(getNativeObject(), surface, flags);
@@ -114,6 +290,18 @@ public class Engine {
throw new IllegalArgumentException("Invalid surface " + surface);
}
/**
* Creates a {@link SwapChain} from a {@link NativeSurface}.
*
* @param surface a properly initialized {@link NativeSurface}
*
* @param flags configuration flags, see {@link SwapChain}
*
* @return a newly created {@link SwapChain} object
*
* @exception IllegalStateException can be thrown if the {@link SwapChain} couldn't be created
*/
@NonNull
public SwapChain createSwapChainFromNativeSurface(@NonNull NativeSurface surface, long flags) {
long nativeSwapChain =
nCreateSwapChainFromRawPointer(getNativeObject(), surface.getNativeObject(), flags);
@@ -121,6 +309,10 @@ public class Engine {
return new SwapChain(nativeSwapChain, surface);
}
/**
* Destroys a {@link SwapChain} and frees all its associated resources.
* @param swapChain the {@link SwapChain} to destroy
*/
public void destroySwapChain(@NonNull SwapChain swapChain) {
nDestroySwapChain(getNativeObject(), swapChain.getNativeObject());
swapChain.clearNativeObject();
@@ -128,6 +320,11 @@ public class Engine {
// View
/**
* Creates a {@link View}.
* @return a newly created {@link View}
* @exception IllegalStateException can be thrown if the {@link View} couldn't be created
*/
@NonNull
public View createView() {
long nativeView = nCreateView(getNativeObject());
@@ -135,6 +332,10 @@ public class Engine {
return new View(nativeView);
}
/**
* Destroys a {@link View} and frees all its associated resources.
* @param view the {@link View} to destroy
*/
public void destroyView(@NonNull View view) {
nDestroyView(getNativeObject(), view.getNativeObject());
view.clearNativeObject();
@@ -142,6 +343,11 @@ public class Engine {
// Renderer
/**
* Creates a {@link Renderer}.
* @return a newly created {@link Renderer}
* @exception IllegalStateException can be thrown if the {@link Renderer} couldn't be created
*/
@NonNull
public Renderer createRenderer() {
long nativeRenderer = nCreateRenderer(getNativeObject());
@@ -149,6 +355,10 @@ public class Engine {
return new Renderer(this, nativeRenderer);
}
/**
* Destroys a {@link Renderer} and frees all its associated resources.
* @param renderer the {@link Renderer} to destroy
*/
public void destroyRenderer(@NonNull Renderer renderer) {
nDestroyRenderer(getNativeObject(), renderer.getNativeObject());
renderer.clearNativeObject();
@@ -156,6 +366,12 @@ public class Engine {
// Camera
/**
* Creates a new <code>entity</code> and adds a {@link Camera} component to it.
*
* @return A newly created {@link Camera}
* @exception IllegalStateException can be thrown if the {@link Camera} couldn't be created
*/
@NonNull
public Camera createCamera() {
long nativeCamera = nCreateCamera(getNativeObject());
@@ -163,6 +379,13 @@ public class Engine {
return new Camera(nativeCamera);
}
/**
* Creates and adds a {@link Camera} component to a given <code>entity</code>.
*
* @param entity <code>entity</code> to add the camera component to
* @return A newly created {@link Camera}
* @exception IllegalStateException can be thrown if the {@link Camera} couldn't be created
*/
@NonNull
public Camera createCamera(@Entity int entity) {
long nativeCamera = nCreateCameraWithEntity(getNativeObject(), entity);
@@ -170,6 +393,10 @@ public class Engine {
return new Camera(nativeCamera);
}
/**
* Destroys a {@link Camera} component and frees all its associated resources.
* @param camera the {@link Camera} to destroy
*/
public void destroyCamera(@NonNull Camera camera) {
nDestroyCamera(getNativeObject(), camera.getNativeObject());
camera.clearNativeObject();
@@ -177,6 +404,11 @@ public class Engine {
// Scene
/**
* Creates a {@link Scene}.
* @return a newly created {@link Scene}
* @exception IllegalStateException can be thrown if the {@link Scene} couldn't be created
*/
@NonNull
public Scene createScene() {
long nativeScene = nCreateScene(getNativeObject());
@@ -184,6 +416,10 @@ public class Engine {
return new Scene(nativeScene);
}
/**
* Destroys a {@link Scene} and frees all its associated resources.
* @param scene the {@link Scene} to destroy
*/
public void destroyScene(@NonNull Scene scene) {
nDestroyScene(getNativeObject(), scene.getNativeObject());
scene.clearNativeObject();
@@ -191,6 +427,10 @@ public class Engine {
// Stream
/**
* Destroys a {@link Stream} and frees all its associated resources.
* @param stream the {@link Stream} to destroy
*/
public void destroyStream(@NonNull Stream stream) {
nDestroyStream(getNativeObject(), stream.getNativeObject());
stream.clearNativeObject();
@@ -198,13 +438,22 @@ public class Engine {
// Fence
/**
* Creates a {@link Fence}.
* @return a newly created {@link Fence}
* @exception IllegalStateException can be thrown if the {@link Fence} couldn't be created
*/
@NonNull
public Fence createFence(@NonNull Fence.Type type) {
long nativeFence = nCreateFence(getNativeObject(), type.ordinal());
public Fence createFence() {
long nativeFence = nCreateFence(getNativeObject());
if (nativeFence == 0) throw new IllegalStateException("Couldn't create Fence");
return new Fence(nativeFence);
}
/**
* Destroys a {@link Fence} and frees all its associated resources.
* @param fence the {@link Fence} to destroy
*/
public void destroyFence(@NonNull Fence fence) {
nDestroyFence(getNativeObject(), fence.getNativeObject());
fence.clearNativeObject();
@@ -212,73 +461,139 @@ public class Engine {
// others...
/**
* Destroys a {@link IndexBuffer} and frees all its associated resources.
* @param indexBuffer the {@link IndexBuffer} to destroy
*/
public void destroyIndexBuffer(@NonNull IndexBuffer indexBuffer) {
nDestroyIndexBuffer(getNativeObject(), indexBuffer.getNativeObject());
indexBuffer.clearNativeObject();
}
/**
* Destroys a {@link VertexBuffer} and frees all its associated resources.
* @param vertexBuffer the {@link VertexBuffer} to destroy
*/
public void destroyVertexBuffer(@NonNull VertexBuffer vertexBuffer) {
nDestroyVertexBuffer(getNativeObject(), vertexBuffer.getNativeObject());
vertexBuffer.clearNativeObject();
}
/**
* Destroys a {@link IndirectLight} and frees all its associated resources.
* @param ibl the {@link IndirectLight} to destroy
*/
public void destroyIndirectLight(@NonNull IndirectLight ibl) {
nDestroyIndirectLight(getNativeObject(), ibl.getNativeObject());
ibl.clearNativeObject();
}
/**
* Destroys a {@link Material} and frees all its associated resources.
* <p>
* All {@link MaterialInstance} of the specified {@link Material} must be destroyed before
* destroying it; if some {@link MaterialInstance} remain, this method fails silently.
*
* @param material the {@link Material} to destroy
*/
public void destroyMaterial(@NonNull Material material) {
nDestroyMaterial(getNativeObject(), material.getNativeObject());
material.clearNativeObject();
}
/**
* Destroys a {@link MaterialInstance} and frees all its associated resources.
* @param materialInstance the {@link MaterialInstance} to destroy
*/
public void destroyMaterialInstance(@NonNull MaterialInstance materialInstance) {
nDestroyMaterialInstance(getNativeObject(), materialInstance.getNativeObject());
materialInstance.clearNativeObject();
}
/**
* Destroys a {@link Skybox} and frees all its associated resources.
* @param skybox the {@link Skybox} to destroy
*/
public void destroySkybox(@NonNull Skybox skybox) {
nDestroySkybox(getNativeObject(), skybox.getNativeObject());
skybox.clearNativeObject();
}
/**
* Destroys a {@link Texture} and frees all its associated resources.
* @param texture the {@link Texture} to destroy
*/
public void destroyTexture(@NonNull Texture texture) {
nDestroyTexture(getNativeObject(), texture.getNativeObject());
texture.clearNativeObject();
}
/**
* Destroys a {@link RenderTarget} and frees all its associated resources.
* @param target the {@link RenderTarget} to destroy
*/
public void destroyRenderTarget(@NonNull RenderTarget target) {
nDestroyRenderTarget(getNativeObject(), target.getNativeObject());
target.clearNativeObject();
}
/**
* Destroys an <code>entity</code> and all its components.
* <p>
* It is recommended to destroy components individually before destroying their
* <code>entity</code>, this gives more control as to when the destruction really happens.
* Otherwise, orphaned components are garbage collected, which can happen at a later time.
* Even when component are garbage collected, the destruction of their <code>entity</code>
* terminates their participation immediately.
*
* @param entity the <code>entity</code> to destroy
*/
public void destroyEntity(@Entity int entity) {
nDestroyEntity(getNativeObject(), entity);
}
// Managers
/**
* @return the {@link TransformManager} used by this {@link Engine}
*/
@NonNull
public TransformManager getTransformManager() {
return mTransformManager;
}
/**
* @return the {@link LightManager} used by this {@link Engine}
*/
@NonNull
public LightManager getLightManager() {
return mLightManager;
}
/**
* @return the {@link RenderableManager} used by this {@link Engine}
*/
@NonNull
public RenderableManager getRenderableManager() {
return mRenderableManager;
}
/**
* Kicks the hardware thread (e.g.: the OpenGL, Vulkan or Metal thread) and blocks until
* all commands to this point are executed. Note that this doesn't guarantee that the
* hardware is actually finished.
*
* <p>This is typically used right after destroying the <code>SwapChain</code>,
* in cases where a guarantee about the SwapChain destruction is needed in a timely fashion,
* such as when responding to Android's
* {@link android.view.SurfaceHolder.Callback#surfaceDestroyed surfaceDestroyed}.</p>
*/
public void flushAndWait() {
Fence.waitAndDestroy(createFence(Fence.Type.HARD), Fence.Mode.FLUSH);
nFlushAndWait(getNativeObject());
}
@UsedByReflection("TextureHelper.java")
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Engine");
}
@@ -304,7 +619,7 @@ public class Engine {
private static native void nDestroyCamera(long nativeEngine, long nativeCamera);
private static native long nCreateScene(long nativeEngine);
private static native void nDestroyScene(long nativeEngine, long nativeScene);
private static native long nCreateFence(long nativeEngine, int fenceType);
private static native long nCreateFence(long nativeEngine);
private static native void nDestroyFence(long nativeEngine, long nativeFence);
private static native void nDestroyStream(long nativeEngine, long nativeStream);
private static native void nDestroyIndexBuffer(long nativeEngine, long nativeIndexBuffer);
@@ -316,6 +631,7 @@ public class Engine {
private static native void nDestroyTexture(long nativeEngine, long nativeTexture);
private static native void nDestroyRenderTarget(long nativeEngine, long nativeTarget);
private static native void nDestroyEntity(long nativeEngine, int entity);
private static native void nFlushAndWait(long nativeEngine);
private static native long nGetTransformManager(long nativeEngine);
private static native long nGetLightManager(long nativeEngine);
private static native long nGetRenderableManager(long nativeEngine);

View File

@@ -16,11 +16,11 @@
package com.google.android.filament;
import com.google.android.filament.proguard.UsedByReflection;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import com.google.android.filament.proguard.UsedByReflection;
public class EntityManager {
private long mNativeObject = nGetEntityManager();
@@ -69,7 +69,7 @@ public class EntityManager {
}
@UsedByReflection("AssetLoader.java")
long getNativeObject() {
public long getNativeObject() {
return mNativeObject;
}

View File

@@ -26,11 +26,6 @@ public class Fence {
public static final long WAIT_FOR_EVER = -1;
public enum Type {
SOFT,
HARD
}
public enum Mode {
FLUSH,
DONT_FLUSH
@@ -70,7 +65,7 @@ public class Fence {
}
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Fence");
}

View File

@@ -23,6 +23,17 @@ import android.support.annotation.Nullable;
import java.nio.Buffer;
import java.nio.BufferOverflowException;
/**
* A buffer containing vertex indices into a <code>VertexBuffer</code>. Indices can be 16 or 32 bit.
* The buffer itself is a GPU resource, therefore mutating the data can be relatively slow.
* Typically these buffers are constant.
*
* It is possible, and even encouraged, to use a single index buffer for several
* <code>Renderables</code>.
*
* @see VertexBuffer
* @see RenderableManager
*/
public class IndexBuffer {
private long mNativeObject;
@@ -30,21 +41,20 @@ public class IndexBuffer {
mNativeObject = nativeIndexBuffer;
}
long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed IndexBuffer");
}
return mNativeObject;
}
public static class Builder {
@SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"})
// Keep to finalize native resources
private final BuilderFinalizer mFinalizer;
private final long mNativeBuilder;
/**
* Type of the index buffer.
*/
public enum IndexType {
/** 16-bit indices */
USHORT,
/** 32-bit indices */
UINT,
}
@@ -53,18 +63,45 @@ public class IndexBuffer {
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Size of the index buffer in elements.
*
* @param indexCount number of indices the <code>IndexBuffer</code> can hold
*
* @return this <code>Builder</code> object for chaining calls
*/
@NonNull
public Builder indexCount(@IntRange(from = 1) int indexCount) {
nBuilderIndexCount(mNativeBuilder, indexCount);
return this;
}
/**
* Type of the index buffer, 16-bit or 32-bit.
*
* @param indexType type of indices stored in the <code>IndexBuffer</code>
*
* @return this <code>Builder</code> object for chaining calls
*/
@NonNull
public Builder bufferType(@NonNull IndexType indexType) {
nBuilderBufferType(mNativeBuilder, indexType.ordinal());
return this;
}
/**
* Creates and returns the <code>IndexBuffer</code> object. After creation, the index buffer
* is uninitialized. Use {@link #setBuffer} to initialized the <code>IndexBuffer</code>.
*
* @param engine reference to the {@link Engine} to associate this <code>IndexBuffer</code>
* with
*
* @return the newly created <code>IndexBuffer</code> object
*
* @exception IllegalStateException if the IndexBuffer could not be created
*
* @see #setBuffer
*/
@NonNull
public IndexBuffer build(@NonNull Engine engine) {
long nativeIndexBuffer = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -92,24 +129,69 @@ public class IndexBuffer {
}
}
/**
* Returns the size of this <code>IndexBuffer</code> in elements.
*
* @return the number of indices the <code>IndexBuffer</code> holds
*/
@IntRange(from = 0)
public int getIndexCount() {
return nGetIndexCount(getNativeObject());
}
/**
* Asynchronously copy-initializes this <code>IndexBuffer</code> from the data provided.
*
* @param engine reference to the {@link Engine} to associate this
* <code>IndexBuffer</code> with
* @param buffer a CPU-side {@link Buffer} with the data used to initialize the
* <code>IndexBuffer</code>. <code>buffer</code> should contain raw,
* untyped data that will be interpreted as either 16-bit or 32-bits
* indices based on the <code>IndexType</code> of this
* <code>IndexBuffer</code>.
*/
public void setBuffer(@NonNull Engine engine, @NonNull Buffer buffer) {
setBuffer(engine, buffer, 0, 0, null, null);
}
/**
* Asynchronously copy-initializes a region of this <code>IndexBuffer</code> from the data
* provided.
*
* @param engine reference to the {@link Engine} to associate this
* <code>IndexBuffer</code> with
* @param buffer a CPU-side {@link Buffer} with the data used to initialize the
* <code>IndexBuffer</code>. <code>buffer</code> should contain raw,
* untyped data that will be interpreted as either 16-bit or 32-bits
* indices based on the <code>IndexType</code> of this
* <code>IndexBuffer</code>.
* @param destOffsetInBytes offset in <i>bytes</i> into the <code>IndexBuffer</code>
* @param count number of buffer elements to consume, defaults to
* <code>buffer.remaining()</code>
*/
public void setBuffer(@NonNull Engine engine, @NonNull Buffer buffer,
@IntRange(from = 0) int destOffsetInBytes, @IntRange(from = 0) int count) {
setBuffer(engine, buffer, destOffsetInBytes, count, null, null);
}
/**
* Valid handler types:
* - Android: Handler, Executor
* - Other: Executor
* Asynchronously copy-initializes a region of this <code>IndexBuffer</code> from the data
* provided.
*
* @param engine reference to the {@link Engine} to associate this
* <code>IndexBuffer</code> with
* @param buffer a CPU-side {@link Buffer} with the data used to initialize the
* <code>IndexBuffer</code>. <code>buffer</code> should contain raw,
* untyped data that will be interpreted as either 16-bit or 32-bits
* indices based on the <code>IndexType</code> of this
* <code>IndexBuffer</code>.
* @param destOffsetInBytes offset in <i>bytes</i> into the <code>IndexBuffer</code>
* @param count number of buffer elements to consume, defaults to
* <code>buffer.remaining()</code>
* @param handler an {@link java.util.concurrent.Executor Executor}. On Android this
* can also be a {@link android.os.Handler Handler}.
* @param callback a callback executed by <code>handler</code> when <code>buffer</code>
* is no longer needed.
*/
public void setBuffer(@NonNull Engine engine, @NonNull Buffer buffer,
@IntRange(from = 0) int destOffsetInBytes, @IntRange(from = 0) int count,
@@ -121,6 +203,13 @@ public class IndexBuffer {
}
}
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed IndexBuffer");
}
return mNativeObject;
}
void clearNativeObject() {
mNativeObject = 0;
}

View File

@@ -16,13 +16,70 @@
package com.google.android.filament;
import com.google.android.filament.proguard.UsedByReflection;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.Size;
import com.google.android.filament.proguard.UsedByReflection;
/**
* <code>IndirectLight</code> is used to simulate environment lighting, a form of global illumination.
*
* <p>Environment lighting has a two components:</p>
* <ol>
* <li>irradiance</li>
* <li>reflections (specular component)</li>
* </ol>
*
* <p>Environments are usually captured as high-resolution HDR equirectangular images and processed
* by the <b>cmgen</b> tool to generate the data needed by {@link IndirectLight}.</p>
*
* <p>Currently {@link IndirectLight} is intended to be used for "distant probes", that is, to represent
* global illumination from a distant (i.e. at infinity) environment, such as the sky or distant
* mountains. Only a single {@link IndirectLight} can be used in a {@link Scene}.
* This limitation will be lifted in the future.</p>
*
*
* <h1>Creation and destruction</h1>
*
* <p>An {@link IndirectLight} object is created using the {@link IndirectLight.Builder} and
* destroyed by calling {@link Engine#destroyIndirectLight}.</p>
*
* <pre>
* Engine engine = Engine.create();
*
* Scene scene = engine.createScene();
*
* IndirectLight environment = new IndirectLight.Builder()
* .reflections(cubemap)
* .irradiance(numBands, sphericalHarmonicsCoefficients)
* .build(engine);
*
* scene.setIndirectLight(environment);
* </pre>
*
*
* <h1>Irradiance</h1>
*
* <p>The irradiance represents the light that comes from the environment and shines an
* object's surface. It is represented as
* <a href="https://en.wikipedia.org/wiki/Spherical_harmonics">Spherical Harmonics</a> (SH) of 1, 2 or
* 3 bands, respectively 1, 4 or 9 coefficients.</p>
*
* <p>Use the <b>cmgen</b> tool to generate the Spherical Harmonics for a given environment.</p>
*
*
* <h1>Reflections</h1>
*
* <p>The reflections on object surfaces (specular component) is calculated from a specially
* filtered cubemap pyramid generated by the <b>cmgen</b> tool.</p>
*
* @see Scene
* @see LightManager
* @see Texture
* @see Skybox
*/
public class IndirectLight {
long mNativeObject;
@@ -31,22 +88,97 @@ public class IndirectLight {
mNativeObject = indirectLight;
}
/**
* Use <code>Builder</code> to construct an <code>IndirectLight</code> object instance.
*/
public static class Builder {
@SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"}) // Keep to finalize native resources
private final BuilderFinalizer mFinalizer;
private final long mNativeBuilder;
/**
* Use <code>Builder</code> to construct an <code>IndirectLight</code> object instance.
*/
public Builder() {
mNativeBuilder = nCreateBuilder();
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Set the reflections cubemap mipmap chain.
*
* @param cubemap A mip-mapped cubemap generated by <b>cmgen</b>. Each cubemap level
* encodes the irradiance for a roughness level.
*
* @return This Builder, for chaining calls.
*
*/
@NonNull
public Builder reflections(@NonNull Texture cubemap) {
nBuilderReflections(mNativeBuilder, cubemap.getNativeObject());
return this;
}
/**
* Sets the irradiance as Spherical Harmonics.
*
* <p>The irradiance coefficients must be pre-convolved by <code>&lt n &sdot l &gt</code> and
* pre-multiplied by the Lambertian diffuse BRDF <code>1/&pi</code> and
* specified as Spherical Harmonics coefficients.</p>
*
* <p>Additionally, these Spherical Harmonics coefficients must be pre-scaled by the
* reconstruction factors A<sup>l,m</sup>.</p>
*
* <p>The final coefficients can be generated using the <code>cmgen</code> tool.</p>
*
* <p>The index in the <code>sh</code> array is given by:
* <br><code>index(l, m) = 3 &times (l * (l + 1) + m)</code>
* <br><code>sh[index(l,m) + 0] = L<sub>R</sub><sup>l,m</sup>
* &times 1/&pi
* &times A<sup>l,m</sup>
* &times C<sup>l</sup> </code>
* <br><code>sh[index(l,m) + 1] = L<sub>G</sub><sup>l,m</sup>
* &times 1/&pi
* &times A<sup>l,m</sup>
* &times C<sup>l</sup> </code>
* <br><code>sh[index(l,m) + 2] = L<sub>B</sub><sup>l,m</sup>
* &times 1/&pi
* &times A<sup>l,m</sup>
* &times C<sup>l</sup> </code>
* </p>
*
* <center>
* <table border="1" cellpadding="3">
* <tr><th> index </th><th> l </th><th> m </th><th> A<sup>l,m</sup> </th><th> C<sup>l</sup> </th>
* <th> 1/&pi &times A<sup>l,m</sup> &times C<sup>l</sup></th></tr>
* <tr align="right"><td>0</td><td>0</td><td> 0</td><td> 0.282095</td><td>3.1415926</td><td> 0.282095</td></tr>
* <tr align="right"><td>1</td><td>1</td><td>-1</td><td>-0.488602</td><td>2.0943951</td><td>-0.325735</td></tr>
* <tr align="right"><td>2</td><td>1</td><td> 0</td><td> 0.488602</td><td>2.0943951</td><td> 0.325735</td></tr>
* <tr align="right"><td>3</td><td>1</td><td> 1</td><td>-0.488602</td><td>2.0943951</td><td>-0.325735</td></tr>
* <tr align="right"><td>4</td><td>2</td><td>-2</td><td> 1.092548</td><td>0.785398 </td><td> 0.273137</td></tr>
* <tr align="right"><td>5</td><td>2</td><td>-1</td><td>-1.092548</td><td>0.785398 </td><td>-0.273137</td></tr>
* <tr align="right"><td>6</td><td>2</td><td> 0</td><td> 0.315392</td><td>0.785398 </td><td> 0.078848</td></tr>
* <tr align="right"><td>7</td><td>2</td><td> 1</td><td>-1.092548</td><td>0.785398 </td><td>-0.273137</td></tr>
* <tr align="right"><td>8</td><td>2</td><td> 2</td><td> 0.546274</td><td>0.785398 </td><td> 0.136569</td></tr>
* </table>
* </center>
*
*
* <p>Only 1, 2 or 3 bands are allowed.</p>
*
* <p>Because the coefficients are pre-scaled, <code>sh[0]</code> is the environment's
* average irradiance.</p>
*
* @param bands Number of spherical harmonics bands. Must be 1, 2 or 3.
* @param sh Array containing the spherical harmonics coefficients.
* The size of the array must be <code>3 &times bands<sup>2</sup></code>
* (i.e. 1, 4 or 9 <code>float3</code> coefficients respectively).
*
* @return This Builder, for chaining calls.
*
* @exception ArrayIndexOutOfBoundsException if the <code>sh</code> array length is smaller
* than 3 &times bands<sup>2</sup>
*/
@NonNull
public Builder irradiance(@IntRange(from=1, to=3) int bands, @NonNull float[] sh) {
switch (bands) {
@@ -65,6 +197,44 @@ public class IndirectLight {
return this;
}
/**
* Sets the irradiance from the radiance expressed as Spherical Harmonics.
*
* <p>The radiance must be specified as Spherical Harmonics coefficients L<sup>l,m</sup>, where
* each coefficient is comprised of three floats for red, green and blue components, respectively</p>
*
* <p>The index in the <code>sh</code> array is given by:
* <br><code>index(l, m) = 3 &times (l * (l + 1) + m)</code>
* <br><code>sh[index(l,m) + 0] = L<sub>R</sub><sup>l,m</sup></code>
* <br><code>sh[index(l,m) + 1] = L<sub>G</sub><sup>l,m</sup></code>
* <br><code>sh[index(l,m) + 2] = L<sub>B</sub><sup>l,m</sup></code>
* </p>
*
* <center>
* <table border="1" cellpadding="3">
* <tr><th> index </th><th> l </th><th> m </th>
* <tr align="right"><td>0</td><td>0</td><td> 0</td>
* <tr align="right"><td>1</td><td>1</td><td>-1</td>
* <tr align="right"><td>2</td><td>1</td><td> 0</td>
* <tr align="right"><td>3</td><td>1</td><td> 1</td>
* <tr align="right"><td>4</td><td>2</td><td>-2</td>
* <tr align="right"><td>5</td><td>2</td><td>-1</td>
* <tr align="right"><td>6</td><td>2</td><td> 0</td>
* <tr align="right"><td>7</td><td>2</td><td> 1</td>
* <tr align="right"><td>8</td><td>2</td><td> 2</td>
* </table>
* </center>
*
* @param bands Number of spherical harmonics bands. Must be 1, 2 or 3.
* @param sh Array containing the spherical harmonics coefficients.
* The size of the array must be 3 &times <code>bands<sup>2</sup></code>
* (i.e. 1, 4 or 9 <code>float3</code> coefficients respectively).
*
* @return This Builder, for chaining calls.
*
* @exception ArrayIndexOutOfBoundsException if the <code>sh</code> array length is smaller
* than 3 &times bands<sup>2</sup>
*/
@NonNull
public Builder radiance(@IntRange(from=1, to=3) int bands, @NonNull float[] sh) {
switch (bands) {
@@ -83,18 +253,52 @@ public class IndirectLight {
return this;
}
/**
* Sets the irradiance as a cubemap.
* <p></p>
* The irradiance can alternatively be specified as a cubemap instead of Spherical
* Harmonics coefficients. It may or may not be more efficient, depending on your
* hardware (essentially, it's trading ALU for bandwidth).
* <p></p>
* This irradiance cubemap can be generated with the <code>cmgen</code> tool.
*
* @param cubemap Cubemap representing the Irradiance pre-convolved by
* <code>&lt n &sdot l &gt</code>.
*
* @return This Builder, for chaining calls.
*
* @see #irradiance(int bands, float[] sh)
*/
@NonNull
public Builder irradiance(@NonNull Texture cubemap) {
nIrradianceAsTexture(mNativeBuilder, cubemap.getNativeObject());
return this;
}
/**
* Environment intensity (optional).
*
* <p>Because the environment is encoded usually relative to some reference, the
* range can be adjusted with this method.</p>
*
* @param envIntensity Scale factor applied to the environment and irradiance such that
* the result is in cd/m^2 (lux) units (default = 30000)
*
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder intensity(float envIntensity) {
nIntensity(mNativeBuilder, envIntensity);
return this;
}
/**
* Specifies the rigid-body transformation to apply to the IBL.
*
* @param rotation 3x3 rotation matrix. Must be a rigid-body transform.
*
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder rotation(@NonNull @Size(min = 9) float[] rotation) {
nRotation(mNativeBuilder,
@@ -104,6 +308,15 @@ public class IndirectLight {
return this;
}
/**
* Creates the IndirectLight object and returns a pointer to it.
*
* @param engine The {@link Engine} to associate this <code>IndirectLight</code> with.
*
* @return A newly created <code>IndirectLight</code>
*
* @exception IllegalStateException if a parameter to a builder function was invalid.
*/
@NonNull
public IndirectLight build(@NonNull Engine engine) {
long nativeIndirectLight = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -128,14 +341,31 @@ public class IndirectLight {
}
}
/**
* Sets the environment's intensity.
*
* <p>Because the environment is encoded usually relative to some reference, the
* range can be adjusted with this method.</p>
*
* @param intensity Scale factor applied to the environment and irradiance such that
* the result is in cd/m^2 units (default = 30000)
*/
public void setIntensity(float intensity) {
nSetIntensity(getNativeObject(), intensity);
}
/**
* Returns the environment's intensity in cd/m<sup>2</sup>.
*/
public float getIntensity() {
return nGetIntensity(getNativeObject());
}
/**
* Sets the rigid-body transformation to apply to the IBL.
*
* @param rotation 3x3 rotation matrix. Must be a rigid-body transform.
*/
public void setRotation(@NonNull @Size(min = 9) float[] rotation) {
Asserts.assertMat3fIn(rotation);
nSetRotation(getNativeObject(),
@@ -144,6 +374,14 @@ public class IndirectLight {
rotation[6], rotation[7], rotation[8]);
}
/**
* Returns the rigid-body transformation applied to the IBL.
*
* @param rotation an array of 9 floats to receive the rigid-body transformation applied to
* the IBL or <code>null</code>
* @return the <code>rotation</code> paramter if it was provided, or a newly allocated float
* array containing the rigid-body transformation applied to the IBL
*/
@NonNull @Size(min = 9)
public float[] getRotation(@Nullable @Size(min = 9) float[] rotation) {
rotation = Asserts.assertMat3f(rotation);
@@ -151,6 +389,27 @@ public class IndirectLight {
return rotation;
}
/**
* Helper to estimate the direction of the dominant light in the environment.
*
* <p>This assumes that there is only a single dominant light (such as the sun in outdoors
* environments), if it's not the case the direction returned will be an average of the
* various lights based on their intensity.</p>
*
* <p>If there are no clear dominant light, as is often the case with low dynamic range (LDR)
* environments, this method may return a wrong or unexpected direction.</p>
*
* <p>The dominant light direction can be used to set a directional light's direction,
* for instance to produce shadows that match the environment.</p>
*
* @param direction an array of 3 floats to receive a unit vector representing the direction of
* the dominant light or <code>null</code>
* @return the <code>direction</code> paramter if it was provided, or a newly allocated float
* array containing a unit vector representing the direction of the dominant light
*
* @see LightManager.Builder#direction
* @see #getColorEstimate
*/
@NonNull @Size(min = 3)
public float[] getDirectionEstimate(@Nullable @Size(min = 3) float[] direction) {
direction = Asserts.assertFloat3(direction);
@@ -158,6 +417,26 @@ public class IndirectLight {
return direction;
}
/**
* Helper to estimate the color and relative intensity of the environment in a given direction.
*
* <p>This can be used to set the color and intensity of a directional light. In this case
* make sure to multiply this relative intensity by the the intensity of this indirect light.</p>
*
* @param colorIntensity an array of 4 floats to receive the result or <code>null</code>
* @param x the x coordinate of a unit vector representing the direction of the light
* @param y the x coordinate of a unit vector representing the direction of the light
* @param z the x coordinate of a unit vector representing the direction of the light
*
* @return A vector of 4 floats where the first 3 components represent the linear color and
* the 4th component represents the intensity of the dominant light
*
* @see LightManager.Builder#color
* @see LightManager.Builder#intensity
* @see #getDirectionEstimate
* @see #getIntensity
* @see #setIntensity
*/
@NonNull @Size(min = 4)
public float[] getColorEstimate(@Nullable @Size(min = 4) float[] colorIntensity, float x, float y, float z) {
colorIntensity = Asserts.assertFloat4(colorIntensity);
@@ -165,7 +444,7 @@ public class IndirectLight {
return colorIntensity;
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed IndirectLight");
}

View File

@@ -270,6 +270,10 @@ public class LightManager {
return nIsShadowCaster(mNativeObject, i);
}
public long getNativeObject() {
return mNativeObject;
}
private static native boolean nHasComponent(long nativeLightManager, int entity);
private static native int nGetInstance(long nativeLightManager, int entity);
private static native void nDestroy(long nativeLightManager, int entity);

View File

@@ -19,6 +19,7 @@ package com.google.android.filament;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.support.annotation.Size;
import com.google.android.filament.proguard.UsedByNative;
import java.nio.Buffer;
@@ -28,45 +29,157 @@ import java.util.EnumSet;
import java.util.List;
import java.util.Set;
/**
* A Filament Material defines the visual appearance of an object. Materials function as a
* templates from which {@link MaterialInstance}s can be spawned. Use {@link Builder} to construct
* a Material object.
*
* @see <a href="https://google.github.io/filament/Materials.html">Filament Materials Guide</a>
*/
public class Material {
private long mNativeObject;
private final MaterialInstance mDefaultInstance;
private Set<VertexBuffer.VertexAttribute> mRequiredAttributes;
/** Supported shading models */
public enum Shading {
/**
* No lighting applied, emissive possible
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialmodels/unlitmodel">
* Unlit model</a>
*/
UNLIT,
/**
* Default, standard lighting
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialmodels/litmodel">
* Lit model</a>
*/
LIT,
/**
* Subsurface lighting model
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialmodels/subsurfacemodel">
* Subsurface model</a>
*/
SUBSURFACE,
/**
* Cloth lighting model
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialmodels/clothmodel">
* Cloth model</a>
*/
CLOTH,
/**
* Legacy lighting model
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialmodels/specularglossiness">
* Specular glossiness</a>
*/
SPECULAR_GLOSSINESS
}
/**
* Attribute interpolation types in the fragment shader
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/vertexandattributes:interpolation">
* Vertex and attributes: interpolation</a>
*/
public enum Interpolation {
/** Default, smooth interpolation */
SMOOTH,
/** Flat interpolation */
FLAT
}
/**
* Supported blending modes
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/blendingandtransparency:blending">
* Blending and transparency: blending</a>
*/
public enum BlendingMode {
/** Material is opaque. */
OPAQUE,
/**
* Material is transparent and color is alpha-pre-multiplied.
* Affects diffuse lighting only.
*/
TRANSPARENT,
/** Material is additive (e.g.: hologram). */
ADD,
MODULATE,
/** Material is masked (i.e. alpha tested). */
MASKED,
FADE
/**
* Material is transparent and color is alpha-pre-multiplied.
* Affects specular lighting.
*/
FADE,
/** Material darkens what's behind it. */
MULTIPLY,
/** Material brightens what's behind it. */
SCREEN,
}
/**
* Supported types of vertex domains
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/vertexandattributes:vertexdomain">
* Vertex and attributes: vertexDomain</a>
*/
public enum VertexDomain {
/** Vertices are in object space, default. */
OBJECT,
/** Vertices are in world space. */
WORLD,
/** Vertices are in view space. */
VIEW,
/** Vertices are in normalized device space. */
DEVICE
}
/**
* Face culling Mode
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:culling">
* Rasterization: culling</a>
*/
public enum CullingMode {
/** No culling. Front and back faces are visible. */
NONE,
/** Front face culling. Only back faces are visible. */
FRONT,
/** Back face culling. Only front faces are visible. */
BACK,
/** Front and back culling. Geometry is not visible. */
FRONT_AND_BACK
}
@@ -144,6 +257,13 @@ public class Material {
private Buffer mBuffer;
private int mSize;
/**
* Specifies the material data. The material data is a binary blob produced by
* libfilamat or by matc.
*
* @param buffer buffer containing material data
* @param size size of the material data in bytes
*/
@NonNull
public Builder payload(@NonNull Buffer buffer, @IntRange(from = 0) int size) {
mBuffer = buffer;
@@ -151,6 +271,15 @@ public class Material {
return this;
}
/**
* Creates and returns the Material object.
*
* @param engine reference to the Engine instance to associate this Material with
*
* @return the newly created object
*
* @exception IllegalStateException if the material could not be created
*/
@NonNull
public Material build(@NonNull Engine engine) {
long nativeMaterial = nBuilderBuild(engine.getNativeObject(), mBuffer, mSize);
@@ -159,6 +288,12 @@ public class Material {
}
}
/**
* Creates a new instance of this material. Material instances should be freed using
* {@link Engine#destroyMaterialInstance(MaterialInstance)}.
*
* @return the new instance
*/
@NonNull
public MaterialInstance createInstance() {
long nativeInstance = nCreateInstance(getNativeObject());
@@ -166,63 +301,163 @@ public class Material {
return new MaterialInstance(this, nativeInstance);
}
/** Returns the material's default instance. */
@NonNull
public MaterialInstance getDefaultInstance() {
return mDefaultInstance;
}
/**
* Returns the name of this material. The material name is used for debugging purposes.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/general:name">
* General: name</a>
*/
public String getName() {
return nGetName(getNativeObject());
}
/**
* Returns the shading model of this material.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialmodels">
* Material Models</a>
*/
public Shading getShading() {
return Shading.values()[nGetShading(getNativeObject())];
}
/**
* Returns the interpolation mode of this material. This affects how variables are interpolated.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/vertexandattributes:interpolation">
* Vertex and attributes: interpolation</a>
*/
public Interpolation getInterpolation() {
return Interpolation.values()[nGetInterpolation(getNativeObject())];
}
/**
* Returns the blending mode of this material.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/blendingandtransparency:blending">
* Blending and transparency: blending</a>
*/
public BlendingMode getBlendingMode() {
return BlendingMode.values()[nGetBlendingMode(getNativeObject())];
}
/**
* Returns the vertex domain of this material.
*
* @se
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/vertexandattributes:vertexdomain">
* Vertex and attributes: vertexDomain</a>
* @return
*/
public VertexDomain getVertexDomain() {
return VertexDomain.values()[nGetVertexDomain(getNativeObject())];
}
/**
* Returns the default culling mode of this material.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:culling">
* Rasterization: culling</a>
*/
public CullingMode getCullingMode() {
return CullingMode.values()[nGetCullingMode(getNativeObject())];
}
/**
* Indicates whether this material will write to the color buffer.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:colorwrite">
* Rasterization: colorWrite</a>
*/
public boolean isColorWriteEnabled() {
return nIsColorWriteEnabled(getNativeObject());
}
/**
* Indicates whether this material will write to the depth buffer.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:depthwrite">
* Rasterization: depthWrite</a>
*/
public boolean isDepthWriteEnabled() {
return nIsDepthWriteEnabled(getNativeObject());
}
/**
* Indicates whether this material will use depth testing.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:depthculling">
* Rasterization: depthCulling</a>
*/
public boolean isDepthCullingEnabled() {
return nIsDepthCullingEnabled(getNativeObject());
}
/**
* Indicates whether this material is double-sided.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:doublesided">
* Rasterization: doubleSided</a>
*/
public boolean isDoubleSided() {
return nIsDoubleSided(getNativeObject());
}
/**
* Returns the alpha mask threshold used when the blending mode is set to masked.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/blendingandtransparency:maskthreshold">
* Blending and transparency: maskThreshold</a>
*/
public float getMaskThreshold() {
return nGetMaskThreshold(getNativeObject());
}
/**
* Returns the screen-space variance for specular-antialiasing. This value is between 0 and 1.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/anti-aliasing:specularantialiasingvariance">
* Anti-aliasing: specularAntiAliasingVariance</a>
*/
public float getSpecularAntiAliasingVariance() {
return nGetSpecularAntiAliasingVariance(getNativeObject());
}
/**
* Returns the clamping threshold for specular-antialiasing. This value is between 0 and 1.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/anti-aliasing:specularantialiasingthreshold">
* Anti-aliasing: specularAntiAliasingThreshold</a>
*/
public float getSpecularAntiAliasingThreshold() {
return nGetSpecularAntiAliasingThreshold(getNativeObject());
}
/**
* Returns a set of {@link VertexBuffer.VertexAttribute}s that are required by this material.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/vertexandattributes:requires">
* Vertex and attributes: requires</a>
*/
public Set<VertexBuffer.VertexAttribute> getRequiredAttributes() {
if (mRequiredAttributes == null) {
int bitSet = nGetRequiredAttributes(getNativeObject());
@@ -238,14 +473,38 @@ public class Material {
return mRequiredAttributes;
}
/**
* Returns a bit set representing the set of {@link VertexBuffer.VertexAttribute}s that are
* required by this material. Use {@link #getRequiredAttributes()} to get these as a Set object.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/vertexandattributes:requires">
* Vertex and attributes: requires</a>
*/
int getRequiredAttributesAsInt() {
return nGetRequiredAttributes(getNativeObject());
}
/**
* Returns the number of parameters declared by this material.
* The returned value can be 0.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/general:parameters">
* General: parameters</a>
*/
public int getParameterCount() {
return nGetParameterCount(getNativeObject());
}
/**
* Returns a list of Parameter objects representing this material's parameters.
* The list may be empty if the material has no declared parameters.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/general:parameters">
* General: parameters</a>
*/
public List<Parameter> getParameters() {
int count = getParameterCount();
List<Parameter> parameters = new ArrayList<>(count);
@@ -253,92 +512,314 @@ public class Material {
return parameters;
}
/**
* Indicates whether a parameter of the given name exists on this material.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/general:parameters">
* General: parameters</a>
*/
public boolean hasParameter(@NonNull String name) {
return nHasParameter(getNativeObject(), name);
}
/**
* Sets the value of a bool parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the material parameter
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, boolean x) {
mDefaultInstance.setParameter(name, x);
}
/**
* Sets the value of a float parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the material parameter
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, float x) {
mDefaultInstance.setParameter(name, x);
}
/**
* Sets the value of an int parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the material parameter
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, int x) {
mDefaultInstance.setParameter(name, x);
}
/**
* Sets the value of a bool2 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, boolean x, boolean y) {
mDefaultInstance.setParameter(name, x, y);
}
/**
* Sets the value of a float2 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, float x, float y) {
mDefaultInstance.setParameter(name, x, y);
}
/**
* Sets the value of an int2 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, int x, int y) {
mDefaultInstance.setParameter(name, x, y);
}
/**
* Sets the value of a bool3 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, boolean x, boolean y, boolean z) {
mDefaultInstance.setParameter(name, x, y, z);
}
/**
* Sets the value of a float3 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, float x, float y, float z) {
mDefaultInstance.setParameter(name, x, y, z);
}
/**
* Sets the value of a int3 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, int x, int y, int z) {
mDefaultInstance.setParameter(name, x, y, z);
}
/**
* Sets the value of a bool4 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
* @param w the value of the fourth component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, boolean x, boolean y, boolean z, boolean w) {
mDefaultInstance.setParameter(name, x, y, z, w);
}
/**
* Sets the value of a float4 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
* @param w the value of the fourth component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, float x, float y, float z, float w) {
mDefaultInstance.setParameter(name, x, y, z, w);
}
/**
* Sets the value of a int4 parameter on this material's default instance.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
* @param w the value of the fourth component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, int x, int y, int z, int w) {
mDefaultInstance.setParameter(name, x, y, z, w);
}
/**
* Set a bool parameter array by name.
*
* @param name name of the parameter array as defined by this Material
* @param type the number of components for each individual parameter
* @param v array of values to set to the named parameter array
* @param offset the number of elements to skip
* @param count the number of elements in the parameter array to set
*
* <p>For example, to set a parameter array of 4 bool4s:
* <pre>{@code
* boolean[] a = new boolean[4 * 4];
* material.setDefaultParameter("param", MaterialInstance.BooleanElement.BOOL4, a, 0, 4);
* }</pre>
* To only set the last 3 elements, specify an offset of 1 and a count of 3:
* <pre>{@code
* boolean[] a = new boolean[4 * 3];
* material.setDefaultParameter("param", MaterialInstance.BooleanElement.BOOL4, a, 1, 3);
* }</pre>
* </p>
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name,
@NonNull MaterialInstance.BooleanElement type, @NonNull @Size(min = 1) boolean[] v,
@IntRange(from = 0) int offset, @IntRange(from = 1) int count) {
mDefaultInstance.setParameter(name, type, v, offset, count);
}
/**
* Set an int parameter array by name.
*
* @param name name of the parameter array as defined by this Material
* @param type the number of components for each individual parameter
* @param v array of values to set to the named parameter array
* @param offset the number of elements to skip
* @param count the number of elements in the parameter array to set
*
* <p>For example, to set a parameter array of 4 int4s:
* <pre>{@code
* int[] a = new int[4 * 4];
* material.setDefaultParameter("param", MaterialInstance.IntElement.INT4, a, 0, 4);
* }</pre>
* To only set the last 3 elements, specify an offset of 1 and a count of 3:
* <pre>{@code
* int[] a = new int[4 * 3];
* material.setDefaultParameter("param", MaterialInstance.IntElement.INT4, a, 1, 3);
* }</pre>
* </p>
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name,
@NonNull MaterialInstance.IntElement type, @NonNull @Size(min = 1) int[] v,
@IntRange(from = 0) int offset, @IntRange(from = 1) int count) {
mDefaultInstance.setParameter(name, type, v, offset, count);
}
/**
* Set a float parameter array by name.
*
* @param name name of the parameter array as defined by this Material
* @param type the number of components for each individual parameter
* @param v array of values to set to the named parameter array
* @param offset the number of elements to skip
* @param count the number of elements in the parameter array to set
*
* <p>For example, to set a parameter array of 4 float4s:
* <pre>{@code
* float[] a = new float[4 * 4];
* material.setDefaultParameter("param", MaterialInstance.FloatElement.FLOAT4, a, 0, 4);
* }</pre>
* To only set the last 3 elements, specify an offset of 1 and a count of 3:
* <pre>{@code
* float[] a = new float[4 * 3];
* material.setDefaultParameter("param", MaterialInstance.FloatElement.FLOAT4, a, 1, 3);
* }</pre>
* </p>
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name,
@NonNull MaterialInstance.FloatElement type, @NonNull @Size(min = 1) float[] v,
@IntRange(from = 0) int offset, @IntRange(from = 1) int count) {
mDefaultInstance.setParameter(name, type, v, offset, count);
}
/**
* Sets the color of the given parameter on this material's default instance.
*
* @param name the name of the material color parameter
* @param type whether the color is specified in the linear or sRGB space
* @param r red component
* @param g green component
* @param b blue component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, @NonNull Colors.RgbType type,
float r, float g, float b) {
mDefaultInstance.setParameter(name, type, r, g, b);
}
/**
* Sets the color of the given parameter on this material's default instance.
*
* @param name the name of the material color parameter
* @param type whether the color is specified in the linear or sRGB space
* @param r red component
* @param g green component
* @param b blue component
* @param a alpha component
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name, @NonNull Colors.RgbaType type,
float r, float g, float b, float a) {
mDefaultInstance.setParameter(name, type, r, g, b, a);
}
/**
* Sets a texture and sampler parameter on this material's default instance.
*
* @param name The name of the material texture parameter
* @param texture The texture to set as parameter
* @param sampler The sampler to be used with this texture
*
* @see Material#getDefaultInstance()
*/
public void setDefaultParameter(@NonNull String name,
@NonNull Texture texture, @NonNull TextureSampler sampler) {
mDefaultInstance.setParameter(name, texture, sampler);
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Material");
}

View File

@@ -58,6 +58,7 @@ public class MaterialInstance {
mNativeObject = nativeMaterialInstance;
}
/** @return the {@link Material} associated with this instance */
@NonNull
public Material getMaterial() {
if (mMaterial == null) {
@@ -66,119 +67,370 @@ public class MaterialInstance {
return mMaterial;
}
/**
* Sets the value of a bool parameter.
*
* @param name the name of the material parameter
* @param x the value of the material parameter
*/
public void setParameter(@NonNull String name, boolean x) {
nSetParameterBool(getNativeObject(), name, x);
}
/**
* Sets the value of a float parameter.
*
* @param name the name of the material parameter
* @param x the value of the material parameter
*/
public void setParameter(@NonNull String name, float x) {
nSetParameterFloat(getNativeObject(), name, x);
}
/**
* Sets the value of an int parameter.
*
* @param name the name of the material parameter
* @param x the value of the material parameter
*/
public void setParameter(@NonNull String name, int x) {
nSetParameterInt(getNativeObject(), name, x);
}
/**
* Sets the value of a bool2 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
*/
public void setParameter(@NonNull String name, boolean x, boolean y) {
nSetParameterBool2(getNativeObject(), name, x, y);
}
/**
* Sets the value of a float2 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
*/
public void setParameter(@NonNull String name, float x, float y) {
nSetParameterFloat2(getNativeObject(), name, x, y);
}
/**
* Sets the value of an int2 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
*/
public void setParameter(@NonNull String name, int x, int y) {
nSetParameterInt2(getNativeObject(), name, x, y);
}
/**
* Sets the value of a bool3 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
*/
public void setParameter(@NonNull String name, boolean x, boolean y, boolean z) {
nSetParameterBool3(getNativeObject(), name, x, y, z);
}
/**
* Sets the value of a float3 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
*/
public void setParameter(@NonNull String name, float x, float y, float z) {
nSetParameterFloat3(getNativeObject(), name, x, y, z);
}
/**
* Sets the value of a int3 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
*/
public void setParameter(@NonNull String name, int x, int y, int z) {
nSetParameterInt3(getNativeObject(), name, x, y, z);
}
/**
* Sets the value of a bool4 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
* @param w the value of the fourth component
*/
public void setParameter(@NonNull String name, boolean x, boolean y, boolean z, boolean w) {
nSetParameterBool4(getNativeObject(), name, x, y, z, w);
}
/**
* Sets the value of a float4 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
* @param w the value of the fourth component
*/
public void setParameter(@NonNull String name, float x, float y, float z, float w) {
nSetParameterFloat4(getNativeObject(), name, x, y, z, w);
}
/**
* Sets the value of a int4 parameter.
*
* @param name the name of the material parameter
* @param x the value of the first component
* @param y the value of the second component
* @param z the value of the third component
* @param w the value of the fourth component
*/
public void setParameter(@NonNull String name, int x, int y, int z, int w) {
nSetParameterInt4(getNativeObject(), name, x, y, z, w);
}
/**
* Sets a texture and sampler parameter on this material's default instance.
*
* @param name The name of the material texture parameter
* @param texture The texture to set as parameter
* @param sampler The sampler to be used with this texture
*/
public void setParameter(@NonNull String name,
@NonNull Texture texture, @NonNull TextureSampler sampler) {
nSetParameterTexture(getNativeObject(), name, texture.getNativeObject(), sampler.mSampler);
}
/**
* Set a bool parameter array by name.
*
* @param name name of the parameter array as defined by this Material
* @param type the number of components for each individual parameter
* @param v array of values to set to the named parameter array
* @param offset the number of elements to skip
* @param count the number of elements in the parameter array to set
*
* <p>For example, to set a parameter array of 4 bool4s:
* <pre>{@code
* boolean[] a = new boolean[4 * 4];
* instance.setParameter("param", MaterialInstance.BooleanElement.BOOL4, a, 0, 4);
* }</pre>
* To only set the last 3 elements, specify an offset of 1 and a count of 3:
* <pre>{@code
* boolean[] a = new boolean[4 * 3];
* instance.setParameter("param", MaterialInstance.BooleanElement.BOOL4, a, 1, 3);
* }</pre>
* </p>
*/
public void setParameter(@NonNull String name,
@NonNull BooleanElement type, @NonNull boolean[] v,
@IntRange(from = 0) int offset, @IntRange(from = 1) int count) {
nSetBooleanParameterArray(getNativeObject(), name, type.ordinal(), v, offset, count);
}
/**
* Set an int parameter array by name.
*
* @param name name of the parameter array as defined by this Material
* @param type the number of components for each individual parameter
* @param v array of values to set to the named parameter array
* @param offset the number of elements to skip
* @param count the number of elements in the parameter array to set
*
* <p>For example, to set a parameter array of 4 int4s:
* <pre>{@code
* int[] a = new int[4 * 4];
* instance.setParameter("param", MaterialInstance.IntElement.INT4, a, 0, 4);
* }</pre>
* To only set the last 3 elements, specify an offset of 1 and a count of 3:
* <pre>{@code
* int[] a = new int[4 * 3];
* instance.setParameter("param", MaterialInstance.IntElement.INT4, a, 1, 3);
* }</pre>
* </p>
*/
public void setParameter(@NonNull String name,
@NonNull IntElement type, @NonNull int[] v,
@IntRange(from = 0) int offset, @IntRange(from = 1) int count) {
nSetIntParameterArray(getNativeObject(), name, type.ordinal(), v, offset, count);
}
/**
* Set a float parameter array by name.
*
* @param name name of the parameter array as defined by this Material
* @param type the number of components for each individual parameter
* @param v array of values to set to the named parameter array
* @param offset the number of elements to skip
* @param count the number of elements in the parameter array to set
*
* <p>For example, to set a parameter array of 4 float4s:
* <pre>{@code
* float[] a = new float[4 * 4];
* material.setDefaultParameter("param", MaterialInstance.FloatElement.FLOAT4, a, 0, 4);
* }</pre>
* To only set the last 3 elements, specify an offset of 1 and a count of 3:
* <pre>{@code
* float[] a = new float[4 * 3];
* material.setDefaultParameter("param", MaterialInstance.FloatElement.FLOAT4, a, 1, 3);
* }</pre>
* </p>
*/
public void setParameter(@NonNull String name,
@NonNull FloatElement type, @NonNull float[] v,
@IntRange(from = 0) int offset, @IntRange(from = 1) int count) {
nSetFloatParameterArray(getNativeObject(), name, type.ordinal(), v, offset, count);
}
/**
* Sets the color of the given parameter on this material's default instance.
*
* @param name the name of the material color parameter
* @param type whether the color is specified in the linear or sRGB space
* @param r red component
* @param g green component
* @param b blue component
*/
public void setParameter(@NonNull String name, @NonNull Colors.RgbType type,
float r, float g, float b) {
float[] color = Colors.toLinear(type, r, g, b);
nSetParameterFloat3(getNativeObject(), name, color[0], color[1], color[2]);
}
/**
* Sets the color of the given parameter on this material's default instance.
*
* @param name the name of the material color parameter
* @param type whether the color is specified in the linear or sRGB space
* @param r red component
* @param g green component
* @param b blue component
* @param a alpha component
*/
public void setParameter(@NonNull String name, @NonNull Colors.RgbaType type,
float r, float g, float b, float a) {
float[] color = Colors.toLinear(type, r, g, b, a);
nSetParameterFloat4(getNativeObject(), name, color[0], color[1], color[2], color[3]);
}
/**
* Set up a custom scissor rectangle; by default this encompasses the View.
*
* @param left left coordinate of the scissor box
* @param bottom bottom coordinate of the scissor box
* @param width width of the scissor box
* @param height height of the scissor box
*/
public void setScissor(@IntRange(from = 0) int left, @IntRange(from = 0) int bottom,
@IntRange(from = 0) int width, @IntRange(from = 0) int height) {
nSetScissor(getNativeObject(), left, bottom, width, height);
}
/** Returns the scissor rectangle to its default setting, which encompasses the View. */
public void unsetScissor() {
nUnsetScissor(getNativeObject());
}
/**
* Sets a polygon offset that will be applied to all renderables drawn with this material
* instance.
*
* The value of the offset is scale * dz + r * constant, where dz is the change in depth
* relative to the screen area of the triangle, and r is the smallest value that is guaranteed
* to produce a resolvable offset for a given implementation. This offset is added before the
* depth test.
*
* @warning using a polygon offset other than zero has a significant negative performance
* impact, as most implementations have to disable early depth culling. DO NOT USE unless
* absolutely necessary.
*
* @param scale scale factor used to create a variable depth offset for each triangle
* @param constant scale factor used to create a constant depth offset for each triangle
*/
public void setPolygonOffset(float scale, float constant) {
nSetPolygonOffset(getNativeObject(), scale, constant);
}
/**
* Overrides the minimum alpha value a fragment must have to not be discarded when the blend
* mode is MASKED. Defaults to 0.4 if it has not been set in the parent Material. The specified
* value should be between 0 and 1 and will be clamped if necessary.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/blendingandtransparency:maskthreshold">
* Blending and transparency: maskThreshold</a>
*/
public void setMaskThreshold(float threshold) {
nSetMaskThreshold(getNativeObject(), threshold);
}
/**
* Sets the screen space variance of the filter kernel used when applying specular
* anti-aliasing. The default value is set to 0.15. The specified value should be between
* 0 and 1 and will be clamped if necessary.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/anti-aliasing:specularantialiasingvariance">
* Anti-aliasing: specularAntiAliasingVariance</a>
*/
public void setSpecularAntiAliasingVariance(float variance) {
nSetSpecularAntiAliasingVariance(getNativeObject(), variance);
}
/**
* Sets the clamping threshold used to suppress estimation errors when applying specular
* anti-aliasing. The default value is set to 0.2. The specified value should be between 0
* and 1 and will be clamped if necessary.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/anti-aliasing:specularantialiasingthreshold">
* Anti-aliasing: specularAntiAliasingThreshold</a>
*/
public void setSpecularAntiAliasingThreshold(float threshold) {
nSetSpecularAntiAliasingThreshold(getNativeObject(), threshold);
}
/**
* Enables or disables double-sided lighting if the parent Material has double-sided capability,
* otherwise prints a warning. If double-sided lighting is enabled, backface culling is
* automatically disabled.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:doublesided">
* Rasterization: doubleSided</a>
*/
public void setDoubleSided(boolean doubleSided) {
nSetDoubleSided(getNativeObject(), doubleSided);
}
long getNativeObject() {
/**
* Overrides the default triangle culling state that was set on the material.
*
* @see
* <a href="https://google.github.io/filament/Materials.html#materialdefinitions/materialblock/rasterization:culling">
* Rasterization: culling</a>
*/
public void setCullingMode(Material.CullingMode mode) {
nSetCullingMode(getNativeObject(), mode.ordinal());
}
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed MaterialInstance");
}
@@ -247,4 +499,6 @@ public class MaterialInstance {
float threshold);
private static native void nSetDoubleSided(long nativeMaterialInstance, boolean doubleSided);
private static native void nSetCullingMode(long nativeMaterialInstance, long mode);
}

View File

@@ -23,6 +23,29 @@ import android.support.annotation.Size;
public final class MathUtils {
private MathUtils() { }
/**
* Packs the tangent frame represented by the specified tangent, bitangent, and normal into a
* quaternion.
*
* <p>
* Reflection is preserved by encoding it as the sign of the w component in the resulting
* quaternion. Since -0 cannot always be represented on the GPU, this function computes a bias
* to ensure values are always either positive or negative, never 0. The bias is computed based
* on a per-element storage size of 2 bytes, making the resulting quaternion suitable for
* storage into an SNORM16 vector.
* </p>
*
* @param tangentX the X component of the tangent
* @param tangentY the Y component of the tangent
* @param tangentZ the Z component of the tangent
* @param bitangentX the X component of the bitangent
* @param bitangentY the Y component of the bitangent
* @param bitangentZ the Z component of the bitangent
* @param normalX the X component of the normal
* @param normalY the Y component of the normal
* @param normalZ the Z component of the normal
* @param quaternion a float array of at least size 4 for the quaternion result to be stored
*/
public static void packTangentFrame(
float tangentX, float tangentY, float tangentZ,
float bitangentX, float bitangentY, float bitangentZ,
@@ -34,6 +57,30 @@ public final class MathUtils {
normalX, normalY, normalZ, quaternion, 0);
}
/**
* Packs the tangent frame represented by the specified tangent, bitangent, and normal into a
* quaternion.
*
* <p>
* Reflection is preserved by encoding it as the sign of the w component in the resulting
* quaternion. Since -0 cannot always be represented on the GPU, this function computes a bias
* to ensure values are always either positive or negative, never 0. The bias is computed based
* on a per-element storage size of 2 bytes, making the resulting quaternion suitable for
* storage into an SNORM16 vector.
* </p>
*
* @param tangentX the X component of the tangent
* @param tangentY the Y component of the tangent
* @param tangentZ the Z component of the tangent
* @param bitangentX the X component of the bitangent
* @param bitangentY the Y component of the bitangent
* @param bitangentZ the Z component of the bitangent
* @param normalX the X component of the normal
* @param normalY the Y component of the normal
* @param normalZ the Z component of the normal
* @param quaternion a float array of at least size 4 for the quaternion result to be stored
* @param offset offset, in elements, into the quaternion array to store the results
*/
public static void packTangentFrame(
float tangentX, float tangentY, float tangentZ,
float bitangentX, float bitangentY, float bitangentZ,

View File

@@ -17,6 +17,7 @@
package com.google.android.filament;
import android.support.annotation.NonNull;
import com.google.android.filament.proguard.UsedByNative;
import java.nio.Buffer;

View File

@@ -20,9 +20,16 @@ import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import java.nio.Buffer;
import java.nio.BufferOverflowException;
/**
* An offscreen render target that can be associated with a {@link View} and contains
* weak references to a set of attached {@link Texture} objects.
*
* <p>
* Clients are responsible for the lifetime of all associated <code>Texture</code> attachments.
* </p>
*
* @see View
*/
public class RenderTarget {
private long mNativeObject;
private final Texture[] mTextures = new Texture[2];
@@ -33,18 +40,24 @@ public class RenderTarget {
mTextures[1] = builder.mTextures[1];
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed RenderTarget");
}
return mNativeObject;
}
/**
* An attachment point is a slot that can be assigned to a {@link Texture}.
*/
public enum AttachmentPoint {
COLOR,
DEPTH,
}
/**
* Constructs <code>RenderTarget</code> objects using a builder pattern.
*/
public static class Builder {
@SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"})
private final BuilderFinalizer mFinalizer;
@@ -56,6 +69,15 @@ public class RenderTarget {
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Sets a texture to a given attachment point.
*
* <p>All RenderTargets must have a non-null <code>COLOR</code> attachment.</p>
*
* @param attachment The attachment point of the texture.
* @param texture The associated texture object.
* @return A reference to this Builder for chaining calls.
*/
@NonNull
public Builder texture(@NonNull AttachmentPoint attachment, @Nullable Texture texture) {
mTextures[attachment.ordinal()] = texture;
@@ -63,24 +85,51 @@ public class RenderTarget {
return this;
}
/**
* Sets the mipmap level for a given attachment point.
*
* @param attachment The attachment point of the texture.
* @param level The associated mipmap level, 0 by default.
* @return A reference to this Builder for chaining calls.
*/
@NonNull
public Builder mipLevel(@NonNull AttachmentPoint attachment, @IntRange(from = 0) int level) {
nBuilderMipLevel(mNativeBuilder, attachment.ordinal(), level);
return this;
}
/**
* Sets the cubemap face for a given attachment point.
*
* @param attachment The attachment point.
* @param face The associated cubemap face.
* @return A reference to this Builder for chaining calls.
*/
@NonNull
public Builder face(@NonNull AttachmentPoint attachment, Texture.CubemapFace face) {
nBuilderFace(mNativeBuilder, attachment.ordinal(), face.ordinal());
return this;
}
/**
* Sets the layer for a given attachment point (for 3D textures).
*
* @param attachment The attachment point.
* @param layer The associated cubemap layer.
* @return A reference to this Builder for chaining calls.
*/
@NonNull
public Builder layer(@NonNull AttachmentPoint attachment, @IntRange(from = 0) int layer) {
nBuilderLayer(mNativeBuilder, attachment.ordinal(), layer);
return this;
}
/**
* Creates the RenderTarget object and returns a pointer to it.
*
* @return pointer to the newly created object or nullptr if exceptions are disabled and
* an error occurred.
*/
@NonNull
public RenderTarget build(@NonNull Engine engine) {
long nativeRenderTarget = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -108,20 +157,45 @@ public class RenderTarget {
}
}
/**
* Gets the texture set on the given attachment point.
*
* @param attachment Attachment point
* @return A Texture object or nullptr if no texture is set for this attachment point
*/
@Nullable
public Texture getTexture(@NonNull AttachmentPoint attachment) {
return mTextures[attachment.ordinal()];
}
/**
* Returns the mipmap level set on the given attachment point.
*
* @param attachment Attachment point
* @return the mipmap level set on the given attachment point
*/
@IntRange(from = 0)
public int getMipLevel(@NonNull AttachmentPoint attachment) {
return nGetMipLevel(getNativeObject(), attachment.ordinal());
}
/**
* Returns the face of a cubemap set on the given attachment point.
*
* @param attachment Attachment point
* @return A cubemap face identifier. This is only relevant if the attachment's texture is
* a cubemap.
*/
public Texture.CubemapFace getFace(AttachmentPoint attachment) {
return Texture.CubemapFace.values()[nGetFace(getNativeObject(), attachment.ordinal())];
}
/**
* Returns the texture-layer set on the given attachment point.
*
* @param attachment Attachment point
* @return A texture layer. This is only relevant if the attachment's texture is a 3D texture.
*/
@IntRange(from = 0)
public int getLayer(@NonNull AttachmentPoint attachment) {
return nGetLayer(getNativeObject(), attachment.ordinal());

View File

@@ -23,12 +23,51 @@ import java.nio.Buffer;
import java.nio.BufferOverflowException;
import java.nio.ReadOnlyBufferException;
/**
* A <code>Renderer</code> instance represents an operating system's window.
*
* <p>
* Typically, applications create a <code>Renderer</code> per window. The <code>Renderer</code> generates
* drawing commands for the render thread and manages frame latency.
* <br>
* A Renderer generates drawing commands from a View, itself containing a Scene description.
* </p>
*
* <h1>Creation and Destruction</h1>
*
* <p>A <code>Renderer</code> is created using {@link Engine#createRenderer} and destroyed
* using {@link Engine#destroyRenderer}.</p>
*
* @see Engine
* @see View
*/
public class Renderer {
private final Engine mEngine;
private long mNativeObject;
/**
* Indicates that the <code>dstSwapChain</code> passed into {@link #copyFrame} should be
* committed after the frame has been copied.
*
* @see #copyFrame
*/
public static final int MIRROR_FRAME_FLAG_COMMIT = 0x1;
/**
* Indicates that the presentation time should be set on the <code>dstSwapChain</code>
* passed into {@link #copyFrame} to the monotonic clock time when the frame is
* copied.
*
* @see #copyFrame
*/
public static final int MIRROR_FRAME_FLAG_SET_PRESENTATION_TIME = 0x2;
/**
* Indicates that the <code>dstSwapChain</code> passed into {@link #copyFrame} should be
* cleared to black before the frame is copied into the specified viewport.
*
* @see #copyFrame
*/
public static final int MIRROR_FRAME_FLAG_CLEAR = 0x4;
Renderer(@NonNull Engine engine, long nativeRenderer) {
@@ -36,25 +75,128 @@ public class Renderer {
mNativeObject = nativeRenderer;
}
/**
* Gets the {@link Engine} that created this <code>Renderer</code>.
*
* @return {@link Engine} instance this <code>Renderer</code> is associated to.
*/
@NonNull
public Engine getEngine() {
return mEngine;
}
/**
* Sets up a frame for this <code>Renderer</code>.
* <p><code>beginFrame</code> manages frame pacing, and returns whether or not a frame should be
* drawn. The goal of this is to skip frames when the GPU falls behind in order to keep the frame
* latency low.</p>
*
* <p>If a given frame takes too much time in the GPU, the CPU will get ahead of the GPU. The
* display will draw the same frame twice producing a stutter. At this point, the CPU is
* ahead of the GPU and depending on how many frames are buffered, latency increases.
* beginFrame() attempts to detect this situation and returns <code>false</code> in that case,
* indicating to the caller to skip the current frame.</p>
*
* <p>All calls to render() must happen <b>after</b> beginFrame().</p>
*
* @param swapChain the {@link SwapChain} instance to use
*
* @return <code>false</code> if the current frame must be skipped<br>
* When skipping a frame, the whole frame is canceled, and {@link #endFrame} must not
* be called.
*
* @see #endFrame
* @see #render
*/
public boolean beginFrame(@NonNull SwapChain swapChain) {
return nBeginFrame(getNativeObject(), swapChain.getNativeObject());
}
/**
* Finishes the current frame and schedules it for display.
* <p>
* <code>endFrame()</code> schedules the current frame to be displayed on the
* <code>Renderer</code>'s window.
* </p>
*
* <br><p>All calls to render() must happen <b>before</b> endFrame().</p>
*
* @see #beginFrame
* @see #render
*/
public void endFrame() {
nEndFrame(getNativeObject());
}
/**
* Renders a {@link View} into this <code>Renderer</code>'s window.
*
* <p>
* This is filament's main rendering method, most of the CPU-side heavy lifting is performed
* here. The purpose of the <code>render()</code> function is to generate render commands which
* are asynchronously executed by the {@link Engine}'s render thread.
* </p>
*
* <p><code>render()</code> generates commands for each of the following stages:</p>
* <ul>
* <li>Shadow map pass, if needed (currently only a single shadow map is supported)</li>
* <li>Depth pre-pass</li>
* <li>SSAO pass, if enabled</li>
* <li>Color pass</li>
* <li>Post-processing pass</li>
* </ul>
*
* A typical render loop looks like this:
*
* <pre>
* void renderLoop(Renderer renderer, SwapChain swapChain) {
* do {
* // typically we wait for VSYNC and user input events
* if (renderer.beginFrame(swapChain)) {
* renderer.render(mView);
* renderer.endFrame();
* }
* } while (!quit());
* }
* </pre>
*
* <ul>
*<li><code>render()</code> must be called <b>after</b> {@link #beginFrame} and <b>before</b>
*{@link #endFrame}.</li>
*
*<li><code>render()</code> must be called from the {@link Engine}'s main thread
*(or external synchronization must be provided). In particular, calls to <code>render()</code>
*on different <code>Renderer</code> instances <b>must</b> be synchronized.</li>
*
*<li><code>render()</code> performs potentially heavy computations and cannot be multi-threaded.
*However, internally, it is highly multi-threaded to both improve performance and mitigate
*the call's latency.</li>
*
*<li><code>render()</code> is typically called once per frame (but not necessarily).</li>
* </ul>
*
* @param view the {@link View} to render
*
* @see #beginFrame
* @see #endFrame
* @see View
*
*/
public void render(@NonNull View view) {
nRender(getNativeObject(), view.getNativeObject());
}
/**
* This method MUST be called before endFrame.
* Copies the currently rendered {@link View} to the indicated {@link SwapChain}, using the
* indicated source and destination rectangle.
*
* <p><code>copyFrame()</code> should be called after a frame is rendered using {@link #render}
* but before {@link #endFrame} is called.</p>
*
* @param dstSwapChain the {@link SwapChain} into which the frame should be copied
* @param dstViewport the destination rectangle in which to draw the view
* @param srcViewport the source rectangle to be copied
* @param flags one or more <code>CopyFrameFlag</code> behavior configuration flags
*/
public void copyFrame(
@NonNull SwapChain dstSwapChain, @NonNull Viewport dstViewport,
@@ -73,7 +215,68 @@ public class Renderer {
}
/**
* This method MUST be called before endFrame.
* Reads back the content of the {@link SwapChain} associated with this <code>Renderer</code>.
*
*<pre>
*
* Framebuffer as seen on User buffer (PixelBufferDescriptor)
* screen
* +--------------------+
* | | .stride .alignment
* | | ----------------------->-->
* | | O----------------------+--+ low addresses
* | | | | | |
* | w | | | .top | |
* | <---------> | | V | |
* | +---------+ | | +---------+ | |
* | | ^ | | ======> | | | | |
* | x | h| | | |.left| | | |
* +------>| v | | +---->| | | |
* | +.........+ | | +.........+ | |
* | ^ | | | |
* | y | | +----------------------+--+ high addresses
* O------------+-------+
*
*</pre>
*
*
* <p>Typically <code>readPixels</code> will be called after {@link #render} and before
* {@link #endFrame}.</p>
* <br>
* <p>After calling this method, the callback associated with <code>buffer</code>
* will be invoked on the main thread, indicating that the read-back has completed.
* Typically, this will happen after multiple calls to {@link #beginFrame},
* {@link #render}, {@link #endFrame}.</p>
* <br>
* <p><code>readPixels</code> is intended for debugging and testing.
* It will impact performance significantly.</p>
*
* @param xoffset left offset of the sub-region to read back
* @param yoffset bottom offset of the sub-region to read back
* @param width width of the sub-region to read back
* @param height height of the sub-region to read back
* @param buffer client-side buffer where the read-back will be written
*
* <p>
* The following format are always supported:
* <li>{@link Texture.Format#RGBA}</li>
* <li>{@link Texture.Format#RGBA_INTEGER}</li>
* </p>
*
* <p>
* The following types are always supported:
* <li>{@link Texture.Type#UBYTE}</li>
* <li>{@link Texture.Type#UINT}</li>
* <li>{@link Texture.Type#INT}</li>
* <li>{@link Texture.Type#FLOAT}</li>
* </p>
*
* <p>Other combination of format/type may be supported. If a combination is
* not supported, this operation may fail silently. Use a DEBUG build
* to get some logs about the failure.</p>
*
* @exception BufferOverflowException if the specified parameters would result in reading
* outside of <code>buffer</code>.
*/
public void readPixels(
@IntRange(from = 0) int xoffset, @IntRange(from = 0) int yoffset,
@@ -96,15 +299,71 @@ public class Renderer {
}
}
double getUserTime() {
/**
* Returns a timestamp (in seconds) for the last call to {@link #beginFrame}. This value is
* constant for all {@link View views} rendered during a frame. The epoch is set with
* {@link #resetUserTime}.
* <br>
* <p>In materials, this value can be queried using <code>vec4 getUserTime()</code>. The value
* returned is a <code>highp vec4</code> encoded as follows:</p>
* <pre>
* time.x = (float)Renderer.getUserTime();
* time.y = Renderer.getUserTime() - time.x;
* </pre>
*
* It follows that the following invariants are true:
* <pre>
* (double)time.x + (double)time.y == Renderer.getUserTime()
* time.x == (float)Renderer.getUserTime()
* </pre>
*
* This encoding allows the shader code to perform high precision (i.e. double) time
* calculations when needed despite the lack of double precision in the shader, e.g.:
* <br>
* To compute <code>(double)time * vertex</code> in the material, use the following construct:
* <pre>
* vec3 result = time.x * vertex + time.y * vertex;
* </pre>
*
* Most of the time, high precision computations are not required, but be aware that the
* precision of <code>time.x</code> rapidly diminishes as time passes:
*
* <center>
* <table border="1">
* <tr align="center"><th> time </th><th> precision </th></tr>
* <tr align="center"><td> 16.7s </td><td> us </td></tr>
* <tr align="center"><td> 4h39.7s </td><td> ms </td></tr>
* <tr align="center"><td> 77h </td><td> 1/60s </td></tr>
* </table>
* </center>
* <p>
*
* In other words, it is only possible to get microsecond accuracy for about 16s or millisecond
* accuracy for just under 5h. This problem can be mitigated by calling {@link #resetUserTime},
* or using high precision time as described above.
*
* @return the time in seconds since {@link #resetUserTime} was last called
*
* @see #resetUserTime
*/
public double getUserTime() {
return nGetUserTime(getNativeObject());
}
void resetUserTime() {
/**
* Sets the user time epoch to now, i.e. resets the user time to zero.
* <br>
* <p>Use this method used to keep the precision of time high in materials, in practice it should
* be called at least when the application is paused, e.g.
* {@link android.app.Activity#onPause() Activity.onPause} in Android.</p>
*
* @see #getUserTime
*/
public void resetUserTime() {
nResetUserTime(getNativeObject());
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Renderer");
}

View File

@@ -18,6 +18,28 @@ package com.google.android.filament;
import android.support.annotation.Nullable;
/**
* A <code>Scene</code> is a flat container of {@link RenderableManager} and {@link LightManager}
* components.
* <br>
* <p>A <code>Scene</code> doesn't provide a hierarchy of objects, i.e.: it's not a scene-graph.
* However, it manages the list of objects to render and the list of lights. These can
* be added or removed from a <code>Scene</code> at any time.
* Moreover clients can use {@link TransformManager} to create a graph of transforms.</p>
* <br>
* <p>A {@link RenderableManager} component <b>must</b> be added to a <code>Scene</code> in order
* to be rendered, and the <code>Scene</code> must be provided to a {@link View}.</p>
*
* <h1>Creation and Destruction</h1>
*
* A <code>Scene</code> is created using {@link Engine#createScene} and destroyed using
* {@link Engine#destroyScene(Scene)}.
*
* @see View
* @see LightManager
* @see RenderableManager
* @see TransformManager
*/
public class Scene {
private long mNativeObject;
private @Nullable Skybox mSkybox;
@@ -27,35 +49,78 @@ public class Scene {
mNativeObject = nativeScene;
}
/**
* @return the {@link Skybox} or <code>null</code> if none is set
* @see #setSkybox(Skybox)
*/
@Nullable
public Skybox getSkybox() {
return mSkybox;
}
/**
* Sets the {@link Skybox}.
*
* The {@link Skybox} is drawn last and covers all pixels not touched by geometry.
*
* @param skybox the {@link Skybox} to use to fill untouched pixels,
* or <code>null</code> to unset the {@link Skybox}.
*/
public void setSkybox(@Nullable Skybox skybox) {
mSkybox = skybox;
nSetSkybox(getNativeObject(), mSkybox != null ? mSkybox.getNativeObject() : 0);
}
/**
* @return the {@link IndirectLight} or <code>null</code> if none is set
* @see #setIndirectLight(IndirectLight)
*/
@Nullable
public IndirectLight getIndirectLight() {
return mIndirectLight;
}
/**
* Sets the {@link IndirectLight} to use when rendering the <code>Scene</code>.
*
* Currently, a <code>Scene</code> may only have a single {@link IndirectLight}.
* This call replaces the current {@link IndirectLight}.
*
* @param ibl the {@link IndirectLight} to use when rendering the <code>Scene</code>
* or <code>null</code> to unset.
*/
public void setIndirectLight(@Nullable IndirectLight ibl) {
mIndirectLight = ibl;
nSetIndirectLight(getNativeObject(),
mIndirectLight != null ? mIndirectLight.getNativeObject() : 0);
}
/**
* Adds an {@link Entity} to the <code>Scene</code>.
*
* @param entity the entity is ignored if it doesn't have a {@link RenderableManager} component
* or {@link LightManager} component.<br>
* A given {@link Entity} object can only be added once to a <code>Scene</code>.
*/
public void addEntity(@Entity int entity) {
nAddEntity(getNativeObject(), entity);
}
/**
* Adds a list of entities to the <code>Scene</code>.
*
* @param entities array containing entities to add to the <code>Scene</code>.
*/
public void addEntities(@Entity int[] entities) {
nAddEntities(getNativeObject(), entities);
}
/**
* Removes an {@link Entity} from the <code>Scene</code>.
*
* @param entity the {@link Entity} to remove from the <code>Scene</code>. If the specified
* <code>entity</code> doesn't exist, this call is ignored.
*/
public void removeEntity(@Entity int entity) {
nRemove(getNativeObject(), entity);
}
@@ -67,15 +132,25 @@ public class Scene {
removeEntity(entity);
}
/**
* Returns the number of {@link RenderableManager} components in the <code>Scene</code>.
*
* @return number of {@link RenderableManager} components in the <code>Scene</code>..
*/
public int getRenderableCount() {
return nGetRenderableCount(getNativeObject());
}
/**
* Returns the number of {@link LightManager} components in the <code>Scene</code>.
*
* @return number of {@link LightManager} components in the <code>Scene</code>..
*/
public int getLightCount() {
return nGetLightCount(getNativeObject());
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Scene");
}

View File

@@ -16,11 +16,36 @@
package com.google.android.filament;
import com.google.android.filament.proguard.UsedByReflection;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import com.google.android.filament.proguard.UsedByReflection;
/**
* Skybox
* <p>When added to a {@link Scene}, the <code>Skybox</code> fills all untouched pixels.</p>
*
* <h1>Creation and destruction</h1>
*
* A <code>Skybox</code> object is created using the {@link Skybox.Builder} and destroyed by calling
* {@link Engine#destroySkybox}.<br>
* <pre>
* Engine engine = Engine.create();
*
* Scene scene = engine.createScene();
*
* Skybox skybox = new Skybox.Builder()
* .environment(cubemap)
* .build(engine);
*
* scene.setSkybox(skybox);
* </pre>
*
* Currently only {@link Texture} based sky boxes are supported.
*
* @see Scene
* @see IndirectLight
*/
public class Skybox {
private long mNativeObject;
@@ -29,28 +54,88 @@ public class Skybox {
mNativeObject = nativeSkybox;
}
/**
* Use <code>Builder</code> to construct a <code>Skybox</code> object instance.
*/
public static class Builder {
@SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"}) // Keep to finalize native resources
private final BuilderFinalizer mFinalizer;
private final long mNativeBuilder;
/**
* Use <code>Builder</code> to construct a <code>Skybox</code> object instance.
*/
public Builder() {
mNativeBuilder = nCreateBuilder();
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Set the environment map (i.e. the skybox content).
*
* <p>The <code>Skybox</code> is rendered as though it were an infinitely large cube with the
* camera inside it. This means that the cubemap which is mapped onto the cube's exterior
* will appear mirrored. This follows the OpenGL conventions.</p>
*
* <p>The <code>cmgen</code> tool generates reflection maps by default which are therefore
* ideal to use as skyboxes.</p>
*
* @param cubemap A cubemap {@link Texture}
*
* @return This Builder, for chaining calls.
*
* @see Texture
*/
@NonNull
public Builder environment(@NonNull Texture texture) {
nBuilderEnvironment(mNativeBuilder, texture.getNativeObject());
public Builder environment(@NonNull Texture cubemap) {
nBuilderEnvironment(mNativeBuilder, cubemap.getNativeObject());
return this;
}
/**
* Indicates whether the sun should be rendered. The sun can only be
* rendered if there is at least one light of type {@link LightManager.Type#SUN} in
* the {@link Scene}. The default value is <code>false</code>.
*
* @param show <code>true</code> if the sun should be rendered, <code>false</code> otherwise
*
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder showSun(boolean show) {
nBuilderShowSun(mNativeBuilder, show);
return this;
}
/**
* Sets the <code>Skybox</code> intensity when no {@link IndirectLight} is set
*
* <p>This call is ignored when an {@link IndirectLight} is set, otherwise it is used in
* its place.</p>
*
* @param envIntensity Scale factor applied to the skybox texel values such that
* the result is in cd/m<sup>2</sup> (lux) units (default = 30000)
*
* @return This Builder, for chaining calls.
*
* @see IndirectLight.Builder#intensity
*/
@NonNull
public Builder intensity(float envIntensity) {
nBuilderIntensity(mNativeBuilder, envIntensity);
return this;
}
/**
* Creates a <code>Skybox</code> object
*
* @param engine the {@link Engine} to associate this <code>Skybox</code> with.
*
* @return A newly created <code>Skybox</code>object
*
* @exception IllegalStateException can be thrown if the <code>Skybox</code> couldn't be created
*/
@NonNull
public Skybox build(@NonNull Engine engine) {
long nativeSkybox = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -75,15 +160,36 @@ public class Skybox {
}
}
public void setLayerMask(@IntRange(from = 0, to = 255) int select, @IntRange(from = 0, to = 255) int value) {
nSetLayerMask(getNativeObject(), select & 0xff, value & 0xff);
/**
* Sets bits in a visibility mask. By default, this is <code>0x1</code>.
* <p>This provides a simple mechanism for hiding or showing this <code>Skybox</code> in a
* {@link Scene}.</p>
*
* <p>For example, to set bit 1 and reset bits 0 and 2 while leaving all other bits unaffected,
* call: <code>setLayerMask(7, 2)</code>.</p>
*
* @param select the set of bits to affect
* @param values the replacement values for the affected bits
*
* @see View#setVisibleLayers
*/
public void setLayerMask(@IntRange(from = 0, to = 255) int select, @IntRange(from = 0, to = 255) int values) {
nSetLayerMask(getNativeObject(), select & 0xff, values & 0xff);
}
/**
* @return the visibility mask bits
*/
public int getLayerMask() {
return nGetLayerMask(getNativeObject());
}
long getNativeObject() {
/**
* Returns the <code>Skybox</code>'s intensity in cd/m<sup>2</sup>.
*/
public float getIntensity() { return nGetIntensity(getNativeObject()); }
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Skybox");
}
@@ -98,7 +204,9 @@ public class Skybox {
private static native void nDestroyBuilder(long nativeSkyboxBuilder);
private static native void nBuilderEnvironment(long nativeSkyboxBuilder, long nativeTexture);
private static native void nBuilderShowSun(long nativeSkyboxBuilder, boolean show);
private static native void nBuilderIntensity(long nativeSkyboxBuilder, float intensity);
private static native long nBuilderBuild(long nativeSkyboxBuilder, long nativeEngine);
private static native void nSetLayerMask(long nativeSkybox, int select, int value);
private static native int nGetLayerMask(long nativeSkybox);
private static native float nGetIntensity(long nativeSkybox);
}

View File

@@ -23,6 +23,12 @@ import java.nio.Buffer;
import java.nio.BufferOverflowException;
import java.nio.ReadOnlyBufferException;
/**
* <code>Stream</code> is used to attach a native video stream to a filament {@link Texture}.
*
* @see Texture#setExternalStream
* @see Engine#destroyStream
*/
public class Stream {
private long mNativeObject;
private long mNativeEngine;
@@ -32,20 +38,31 @@ public class Stream {
mNativeEngine = engine.getNativeObject();
}
/**
* Use <code>Builder</code> to construct an Stream object instance.
*/
public static class Builder {
@SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"}) // Keep to finalize native resources
private final BuilderFinalizer mFinalizer;
private final long mNativeBuilder;
/**
* Use <code>Builder</code> to construct an Stream object instance.
*/
public Builder() {
mNativeBuilder = nCreateBuilder();
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Accepted types for the stream source:
* - Android: SurfaceView
* - Other: none
* Creates a native stream. Native streams can sample data directly from an
* opaque platform object such as a {@link android.graphics.SurfaceTexture SurfaceTexture}
* on Android.
*
* @param streamSource an opaque native stream handle, e.g.: on Android this must be a
* {@link android.graphics.SurfaceTexture SurfaceTexture} object
* @return This Builder, for chaining calls.
* @see Texture#setExternalStream
*/
@NonNull
public Builder stream(@NonNull Object streamSource) {
@@ -56,24 +73,57 @@ public class Stream {
throw new IllegalArgumentException("Invalid stream source: " + streamSource);
}
/**
* Creates a copy stream. A copy stream will sample data from the supplied
* external texture and copy it into an internal private texture.
*
* <p>Currently only OpenGL external texture ids are supported.</p>
*
* @param externalTextureId An opaque texture id (typically a GLuint created with
* <code>glGenTextures()</code>) in a context shared with
* filament -- in that case this texture's target must be
* <code>GL_TEXTURE_EXTERNAL_OES.</code>
* @return This Builder, for chaining calls.
* @see Texture#setExternalStream
*/
@NonNull
public Builder stream(long externalTextureId) {
nBuilderStream(mNativeBuilder, externalTextureId);
return this;
}
/**
* @param width initial width of the incoming stream. Whether this value is used is
* stream dependent. On Android, it must be set when using
* {@link #stream(long)}
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder width(int width) {
nBuilderWidth(mNativeBuilder, width);
return this;
}
/**
* @param height initial height of the incoming stream. Whether this value is used is
* stream dependent. On Android, it must be set when using
* {@link #stream(long)}
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder height(int height) {
nBuilderHeight(mNativeBuilder, height);
return this;
}
/**
* Creates a new <code>Stream</code> object instance.
*
* @param engine {@link Engine} instance to associate this <code>Stream</code> with.
*
* @return newly created <code>Stream</code> object
* @exception IllegalStateException if the <code>Stream</code> couldn't be created
*/
@NonNull
public Stream build(@NonNull Engine engine) {
long nativeStream = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -99,14 +149,92 @@ public class Stream {
}
}
/**
* Indicates whether this <code>Stream</code> is a native stream or a copy stream.
*
* @return true if this is a native <code>Stream</code>, false otherwise.
*/
public boolean isNative() {
return nIsNative(getNativeObject());
}
/**
* Updates the size of the incoming stream. Whether this value is used is
* stream dependent. On Android, it must be set when using
* {@link Builder#stream(long)}
*
* @param width new width of the incoming stream
* @param height new height of the incoming stream
*/
public void setDimensions(@IntRange(from = 0) int width, @IntRange(from = 0) int height) {
nSetDimensions(getNativeObject(), width, height);
}
/**
* Reads back the content of the last frame of a <code>Stream</code> since the last call to
* {@link Renderer#beginFrame}.
*
* <p>The Stream must be a copy stream, which can be checked with {@link #isNative()}.
* This function is a no-op otherwise.</p>
*
* <pre>
*
* Stream buffer User buffer (PixelBufferDescriptor)
* +--------------------+
* | | .stride .alignment
* | | ----------------------->-->
* | | O----------------------+--+ low addresses
* | | | | | |
* | w | | | .top | |
* | <---------> | | V | |
* | +---------+ | | +---------+ | |
* | | ^ | | ======> | | | | |
* | x | h| | | |.left| | | |
* +------>| v | | +---->| | | |
* | +.........+ | | +.........+ | |
* | ^ | | | |
* | y | | +----------------------+--+ high addresses
* O------------+-------+
*
* </pre>
*
* <p>Typically readPixels() will be called after {@link Renderer#beginFrame}.</p>
*
* <p>After calling this method, the callback associated with <code>buffer</code>
* will be invoked on the main thread, indicating that the read-back has completed.
* Typically, this will happen after multiple calls to {@link Renderer#beginFrame},
* {@link Renderer#render}, {@link Renderer#endFrame}.</p>
*
* <p><code>readPixels</code> is intended for debugging and testing.
* It will impact performance significantly.</p>
*
* @param xoffset left offset of the sub-region to read back
* @param yoffset bottom offset of the sub-region to read back
* @param width width of the sub-region to read back
* @param height height of the sub-region to read back
* @param buffer client-side buffer where the read-back will be written
*
* <p>
* The following format are always supported:
* <li>{@link Texture.Format#RGBA}</li>
* <li>{@link Texture.Format#RGBA_INTEGER}</li>
* </p>
*
* <p>
* The following types are always supported:
* <li>{@link Texture.Type#UBYTE}</li>
* <li>{@link Texture.Type#UINT}</li>
* <li>{@link Texture.Type#INT}</li>
* <li>{@link Texture.Type#FLOAT}</li>
* </p>
*
* <p>Other combination of format/type may be supported. If a combination is
* not supported, this operation may fail silently. Use a DEBUG build
* to get some logs about the failure.</p>
*
* @exception BufferOverflowException if the specified parameters would result in reading
* outside of <code>buffer</code>.
*/
public void readPixels(
@IntRange(from = 0) int xoffset, @IntRange(from = 0) int yoffset,
@IntRange(from = 0) int width, @IntRange(from = 0) int height,
@@ -128,11 +256,18 @@ public class Stream {
}
}
/**
* Returns the presentation time of the currently displayed frame in nanosecond.
*
* This value can change at any time.
*
* @return timestamp in nanosecond.
*/
public long getTimestamp() {
return nGetTimestamp(getNativeObject());
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Stream");
}

View File

@@ -18,12 +18,71 @@ package com.google.android.filament;
import android.support.annotation.NonNull;
/**
* A <code>SwapChain</code> represents an Operating System's <b>native</b> renderable surface.
*
* <p>Typically it's a native window or a view. Because a <code>SwapChain</code> is initialized
* from a native object, it is given to filament as an <code>Object</code>, which must be of the
* proper type for each platform filament is running on.</p>
*
* <code>
* SwapChain swapChain = engine.createSwapChain(nativeWindow);
* </code>
*
* <p>The <code>nativeWindow</code> parameter above must be of type:</p>
*
* <center>
* <table border="1">
* <tr><th> Platform </th><th> nativeWindow type </th></tr>
* <tr><td> Android </td><td>{@link android.view.Surface Surface}</td></tr>
* </table>
* </center>
* <p>
*
* <h1>Examples</h1>
*
* <h2>Android</h2>
*
*
* <p>A {@link android.view.Surface Surface} can be retrieved from a
* {@link android.view.SurfaceView SurfaceView} or {@link android.view.SurfaceHolder SurfaceHolder}
* easily using {@link android.view.SurfaceHolder#getSurface SurfaceHolder.getSurface()} and/or
* {@link android.view.SurfaceView#getHolder SurfaceView.getHolder()}.</p>
*
* <p>To use a {@link android.view.TextureView Textureview} as a <code>SwapChain</code>, it is
* necessary to first get its {@link android.graphics.SurfaceTexture SurfaceTexture},
* for instance using {@link android.view.TextureView.SurfaceTextureListener SurfaceTextureListener}
* and then create a {@link android.view.Surface Surface}:</p>
*
* <pre>
* // using a TextureView.SurfaceTextureListener:
* public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int width, int height) {
* mSurface = new Surface(surfaceTexture);
* // mSurface can now be used with Engine.createSwapChain()
* }
* </pre>
*
* @see Engine
*/
public class SwapChain {
private final Object mSurface;
private long mNativeObject;
public static final long CONFIG_DEFAULT = 0x0;
/**
* This flag indicates that the <code>SwapChain</code> must be allocated with an
* alpha-channel.
*/
public static final long CONFIG_TRANSPARENT = 0x1;
/**
* This flag indicates that the <code>SwapChain</code> may be used as a source surface
* for reading back render results. This config must be set when creating
* any <code>SwapChain</code> that will be used as the source for a blit operation.
*
* @see Renderer#copyFrame
*/
public static final long CONFIG_READABLE = 0x2;
SwapChain(long nativeSwapChain, @NonNull Object surface) {
@@ -31,12 +90,15 @@ public class SwapChain {
mSurface = surface;
}
/**
* @return the native <code>Object</code> this <code>SwapChain</code> was created from.
*/
@NonNull
public Object getNativeWindow() {
return mSurface;
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed SwapChain");
}

View File

@@ -16,19 +16,60 @@
package com.google.android.filament;
import com.google.android.filament.proguard.UsedByReflection;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.Size;
import com.google.android.filament.proguard.UsedByReflection;
import java.nio.Buffer;
import java.nio.BufferOverflowException;
import java.nio.ByteBuffer;
import static com.google.android.filament.Texture.Type.COMPRESSED;
/**
* Texture
* <p>The <code>Texture</code> class supports:</p>
* <ul>
* <li>2D textures</li>
* <li>3D textures</li>
* <li>Cube maps</li>
* <li>mip mapping</li>
* </ul>
*
*
* <h1>Usage example</h1>
*
* A <code>Texture</code> object is created using the {@link Texture.Builder} and destroyed by
* calling {@link Engine#destroyTexture}. They're bound using {@link MaterialInstance#setParameter}.
*
* <pre>
* Engine engine = Engine.create();
*
* Material material = new Material.Builder()
* .payload( ... )
* .build(ending);
*
* MaterialInstance mi = material.getDefaultInstance();
*
* Texture texture = new Texture.Builder()
* .width(64)
* .height(64)
* .build(engine);
*
*
* texture.setImage(engine, 0,
* new Texture.PixelBufferDescriptor( ... ));
*
* mi.setParameter("parameterName", texture, new TextureSampler());
* </pre>
*
* @see #setImage
* @see PixelBufferDescriptor
* @see MaterialInstance#setParameter(String, Texture, TextureSampler)
*/
public class Texture {
private long mNativeObject;
@@ -37,12 +78,95 @@ public class Texture {
mNativeObject = nativeTexture;
}
/**
* Type of sampler
*/
public enum Sampler {
/** 2D sampler */
SAMPLER_2D,
/** Cubemap sampler */
SAMPLER_CUBEMAP,
/** External texture sampler */
SAMPLER_EXTERNAL
}
/**
* Internal texel formats
*
* <p>These formats are used to specify a texture's internal storage format.</p>
*
* <h1>Enumerants syntax format</h1>
*
* <code>[components][size][type]</code>
* <br><code>components</code> : List of stored components by this format
* <br><code>size</code> : Size in bit of each component
* <br><code>type</code> : Type this format is stored as
*
* <center>
* <table border="1">
* <tr><th> Name </th><th> Component </th></tr>
* <tr><td> R </td><td> Linear Red </td></tr>
* <tr><td> RG </td><td> Linear Red, Green </td></tr>
* <tr><td> RGB </td><td> Linear Red, Green, Blue </td></tr>
* <tr><td> RGBA </td><td> Linear Red, Green Blue, Alpha </td></tr>
* <tr><td> SRGB </td><td> sRGB encoded Red, Green, Blue </td></tr>
* <tr><td> DEPTH </td><td> Depth </td></tr>
* <tr><td> STENCIL </td><td> Stencil </td></tr>
* </table>
* </center>
* <br>
*
* <center>
* <table border="1">
* <tr><th> Name </th><th> Type </th></tr>
* <tr><td> (none) </td><td> Unsigned Normalized Integer [0, 1] </th></tr>
* <tr><td> _SNORM </td><td> Signed Normalized Integer [-1, 1] </td></tr>
* <tr><td> UI </td><td> Unsigned Integer [0, 2<sup>size</sup>] </td></tr>
* <tr><td> I </td><td> Signed Integer [-2<sup>size-1</sup>, 2<sup>size-1</sup>-1] </td></tr>
* <tr><td> F </td><td> Floating-point </td></tr>
* </table>
* </center>
* <br>
*
* <h1>Special color formats</h1>
*
* There are a few special color formats that don't follow the convention above:
*
* <center>
* <table border="1">
* <tr><th> Name </th><th> Format </th></tr>
* <tr><td> RGB565 </td><td> 5-bits for R and B, 6-bits for G. </td></tr>
* <tr><td> RGB5_A1 </td><td> 5-bits for R, G and B, 1-bit for A. </td></tr>
* <tr><td> RGB10_A2 </td><td> 10-bits for R, G and B, 2-bits for A. </td></tr>
* <tr><td> RGB9_E5 </td><td> <b>Unsigned</b> floating point. 9-bits mantissa for RGB, 5-bits shared exponent </td></tr>
* <tr><td> R11F_G11F_B10F </td><td> <b>Unsigned</b> floating point. 6-bits mantissa, for R and G, 5-bits for B. 5-bits exponent. </td></tr>
* <tr><td> SRGB8_A8 </td><td> sRGB 8-bits with linear 8-bits alpha. </td></tr>
* <tr><td> DEPTH24_STENCIL8 </td><td> 24-bits unsigned normalized integer depth, 8-bits stencil. </td></tr>
* </table>
* </center>
* <br>
*
* <h1>Compressed texture formats</h1>
*
* Many compressed texture formats are supported as well, which include (but are not limited to)
* the following list:
*
* <center>
* <table border="1">
* <tr><th> Name </th><th> Format </th></tr>
* <tr><td> EAC_R11 </td><td> Compresses R11UI </td></tr>
* <tr><td> EAC_R11_SIGNED </td><td> Compresses R11I </td></tr>
* <tr><td> EAC_RG11 </td><td> Compresses RG11UI </td></tr>
* <tr><td> EAC_RG11_SIGNED </td><td> Compresses RG11I </td></tr>
* <tr><td> ETC2_RGB8 </td><td> Compresses RGB8 </td></tr>
* <tr><td> ETC2_SRGB8 </td><td> compresses SRGB8 </td></tr>
* <tr><td> ETC2_EAC_RGBA8 </td><td> Compresses RGBA8 </td></tr>
* <tr><td> ETC2_EAC_SRGBA8 </td><td> Compresses SRGB8_A8 </td></tr>
* <tr><td> ETC2_RGB8_A1 </td><td> Compresses RGB8 with 1-bit alpha </td></tr>
* <tr><td> ETC2_SRGB8_A1 </td><td> Compresses sRGB8 with 1-bit alpha </td></tr>
* </table>
* </center>
*/
public enum InternalFormat {
// 8-bits per element
R8, R8_SNORM, R8UI, R8I, STENCIL8,
@@ -92,6 +216,10 @@ public class Texture {
DXT1_RGB, DXT1_RGBA, DXT3_RGBA, DXT5_RGBA
}
/**
* Compressed data types for use with {@link PixelBufferDescriptor}
* @see InternalFormat
*/
public enum CompressedFormat {
// Mandatory in GLES 3.0 and GL 4.3
EAC_R11, EAC_R11_SIGNED, EAC_RG11, EAC_RG11_SIGNED,
@@ -103,15 +231,27 @@ public class Texture {
DXT1_RGB, DXT1_RGBA, DXT3_RGBA, DXT5_RGBA
}
/**
* Cubemap faces
*/
public enum CubemapFace {
/** +x face */
POSITIVE_X,
/** -x face */
NEGATIVE_X,
/** +y face */
POSITIVE_Y,
/** -y face */
NEGATIVE_Y,
/** +z face */
POSITIVE_Z,
/** -z face */
NEGATIVE_Z
}
/**
* Pixel color format
*/
public enum Format {
R,
R_INTEGER,
@@ -128,19 +268,43 @@ public class Texture {
ALPHA
}
/**
* Pixel data type
*/
public enum Type {
/** unsigned byte, 8-bits */
UBYTE,
/** signed byte, 8-bits */
BYTE,
/** unsigned short, 16-bits */
USHORT,
/** signed short, 16-bits */
SHORT,
/** unsigned int, 32-bits */
UINT,
/** signed int, 32-bits */
INT,
/** half-float, 16-bits float with 10 bits mantissa */
HALF,
/** float, 32-bits float, with 24 bits mantissa */
FLOAT,
/** a compessed type */
COMPRESSED,
/** unsigned 5.6 (5.5 for blue) float packed in 32-bits */
UINT_10F_11F_11F_REV
}
/**
* A descriptor to an image in main memory, typically used to transfer image data from the CPU
* to the GPU.
* <p>A <code>PixelBufferDescriptor</code> owns the memory buffer it references,
* therefore <code>PixelBufferDescriptor</code> cannot be copied, but can be moved.</p>
*
* <code>PixelBufferDescriptor</code> releases ownership of the memory-buffer when it's
* destroyed.
*
* @see #setImage
*/
public static class PixelBufferDescriptor {
public Buffer storage;
@@ -165,6 +329,20 @@ public class Texture {
* - Android: Handler, Executor
* - Other: Executor
*/
/**
* Creates a <code>PixelBufferDescriptor</code>
*
* @param storage CPU-side buffer containing the image data to upload into the texture
* @param format Pixel {@link Format format} of the CPU-side image
* @param type Pixel data {@link Type type} of the CPU-side image
* @param alignment Row-alignment in bytes of the CPU-side image (1 to 8 bytes)
* @param left Left coordinate in pixels of the CPU-side image
* @param top Top coordinate in pixels of the CPU-side image
* @param stride Stride in pixels of the CPU-side image (i.e. distance in pixels to the next row)
* @param handler An {@link java.util.concurrent.Executor Executor}. On Android this can also be a {@link android.os.Handler Handler}.
* @param callback A callback executed by <code>handler</code> when <code>storage</code> is no longer needed.
*/
public PixelBufferDescriptor(@NonNull Buffer storage,
@NonNull Format format, @NonNull Type type,
@IntRange(from = 1, to = 8) int alignment,
@@ -182,17 +360,48 @@ public class Texture {
this.callback = callback;
}
/**
* Creates a <code>PixelBufferDescriptor</code> with some default values and no callback.
*
* @param storage CPU-side buffer containing the image data to upload into the texture
* @param format Pixel {@link Format format} of the CPU-side image
* @param type Pixel data {@link Type type} of the CPU-side image
*
* @see #setCallback
*/
public PixelBufferDescriptor(@NonNull Buffer storage,
@NonNull Format format, @NonNull Type type) {
this(storage, format, type, 1, 0, 0, 0, null, null);
}
/**
* Creates a <code>PixelBufferDescriptor</code> with some default values and no callback.
*
* @param storage CPU-side buffer containing the image data to upload into the texture
* @param format Pixel {@link Format format} of the CPU-side image
* @param type Pixel data {@link Type type} of the CPU-side image
* @param alignment Row-alignment in bytes of the CPU-side image (1 to 8 bytes)
*
* @see #setCallback
*/
public PixelBufferDescriptor(@NonNull Buffer storage,
@NonNull Format format, @NonNull Type type,
@IntRange(from = 1, to = 8) int alignment) {
this(storage, format, type, alignment, 0, 0, 0, null, null);
}
/**
* Creates a <code>PixelBufferDescriptor</code> with some default values and no callback.
*
* @param storage CPU-side buffer containing the image data to upload into the texture
* @param format Pixel {@link Format format} of the CPU-side image
* @param type Pixel data {@link Type type} of the CPU-side image
* @param alignment Row-alignment in bytes of the CPU-side image (1 to 8 bytes)
* @param left Left coordinate in pixels of the CPU-side image
* @param top Top coordinate in pixels of the CPU-side image
*
* @see #setCallback
*/
public PixelBufferDescriptor(@NonNull Buffer storage,
@NonNull Format format, @NonNull Type type,
@IntRange(from = 1, to = 8) int alignment,
@@ -200,6 +409,14 @@ public class Texture {
this(storage, format, type, alignment, left, top, 0, null, null);
}
/**
*
* @param storage CPU-side buffer containing the image data to upload into the texture
* @param format Compressed pixel {@link CompressedFormat format} of the CPU-side image
* @param compressedSizeInBytes Size of the compressed data in bytes
*
* @see #setCallback
*/
public PixelBufferDescriptor(@NonNull ByteBuffer storage,
@NonNull CompressedFormat format,
@IntRange(from = 0) int compressedSizeInBytes) {
@@ -211,9 +428,10 @@ public class Texture {
}
/**
* Valid handler types:
* - Android: Handler, Executor
* - Other: Executor
* Set or replace the callback called when the CPU-side data is no longer needed.
*
* @param handler An {@link java.util.concurrent.Executor Executor}. On Android this can also be a {@link android.os.Handler Handler}.
* @param callback A callback executed by <code>handler</code> when <code>storage</code> is no longer needed.
*/
public void setCallback(@Nullable Object handler, @Nullable Runnable callback) {
this.handler = handler;
@@ -288,57 +506,109 @@ public class Texture {
}
}
/**
* Options of {@link #generatePrefilterMipmap}
*/
public static class PrefilterOptions {
/** number of samples for roughness pre-filtering */
public int sampleCount = 8;
/** whether to generate a reflection map (mirror) */
public boolean mirror = true;
}
/**
* Checks whether a given format is supported for texturing in this {@link Engine}.
* This depends on the selected backend.
*
* @param engine {@link Engine} to test the {@link InternalFormat InternalFormat} against
* @param format format to check
* @return <code>true</code> if this format is supported for texturing.
*/
public static boolean isTextureFormatSupported(@NonNull Engine engine,
@NonNull InternalFormat format) {
return nIsTextureFormatSupported(engine.getNativeObject(), format.ordinal());
}
/**
* Use <code>Builder</code> to construct a <code>Texture</code> object instance.
*/
public static class Builder {
@SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"})
// Keep to finalize native resources
private final BuilderFinalizer mFinalizer;
private final long mNativeBuilder;
/**
* Use <code>Builder</code> to construct a <code>Texture</code> object instance.
*/
public Builder() {
mNativeBuilder = nCreateBuilder();
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Specifies the width of the texture in texels.
* @param width texture width in texels, must be at least 1. Default is 1.
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder width(@IntRange(from = 1) int width) {
nBuilderWidth(mNativeBuilder, width);
return this;
}
/**
* Specifies the height of the texture in texels.
* @param height texture height in texels, must be at least 1. Default is 1.
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder height(@IntRange(from = 1) int height) {
nBuilderHeight(mNativeBuilder, height);
return this;
}
/**
* Specifies the texture's number of layers. This creates a 3D texture.
* @param depth texture number of layer, must be at least 1. Default is 1.
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder depth(@IntRange(from = 1) int depth) {
nBuilderDepth(mNativeBuilder, depth);
return this;
}
/**
* Specifies the number of mipmap levels
* @param levels must be at least 1 and less or equal to <code>floor(log<sub>2</sub>(max(width, height))) + 1</code>. Default is 1.
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder levels(@IntRange(from = 1) int levels) {
nBuilderLevels(mNativeBuilder, levels);
return this;
}
/**
* Specifies the type of sampler to use.
* @param target {@link Sampler Sampler} type
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder sampler(@NonNull Sampler target) {
nBuilderSampler(mNativeBuilder, target.ordinal());
return this;
}
/**
* Specifies the texture's internal format.
* <p>The internal format specifies how texels are stored (which may be different from how
* they're specified in {@link #setImage}). {@link InternalFormat InternalFormat} specifies
* both the color components and the data type used.</p>
* @param format texture's {@link InternalFormat internal format}.
* @return This Builder, for chaining calls.
*/
@NonNull
public Builder format(@NonNull InternalFormat format) {
nBuilderFormat(mNativeBuilder, format.ordinal());
@@ -356,6 +626,13 @@ public class Texture {
return this;
}
/**
* Creates a new <code>Texture</code> instance.
* @param engine The {@link Engine} to associate this <code>Texture</code> with.
* @return A newly created <code>Texture</code>
* @exception IllegalStateException if a parameter to a builder function was invalid.
* A mode detailed message about the error is output in the system log.
*/
@NonNull
public Texture build(@NonNull Engine engine) {
long nativeTexture = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -382,38 +659,71 @@ public class Texture {
}
}
/**
* A bitmask to specify how the texture will be used.
*/
public static class Usage {
/** The texture will be used as a color attachment */
public static final int COLOR_ATTACHMENT = 0x1;
/** The texture will be used as a depth attachment */
public static final int DEPTH_ATTACHMENT = 0x2;
/** The texture will be used as a stencil attachment */
public static final int STENCIL_ATTACHMENT = 0x4;
/** The texture content can be set with {@link #setImage} */
public static final int UPLOADABLE = 0x8;
/** The texture can be read from a shader or blitted from */
public static final int SAMPLEABLE = 0x10;
/** by default textures are <code>UPLOADABLE</code> and <code>SAMPLEABLE</code>*/
public static final int DEFAULT = UPLOADABLE | SAMPLEABLE;
}
public static final int BASE_LEVEL = 0;
/**
* Queries the width of a given level of this texture.
* @param level to query the with of. Must be between 0 and {@link #getLevels}
* @return The width in texel of the given level
*/
public int getWidth(@IntRange(from = 0) int level) {
return nGetWidth(getNativeObject(), level);
}
/**
* Queries the height of a given level of this texture.
* @param level to query the height of. Must be between 0 and {@link #getLevels}
* @return The height in texel of the given level
*/
public int getHeight(@IntRange(from = 0) int level) {
return nGetHeight(getNativeObject(), level);
}
/**
* Queries the number of layers of given level of this texture has.
* @param level to query the number of layers of. Must be between 0 and {@link #getLevels}
* @return The number of layers of the given level
*/
public int getDepth(@IntRange(from = 0) int level) {
return nGetDepth(getNativeObject(), level);
}
/**
* @return the number of mipmap levels of this texture
*/
public int getLevels() {
return nGetLevels(getNativeObject());
}
/**
* @return This texture {@link Sampler Sampler} type.
*/
@NonNull
public Sampler getTarget() {
return Sampler.values()[nGetTarget(getNativeObject())];
}
/**
* @return This texture's {@link InternalFormat InternalFormat}.
*/
@NonNull
public InternalFormat getFormat() {
return InternalFormat.values()[nGetInternalFormat(getNativeObject())];
@@ -421,12 +731,65 @@ public class Texture {
// TODO: add a setImage() version that takes an android Bitmap
/**
* <code>setImage</code> is used to modify the whole content of the texure from a CPU-buffer.
*
* <p>This <code>Texture</code> instance must use {@link Sampler#SAMPLER_2D SAMPLER_2D} or
* {@link Sampler#SAMPLER_EXTERNAL SAMPLER_EXTERNAL}. If the later is specified
* and external textures are supported by the driver implementation,
* this method will have no effect, otherwise it will behave as if the
* texture was specified with {@link Sampler#SAMPLER_2D SAMPLER_2D}.</p>
*
* This is equivalent to calling: <br>
*
* <code>setImage(engine, level, 0, 0, getWidth(level), getHeight(level), buffer)</code>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
* @param level Level to set the image for. Must be less than {@link #getLevels()}.
* @param buffer Client-side buffer containing the image to set.
* <code>buffer</code>'s {@link Format format} must match that
* of {@link #getFormat()}
*
* @exception BufferOverflowException if the specified parameters would result in reading
* outside of <code>buffer</code>.
*
* @see Builder#sampler
* @see PixelBufferDescriptor
*/
public void setImage(@NonNull Engine engine,
@IntRange(from = 0) int level,
@NonNull PixelBufferDescriptor buffer) {
setImage(engine, level, 0, 0, getWidth(level), getHeight(level), buffer);
}
/**
* <code>setImage</code> is used to modify a sub-region of the texure from a CPU-buffer.
*
* <p>This <code>Texture</code> instance must use {@link Sampler#SAMPLER_2D SAMPLER_2D} or
* {@link Sampler#SAMPLER_EXTERNAL SAMPLER_EXTERNAL}. If the later is specified
* and external textures are supported by the driver implementation,
* this method will have no effect, otherwise it will behave as if the
* texture was specified with {@link Sampler#SAMPLER_2D SAMPLER_2D}.</p>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
* @param level Level to set the image for. Must be less than {@link #getLevels()}.
* @param xoffset x-offset in texel of the region to modify
* @param yoffset y-offset in texel of the region to modify
* @param width width in texel of the region to modify
* @param height height in texel of the region to modify
* @param buffer Client-side buffer containing the image to set.
* <code>buffer</code>'s {@link Format format} must match that
* of {@link #getFormat()}
*
* @exception BufferOverflowException if the specified parameters would result in reading
* outside of <code>buffer</code>.
*
* @see Builder#sampler
* @see PixelBufferDescriptor
*/
public void setImage(@NonNull Engine engine,
@IntRange(from = 0) int level,
@IntRange(from = 0) int xoffset, @IntRange(from = 0) int yoffset,
@@ -453,9 +816,33 @@ public class Texture {
}
}
// note: faceOffsetsInBytes are offsets in byte in the buffer relative to the current position()
// note: use Texture CubemapFace to index the faceOffsetsInBytes array
// note: we assume all 6 faces are tightly packed
/**
* <code>setImage</code> is used to specify all six images of a cubemap level and
* follows exactly the OpenGL conventions
*
* <p>This <code>Texture</code> instance must use
* {@link Sampler#SAMPLER_CUBEMAP SAMPLER_CUBEMAP}.</p>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
* @param level Level to set the image for. Must be less than {@link #getLevels()}.
* @param buffer Client-side buffer containing the image to set.
* <code>buffer</code>'s {@link Format format} must match that
* of {@link #getFormat()}
* @param faceOffsetsInBytes Offsets in bytes into <code>buffer</code> for all six images.
* The offsets are specified in the following order:
* +x, -x, +y, -y, +z, -z.
*
* <p><code>faceOffsetsInBytes</code> are offsets in byte in the <code>buffer</code> relative
* to the current {@link Buffer#position()}. Use {@link CubemapFace} to index the
* <code>faceOffsetsInBytes</code> array. All six faces must be tightly packed.</p>
*
* @exception BufferOverflowException if the specified parameters would result in reading
* outside of <code>buffer</code>.
*
* @see Builder#sampler
* @see PixelBufferDescriptor
*/
public void setImage(@NonNull Engine engine, @IntRange(from = 0) int level,
@NonNull PixelBufferDescriptor buffer,
@NonNull @Size(min = 6) int[] faceOffsetsInBytes) {
@@ -478,10 +865,60 @@ public class Texture {
}
}
/**
* Specifies the external image to associate with this <code>Texture</code>.
*
* <p>This <code>Texture</code> instance must use
* {@link Sampler#SAMPLER_EXTERNAL SAMPLER_EXTERNAL}.</p>
* <p>Typically the external image is OS specific, and can be a video or camera frame.
* There are many restrictions when using an external image as a texture, such as:</p>
* <ul>
* <li> only the level of detail (LOD) 0 can be specified</li>
* <li> only {@link TextureSampler.MagFilter#NEAREST NEAREST} or
* {@link TextureSampler.MagFilter#LINEAR LINEAR} filtering is supported</li>
* <li> the size and format of the texture is defined by the external image</li>
* </ul>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
* @param eglImage An opaque handle to a platform specific image. Supported types are
* <code>eglImageOES</code> on Android and <code>CVPixelBufferRef</code> on iOS.
* <p>On iOS the following pixel formats are supported: <ul>
* <li><code>kCVPixelFormatType_32BGRA</code></li>
* <li><code>kCVPixelFormatType_420YpCbCr8BiPlanarFullRange</code></li>
* </ul></p>
*
* @see Builder#sampler
*/
public void setExternalImage(@NonNull Engine engine, long eglImage) {
nSetExternalImage(getNativeObject(), engine.getNativeObject(), eglImage);
}
/**
* Specifies the external stream to associate with this <code>Texture</code>.
*
* <p>This <code>Texture</code> instance must use
* {@link Sampler#SAMPLER_EXTERNAL SAMPLER_EXTERNAL}.</p>
* <p>Typically the external image is OS specific, and can be a video or camera frame.
* There are many restrictions when using an external image as a texture, such as:</p>
* <ul>
* <li> only the level of detail (LOD) 0 can be specified</li>
* <li> only {@link TextureSampler.MagFilter#NEAREST NEAREST} or
* {@link TextureSampler.MagFilter#LINEAR LINEAR} filtering is supported</li>
* <li> the size and format of the texture is defined by the external image</li>
* </ul>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
* @param stream A {@link Stream} object
*
* @exception IllegalStateException if the sampler type is not
* {@link Sampler#SAMPLER_EXTERNAL SAMPLER_EXTERNAL}
*
* @see Stream
* @see Builder#sampler
*
*/
public void setExternalStream(@NonNull Engine engine, @NonNull Stream stream) {
long nativeObject = getNativeObject();
long streamNativeObject = stream.getNativeObject();
@@ -492,10 +929,68 @@ public class Texture {
nSetExternalStream(nativeObject, engine.getNativeObject(), streamNativeObject);
}
/**
* Generates all the mipmap levels automatically. This requires the texture to have a
* color-renderable format.
*
* <p>This <code>Texture</code> instance must <b>not</b> use
* {@link Sampler#SAMPLER_CUBEMAP SAMPLER_CUBEMAP}, or it has no effect.</p>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
*/
public void generateMipmaps(@NonNull Engine engine) {
nGenerateMipmaps(getNativeObject(), engine.getNativeObject());
}
/**
* Creates a reflection map from an environment map.
*
* <p>This is a utility function that replaces calls to {@link #setImage}.
* The provided environment map is processed and all mipmap levels are populated. The
* processing is similar to the offline tool <code>cmgen</code> at a lower quality setting.</p>
*
* <p>This function is intended to be used when the environment cannot be processed offline,
* for instance if it's generated at runtime.</p>
*
* <p>The source data must obey to some constraints:</p>
* <ul>
* <li>the data {@link Format format} must be {@link Format#RGB}</li>
* <li>the data {@link Type type} must be one of
* <ul>
* <li>{@link Type#FLOAT}</li>
* <li>{@link Type#HALF}</li>
* </ul>
* </li>
* </ul>
*
* <p>The current texture must be a cubemap.</p>
*
* <p>The reflections cubemap's {@link InternalFormat internal format} cannot be a compressed format.</p>
*
* <p>The reflections cubemap's dimension must be a power-of-two.</p>
*
* <p>This operation is computationally intensive, especially with large environments and
* is currently <b>synchronous</b>. Expect about 1ms for a 16 &times 16 cubemap.</p>
*
* @param engine {@link Engine} this texture is associated to. Must be the
* instance passed to {@link Builder#build Builder.build()}.
* @param buffer Client-side buffer containing the image to set.
* <code>buffer</code>'s {@link Format format} and {@link Type type} must match
* the constraints above.
* @param faceOffsetsInBytes Offsets in bytes into <code>buffer</code> for all six images.
* The offsets are specified in the following order:
* +x, -x, +y, -y, +z, -z.
*
* @param options Optional parameter to control user-specified quality and options.
*
* <p><code>faceOffsetsInBytes</code> are offsets in byte in the <code>buffer</code> relative
* to the current {@link Buffer#position()}. Use {@link CubemapFace} to index the
* <code>faceOffsetsInBytes</code> array. All six faces must be tightly packed.</p>
*
* @exception BufferOverflowException if the specified parameters would result in reading
* outside of <code>buffer</code>.
*/
public void generatePrefilterMipmap(@NonNull Engine engine,
@NonNull PixelBufferDescriptor buffer, @NonNull @Size(min = 6) int[] faceOffsetsInBytes,
PrefilterOptions options) {
@@ -524,7 +1019,7 @@ public class Texture {
@UsedByReflection("TextureHelper.java")
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed Texture");
}

View File

@@ -18,24 +18,60 @@ package com.google.android.filament;
import android.support.annotation.NonNull;
/**
* <code>TextureSampler</code> defines how a texture is accessed.
*/
public class TextureSampler {
public enum WrapMode {
/**
* The edge of the texture extends to infinity.
*/
CLAMP_TO_EDGE,
/**
* The texture infinitely repeats in the wrap direction.
*/
REPEAT,
/**
* The texture infinitely repeats and mirrors in the wrap direction.
*/
MIRRORED_REPEAT
}
public enum MinFilter {
/**
* No filtering. Nearest neighbor is used.
*/
NEAREST,
/**
* Box filtering. Weighted average of 4 neighbors is used.
*/
LINEAR,
/**
* Mip-mapping is activated. But no filtering occurs.
*/
NEAREST_MIPMAP_NEAREST,
/**
* Box filtering within a mip-map level.
*/
LINEAR_MIPMAP_NEAREST,
/**
* Mip-map levels are interpolated, but no other filtering occurs.
*/
NEAREST_MIPMAP_LINEAR,
/**
* Both interpolated Mip-mapping and linear filtering are used.
*/
LINEAR_MIPMAP_LINEAR,
}
public enum MagFilter {
/**
* No filtering. Nearest neighbor is used.
*/
NEAREST,
/**
* Box filtering. Weighted average of 4 neighbors is used.
*/
LINEAR
}
@@ -44,114 +80,246 @@ public class TextureSampler {
COMPARE_TO_TEXTURE
}
/**
* Comparison functions for the depth sampler.
*/
public enum CompareFunction {
/**
* Less or equal
*/
LESS_EQUAL,
/**
* Greater or equal
*/
GREATER_EQUAL,
/**
* Strictly less than
*/
LESS,
/**
* Strictly greater than
*/
GREATER,
/**
* Equal
*/
EQUAL,
/**
* Not equal
*/
NOT_EQUAL,
/**
* Always. Depth testing is deactivated.
*/
ALWAYS,
/**
* Never. The depth test always fails.
*/
NEVER
}
int mSampler = 0; // bit field used by native
/**
* Min filter: LINEAR_MIPMAP_LINEAR
* Mag filter: LINEAR
* Wrap mode: REPEAT
* Initializes the <code>TextureSampler</code> with default values.
* <br>Minification filter: {@link MinFilter#LINEAR_MIPMAP_LINEAR}
* <br>Magnification filter: {@link MagFilter#LINEAR}
* <br>Wrap modes: {@link WrapMode#REPEAT}
*/
public TextureSampler() {
this(MinFilter.LINEAR_MIPMAP_LINEAR, MagFilter.LINEAR, WrapMode.REPEAT);
}
/**
* Initializes the <code>TextureSampler</code> with default values, but specifying the
* minification and magnification filters.
*
* @param minMag {@link MagFilter magnification filter},
* the minification filter will be set to the same value.
*/
public TextureSampler(@NonNull MagFilter minMag) {
this(minMag, WrapMode.CLAMP_TO_EDGE);
}
/**
* Initializes the <code>TextureSampler</code> with user specified values.
*
* @param minMag {@link MagFilter magnification filter},
* the minification filter will be set to the same value.
* @param wrap {@link WrapMode wrapping mode} for all directions
*/
public TextureSampler(@NonNull MagFilter minMag, @NonNull WrapMode wrap) {
this(minFilterFromMagFilter(minMag), minMag, wrap);
}
/**
* Initializes the <code>TextureSampler</code> with user specified values.
*
* @param min {@link MagFilter magnification filter}
* @param mag {@link MinFilter minification filter}
* @param wrap {@link WrapMode wrapping mode} for all directions
*/
public TextureSampler(@NonNull MinFilter min, @NonNull MagFilter mag, @NonNull WrapMode wrap) {
this(min, mag, wrap, wrap, wrap);
}
/**
* Initializes the <code>TextureSampler</code> with user specified values.
*
* @param min {@link MagFilter magnification filter}
* @param mag {@link MinFilter minification filter}
* @param s {@link WrapMode wrapping mode} for the s (horizontal) direction
* @param t {@link WrapMode wrapping mode} for the t (vertical) direction
* @param r {@link WrapMode wrapping mode} fot the r (depth) direction
*/
public TextureSampler(@NonNull MinFilter min, @NonNull MagFilter mag,
@NonNull WrapMode s, @NonNull WrapMode t, @NonNull WrapMode r) {
mSampler = nCreateSampler(min.ordinal(), mag.ordinal(),
s.ordinal(), t.ordinal(), r.ordinal());
}
/**
* Initializes the <code>TextureSampler</code> with user specified comparison mode. The
* comparison fonction is set to {@link CompareFunction#LESS_EQUAL}.
*
* @param mode {@link CompareMode comparison mode}
*/
public TextureSampler(@NonNull CompareMode mode) {
this(mode, CompareFunction.LESS_EQUAL);
}
/**
* Initializes the <code>TextureSampler</code> with user specified comparison mode and function.
*
* @param mode {@link CompareMode comparison mode}
* @param function {@link CompareFunction comparison function}
*/
public TextureSampler(@NonNull CompareMode mode, @NonNull CompareFunction function) {
mSampler = nCreateCompareSampler(mode.ordinal(), function.ordinal());
}
/**
* @return the minification filter
*/
public MinFilter getMinFilter() {
return MinFilter.values()[nGetMinFilter(mSampler)];
}
/**
* Sets the minification filter.
*
* @param filter minification filter
*/
public void setMinFilter(MinFilter filter) {
mSampler = nSetMinFilter(mSampler, filter.ordinal());
}
/**
* @return the magnification filter
*/
public MagFilter getMagFilter() {
return MagFilter.values()[nGetMagFilter(mSampler)];
}
/**
* Sets the magnification filter.
*
* @param filter magnification filter
*/
public void setMagFilter(MagFilter filter) {
mSampler = nSetMagFilter(mSampler, filter.ordinal());
}
/**
* @return the wrapping mode in the s (horizontal) direction
*/
public WrapMode getWrapModeS() {
return WrapMode.values()[nGetWrapModeS(mSampler)];
}
/**
* Sets the wrapping mode in the s (horizontal) direction.
* @param mode wrapping mode
*/
public void setWrapModeS(WrapMode mode) {
mSampler = nSetWrapModeS(mSampler, mode.ordinal());
}
/**
* @return the wrapping mode in the t (vertical) direction
*/
public WrapMode getWrapModeT() {
return WrapMode.values()[nGetWrapModeT(mSampler)];
}
/**
* Sets the wrapping mode in the t (vertical) direction.
* @param mode wrapping mode
*/
public void setWrapModeT(WrapMode mode) {
mSampler = nSetWrapModeT(mSampler, mode.ordinal());
}
/**
* @return the wrapping mode in the r (depth) direction
*/
public WrapMode getWrapModeR() {
return WrapMode.values()[nGetWrapModeR(mSampler)];
}
/**
* Sets the wrapping mode in the t (depth) direction.
* @param mode wrapping mode
*/
public void setWrapModeR(WrapMode mode) {
mSampler = nSetWrapModeR(mSampler, mode.ordinal());
}
/**
* @return the anisotropy value
* @see #setAnisotropy
*/
public float getAnisotropy() {
return nGetAnisotropy(mSampler);
}
/**
* This controls anisotropic filtering.
*
* @param anisotropy Amount of anisotropy, should be a power-of-two. The default is 0.
* The maximum permissible value is 7.
*/
public void setAnisotropy(float anisotropy) {
mSampler = nSetAnisotropy(mSampler, anisotropy);
}
/**
* @return the comparison mode
*/
public CompareMode getCompareMode() {
return CompareMode.values()[nGetCompareMode(mSampler)];
}
/**
* Sets the comparison mode.
*
* @param mode comparison mode
*/
public void setCompareMode(CompareMode mode) {
mSampler = nSetCompareMode(mSampler, mode.ordinal());
}
/**
* @return the comparison function
*/
public CompareFunction getCompareFunction() {
return CompareFunction.values()[nGetCompareFunction(mSampler)];
}
/**
* Sets the comparison function.
* @param function the comparison function
*/
public void setCompareFunction(CompareFunction function) {
mSampler = nSetCompareFunction(mSampler, function.ordinal());
}

View File

@@ -20,47 +20,154 @@ import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.Size;
public class TransformManager {
/**
* <code>TransformManager</code> is used to add transform components to entities.
*
* <p>A transform component gives an entity a position and orientation in space in the coordinate
* space of its parent transform. The <code>TransformManager</code> takes care of computing the
* world-space transform of each component (i.e. its transform relative to the root).</p>
*
* <h1>Creation and destruction</h1>
*
* A transform component is created using {@link TransformManager#create} and destroyed by calling
* {@link TransformManager#destroy}.
*
* <pre>
* Engine engine = Engine.create();
* EntityManager entityManager = EntityManager().get();
* int object = entityManager.create();
*
* TransformManager tcm = engine.getTransformManager();
*
* // create the transform component
* tcm.create(object);
*
* // set its transform
* float[] transform = ...; // transform to set
* EntityInstance i = tcm.getInstance(object);
* tcm.setTransform(i, transform));
*
* // destroy the transform component
* tcm.destroy(object);
* </pre>
*
*/public class TransformManager {
private long mNativeObject;
TransformManager(long nativeTransformManager) {
mNativeObject = nativeTransformManager;
}
/**
* Returns whether a particular {@link Entity} is associated with a component of this
* <code>TransformManager</code>
*
* @param entity an {@link Entity}
* @return true if this {@link Entity} has a component associated with this manager
*/
public boolean hasComponent(@Entity int entity) {
return nHasComponent(mNativeObject, entity);
}
/**
* Gets an {@link EntityInstance} representing the transform component associated with the
* given {@link Entity}.
*
* @param entity an {@link Entity}
* @return an {@link EntityInstance}, which represents the transform component associated with
* the {@link Entity} <code>entity</code>
* @see #hasComponent
*/
@EntityInstance
public int getInstance(@Entity int entity) {
return nGetInstance(mNativeObject, entity);
}
/**
* Creates a transform component and associates it with the given entity. The component is
* initialized with the identity transform.
* If this component already exists on the given entity, it is first
* destroyed as if {@link #destroy} was called.
*
* @param entity an {@link Entity} to associate a transform component to.
* @see #destroy
*/
@EntityInstance
public int create(@Entity int entity) {
return nCreate(mNativeObject, entity);
}
/**
* Creates a transform component with a parent and associates it with the given entity.
* If this component already exists on the given entity, it is first
* destroyed as if {@link #destroy} was called.
*
* @param entity an {@link Entity} to associate a transform component to.
* @param parent the {@link EntityInstance} of the parent transform
* @param localTransform the transform, relative to the parent, to initialize the transform
* component with.
* @see #destroy
*/
@EntityInstance
public int create(@Entity int entity, @EntityInstance int parent,
@Nullable @Size(min = 16) float[] localTransform) {
return nCreateArray(mNativeObject, entity, parent, localTransform);
}
/**
* Destroys this component from the given entity, children are orphaned.
*
* @param entity an {@link Entity}.
* If this transform had children, these are orphaned, which means their local
* transform becomes a world transform. Usually it's nonsensical.
* It's recommended to make sure that a destroyed transform doesn't have children.
* @see #create
*/
public void destroy(@Entity int entity) {
nDestroy(mNativeObject, entity);
}
/**
* Re-parents an entity to a new one.
*
* @param i the {@link EntityInstance} of the transform component to re-parent
* @param newParent the {@link EntityInstance} of the new parent transform.
* It is an error to re-parent an entity to a descendant and will cause
* undefined behaviour.
* @see #getInstance
*/
public void setParent(@EntityInstance int i, @EntityInstance int newParent) {
nSetParent(mNativeObject, i, newParent);
}
/**
* Sets a local transform of a transform component.
* <p>This operation can be slow if the hierarchy of transform is too deep, and this
* will be particularly bad when updating a lot of transforms. In that case,
* consider using {@link #openLocalTransformTransaction} / {@link #commitLocalTransformTransaction}.</p>
*
* @param i the {@link EntityInstance} of the transform component to set the local
* transform to.
* @param localTransform the local transform (i.e. relative to the parent).
* @see #getTransform
*/
public void setTransform(@EntityInstance int i,
@NonNull @Size(min = 16) float[] localTransform) {
Asserts.assertMat4fIn(localTransform);
nSetTransform(mNativeObject, i, localTransform);
}
/**
* Returns the local transform of a transform component.
*
* @param i the {@link EntityInstance} of the transform component to query the
* local transform from.
* @param outLocalTransform a 16 <code>float</code> array to receive the result.
* If <code>null</code> is given, a new suitable array is allocated.
* @return the local transform of the component (i.e. relative to the parent). This always
* returns the value set by setTransform().
* @see #setTransform
*/
@NonNull
@Size(min = 16)
public float[] getTransform(@EntityInstance int i,
@@ -70,6 +177,17 @@ public class TransformManager {
return outLocalTransform;
}
/**
* Returns the world transform of a transform component.
*
* @param i the {@link EntityInstance} of the transform component to query the
* world transform from.
* @param outWorldTransform a 16 <code>float</code> array to receive the result.
* If <code>null</code> is given, a new suitable array is allocated
* @return The world transform of the component (i.e. relative to the root). This is the
* composition of this component's local transform with its parent's world transform.
* @see #setTransform
*/
@NonNull
@Size(min = 16)
public float[] getWorldTransform(@EntityInstance int i,
@@ -79,14 +197,44 @@ public class TransformManager {
return outWorldTransform;
}
/**
* Opens a local transform transaction. During a transaction, {@link #getWorldTransform} can
* return an invalid transform until {@link #commitLocalTransformTransaction} is called.
* However, {@link #setTransform} will perform significantly better and in constant time.
*
* <p>This is useful when updating many transforms and the transform hierarchy is deep (say more
* than 4 or 5 levels).</p>
*
* <p>If the local transform transaction is already open, this is a no-op.</p>
*
* @see #commitLocalTransformTransaction
* @see #setTransform
*/
public void openLocalTransformTransaction() {
nOpenLocalTransformTransaction(mNativeObject);
}
/**
* Commits the currently open local transform transaction. When this returns, calls
* to {@link #getWorldTransform} will return the proper value.
*
* <p>Failing to call this method when done updating the local transform will cause
* a lot of rendering problems. The system never closes the transaction
* automatically.</p>
*
* <p>If the local transform transaction is not open, this is a no-op.</p>
*
* @see #openLocalTransformTransaction
* @see #setTransform
*/
public void commitLocalTransformTransaction() {
nCommitLocalTransformTransaction(mNativeObject);
}
public long getNativeObject() {
return mNativeObject;
}
private static native boolean nHasComponent(long nativeTransformManager, int entity);
private static native int nGetInstance(long nativeTransformManager, int entity);
private static native int nCreate(long nativeTransformManager, int entity);

View File

@@ -23,6 +23,33 @@ import android.support.annotation.Nullable;
import java.nio.Buffer;
import java.nio.BufferOverflowException;
/**
* Holds a set of buffers that define the geometry of a <code>Renderable</code>.
*
* <p>
* The geometry of the <code>Renderable</code> itself is defined by a set of vertex attributes such as
* position, color, normals, tangents, etc...
* </p>
*
* <p>
* There is no need to have a 1-to-1 mapping between attributes and buffer. A buffer can hold the
* data of several attributes -- attributes are then referred as being "interleaved".
* </p>
*
* <p>
* The buffers themselves are GPU resources, therefore mutating their data can be relatively slow.
* For this reason, it is best to separate the constant data from the dynamic data into multiple
* buffers.
* </p>
*
* <p>
* It is possible, and even encouraged, to use a single vertex buffer for several
* <code>Renderable</code>s.
* </p>
*
* @see IndexBuffer
* @see RenderableManager
*/
public class VertexBuffer {
private long mNativeObject;
@@ -31,13 +58,22 @@ public class VertexBuffer {
}
public enum VertexAttribute {
POSITION, // XYZ position (float3)
TANGENTS, // tangent, bitangent and normal, encoded as a quaternion (4 floats or half floats)
COLOR, // vertex color (float4)
UV0, // texture coordinates (float2)
UV1, // texture coordinates (float2)
POSITION, // XYZ position (float3)
TANGENTS, // tangent, bitangent and normal, encoded as a quaternion (4 floats or half floats)
COLOR, // vertex color (float4)
UV0, // texture coordinates (float2)
UV1, // texture coordinates (float2)
BONE_INDICES, // indices of 4 bones (uvec4)
BONE_WEIGHTS // weights of the 4 bones (normalized float4)
BONE_WEIGHTS, // weights of the 4 bones (normalized float4)
UNUSED, // reserved for future use
CUSTOM0, // custom or MORPH_POSITION_0
CUSTOM1, // custom or MORPH_POSITION_1
CUSTOM2, // custom or MORPH_POSITION_2
CUSTOM3, // custom or MORPH_POSITION_3
CUSTOM4, // custom or MORPH_TANGENTS_0
CUSTOM5, // custom or MORPH_TANGENTS_1
CUSTOM6, // custom or MORPH_TANGENTS_2
CUSTOM7 // custom or MORPH_TANGENTS_3
}
public enum AttributeType {
@@ -68,22 +104,47 @@ public class VertexBuffer {
HALF3,
HALF4,
}
/**
* Specifies the quaternion type for the {@link #populateTangentQuaternions} utility.
*/
public enum QuatType {
HALF4, // 2 bytes per component as half-floats (8 bytes per quat)
SHORT4, // 2 bytes per component as normalized integers (8 bytes per quat)
FLOAT4, // 4 bytes per component as floats (16 bytes per quat)
/** 2 bytes per component as half-floats (8 bytes per quat) */
HALF4,
/** 2 bytes per component as normalized integers (8 bytes per quat) */
SHORT4,
/** 4 bytes per component as floats (16 bytes per quat) */
FLOAT4,
}
/**
* Specifies the parameters for the {@link #populateTangentQuaternions} utility.
*/
public static class QuatTangentContext {
public QuatType quatType; // desired quaternion type (required)
public int quatCount; // number of quaternions (required)
public Buffer outBuffer; // pre-allocated output buffer (required)
public int outStride; // desired stride in bytes (optional)
public Buffer normals; // source normals (required)
public int normalsStride; // normals stride in bytes (optional)
public Buffer tangents; // source tangents (optional)
public int tangentsStride; // tangents stride in bytes (optional)
/** desired quaternion type (required) */
public QuatType quatType;
/** number of quaternions (required) */
public int quatCount;
/** pre-allocated output buffer (required) */
public Buffer outBuffer;
/** desired stride in bytes (optional) */
public int outStride;
/** source normals (required) */
public Buffer normals;
/** normals stride in bytes (optional) */
public int normalsStride;
/** source tangents (optional) */
public Buffer tangents;
/** tangents stride in bytes (optional) */
public int tangentsStride;
}
public static class Builder {
@@ -96,18 +157,64 @@ public class VertexBuffer {
mFinalizer = new BuilderFinalizer(mNativeBuilder);
}
/**
* Size of each buffer in this set, expressed in in number of vertices.
*
* @param vertexCount number of vertices in each buffer in this set
*
* @return A reference to this Builder for chaining calls.
*/
@NonNull
public Builder vertexCount(@IntRange(from = 1) int vertexCount) {
nBuilderVertexCount(mNativeBuilder, vertexCount);
return this;
}
/**
* Defines how many buffers will be created in this vertex buffer set. These buffers are
* later referenced by index from 0 to <code>bufferCount</code> - 1.
*
* This call is mandatory. The default is 0.
*
* @param bufferCount number of buffers in this vertex buffer set. The maximum value is 8.
*
* @return this <code>Builder</code> for chaining calls
*/
@NonNull
public Builder bufferCount(@IntRange(from = 1) int bufferCount) {
nBuilderBufferCount(mNativeBuilder, bufferCount);
return this;
}
/**
* Sets up an attribute for this vertex buffer set.
*
* Using <code>byteOffset</code> and <code>byteStride</code>, attributes can be interleaved
* in the same buffer.
*
* <p>
* This is a no-op if the <code>attribute</code> is an invalid enum.
* This is a no-op if the <code>bufferIndex</code> is out of bounds.
* </p>
*
* <p>
* Warning: <code>VertexAttribute.TANGENTS</code> must be specified as a quaternion and is
* how normals are specified.
* </p>
*
* @param attribute the attribute to set up
* @param bufferIndex the index of the buffer containing the data for this attribute. Must
* be between 0 and bufferCount() - 1.
* @param attributeType the type of the attribute data (e.g. byte, float3, etc...)
* @param byteOffset offset in <i>bytes</i> into the buffer <code>bufferIndex</code>
* @param byteStride stride in <i>bytes</i> to the next element of this attribute. When
* set to zero the attribute size, as defined by
* <code>attributeType</code> is used.
*
* @return A reference to this <code>Builder</code> for chaining calls.
*
* @see VertexAttribute
*/
@NonNull
public Builder attribute(@NonNull VertexAttribute attribute,
@IntRange(from = 0) int bufferIndex, @NonNull AttributeType attributeType,
@@ -117,24 +224,82 @@ public class VertexBuffer {
return this;
}
/**
* Sets up an attribute for this vertex buffer set.
*
* Using <code>byteOffset</code> and <code>byteStride</code>, attributes can be interleaved
* in the same buffer.
*
* <p>
* This is a no-op if the <code>attribute</code> is an invalid enum.
* This is a no-op if the <code>bufferIndex</code> is out of bounds.
* </p>
*
* <p>
* Warning: <code>VertexAttribute.TANGENTS</code> must be specified as a quaternion and is
* how normals are specified.
* </p>
*
* @param attribute the attribute to set up
* @param bufferIndex the index of the buffer containing the data for this attribute. Must
* be between 0 and bufferCount() - 1.
* @param attributeType the type of the attribute data (e.g. byte, float3, etc...)
*
* @return A reference to this <code>Builder</code> for chaining calls.
*
* @see VertexAttribute
*/
@NonNull
public Builder attribute(@NonNull VertexAttribute attribute,
@IntRange(from = 0) int bufferIndex, @NonNull AttributeType attributeType) {
return attribute(attribute, bufferIndex, attributeType, 0, 0 );
}
/**
* Sets whether a given attribute should be normalized. By default attributes are not
* normalized. A normalized attribute is mapped between 0 and 1 in the shader. This applies
* only to integer types.
*
* @param attribute enum of the attribute to set the normalization flag to
*
* @return this <code>Builder</code> object for chaining calls.
*
* This is a no-op if the <code>attribute</code> is an invalid enum.
*/
@NonNull
public Builder normalized(@NonNull VertexAttribute attribute) {
nBuilderNormalized(mNativeBuilder, attribute.ordinal(), true);
return this;
}
/**
* Sets whether a given attribute should be normalized. By default attributes are not
* normalized. A normalized attribute is mapped between 0 and 1 in the shader. This applies
* only to integer types.
*
* @param attribute enum of the attribute to set the normalization flag to
* @param enabled true to automatically normalize the given attribute
*
* @return this <code>Builder</code> object for chaining calls.
*
* This is a no-op if the <code>attribute</code> is an invalid enum.
*/
@NonNull
public Builder normalized(@NonNull VertexAttribute attribute, boolean enabled) {
nBuilderNormalized(mNativeBuilder, attribute.ordinal(), enabled);
return this;
}
/**
* Creates the <code>VertexBuffer</code> object and returns a pointer to it.
*
* @param engine reference to the {@link Engine} to associate this <code>VertexBuffer</code>
* with
*
* @return the newly created <code>VertexBuffer</code> object
*
* @exception IllegalStateException if the VertexBuffer could not be created
*/
@NonNull
public VertexBuffer build(@NonNull Engine engine) {
long nativeVertexBuffer = nBuilderBuild(mNativeBuilder, engine.getNativeObject());
@@ -159,24 +324,68 @@ public class VertexBuffer {
}
}
/**
* Returns the vertex count.
*
* @return number of vertices in this vertex buffer set
*/
@IntRange(from = 0)
public int getVertexCount() {
return nGetVertexCount(getNativeObject());
}
/**
* Asynchronously copy-initializes the specified buffer from the given buffer data.
*
* @param engine reference to the {@link Engine} to associate this
* <code>VertexBuffer</code> with
* @param bufferIndex index of the buffer to initialize. Must be between 0 and
* bufferCount() - 1.
* @param buffer a CPU-side {@link Buffer} representing the data used to initialize
* the <code>VertexBuffer</code> at index <code>bufferIndex</code>.
* <code>buffer</code> should contain raw, untyped data that will
* be copied as-is into the buffer.
*/
public void setBufferAt(@NonNull Engine engine, int bufferIndex, @NonNull Buffer buffer) {
setBufferAt(engine, bufferIndex, buffer, 0, 0, null, null);
}
/**
* Asynchronously copy-initializes a region of the specified buffer from the given buffer data.
*
* @param engine reference to the {@link Engine} to associate this
* <code>VertexBuffer</code> with
* @param bufferIndex index of the buffer to initialize. Must be between 0 and
* bufferCount() - 1.
* @param buffer a CPU-side {@link Buffer} representing the data used to initialize
* the <code>VertexBuffer</code> at index <code>bufferIndex</code>.
* <code>buffer</code> should contain raw, untyped data that will
* be copied as-is into the buffer.
* @param destOffsetInBytes offset in <i>bytes</i> into the buffer at index
* <code>bufferIndex</code> of this vertex buffer set.
*/
public void setBufferAt(@NonNull Engine engine, int bufferIndex, @NonNull Buffer buffer,
@IntRange(from = 0) int destOffsetInBytes, @IntRange(from = 0) int count) {
setBufferAt(engine, bufferIndex, buffer, destOffsetInBytes, count, null, null);
}
/**
* Valid handler types:
* - Android: Handler, Executor
* - Other: Executor
* Asynchronously copy-initializes a region of the specified buffer from the given buffer data.
*
* @param engine reference to the {@link Engine} to associate this
* <code>VertexBuffer</code> with
* @param bufferIndex index of the buffer to initialize. Must be between 0 and
* bufferCount() - 1.
* @param buffer a CPU-side {@link Buffer} representing the data used to initialize
* the <code>VertexBuffer</code> at index <code>bufferIndex</code>.
* <code>buffer</code> should contain raw, untyped data that will
* be copied as-is into the buffer.
* @param destOffsetInBytes offset in <i>bytes</i> into the buffer at index
* <code>bufferIndex</code> of this vertex buffer set.
* @param handler an {@link java.util.concurrent.Executor Executor}. On Android this
* can also be a {@link android.os.Handler Handler}.
* @param callback a callback executed by <code>handler</code> when <code>buffer</code>
* is no longer needed.
*/
public void setBufferAt(@NonNull Engine engine, int bufferIndex, @NonNull Buffer buffer,
@IntRange(from = 0) int destOffsetInBytes, @IntRange(from = 0) int count,
@@ -188,7 +397,27 @@ public class VertexBuffer {
throw new BufferOverflowException();
}
}
/**
* Convenience function that consumes normal vectors (and, optionally, tangent vectors) and
* produces quaternions that can be passed into a TANGENTS buffer.
*
* <p>
* The given output buffer must be preallocated with at least quatCount * outStride bytes.
* <p>
*
* <p>
* Normals are required but tangents are optional, in which case this function tries to generate
* reasonable tangents. The given normals should be unit length.
* </p>
*
* <p>
* If supplied, the tangent vectors should be unit length and should be orthogonal to the
* normals. The w component of the tangent is a sign (-1 or +1) indicating handedness of the
* basis.
* </p>
*
* @param context an initialized QuatTangentContext object
*/
public static void populateTangentQuaternions(@NonNull QuatTangentContext context) {
nPopulateTangentQuaternions(context.quatType.ordinal(), context.quatCount,
context.outBuffer, context.outBuffer.remaining(), context.outStride,
@@ -197,7 +426,7 @@ public class VertexBuffer {
context.tangentsStride);
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed VertexBuffer");
}

View File

@@ -21,8 +21,39 @@ import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.annotation.Size;
import static com.google.android.filament.Colors.*;
import static com.google.android.filament.Colors.LinearColor;
/**
* Encompasses all the state needed for rendering a {@link Scene}.
*
* <p>
* {@link Renderer#render} operates on <code>View</code> objects. These <code>View</code> objects
* specify important parameters such as:
* </p>
*
* <ul>
* <li>The Scene</li>
* <li>The Camera</li>
* <li>The Viewport</li>
* <li>Some rendering parameters</li>
* </ul>
*
* <p>
* <code>View</code> instances are heavy objects that internally cache a lot of data needed for
* rendering. It is not advised for an application to use many View objects.
* </p>
*
* <p>
* For example, in a game, a <code>View</code> could be used for the main scene and another one for
* the game's user interface. More <code>View</code> instances could be used for creating special
* effects (e.g. a <code>View</code> is akin to a rendering pass).
* </p>
*
* @see Renderer
* @see Scene
* @see Camera
* @see RenderTarget
*/
public class View {
private long mNativeObject;
private String mName;
@@ -35,24 +66,102 @@ public class View {
private AmbientOcclusionOptions mAmbientOcclusionOptions;
private RenderTarget mRenderTarget;
/**
* Dynamic resolution can be used to either reach a desired target frame rate by lowering the
* resolution of a <code>View</code>, or to increase the quality when the rendering is faster
* than the target frame rate.
*
* <p>
* This structure can be used to specify the minimum scale factor used when lowering the
* resolution of a <code>View</code>, and the maximum scale factor used when increasing the
* resolution for higher quality rendering. The scale factors can be controlled on each X and Y
* axis independently. By default, all scale factors are set to 1.0.
* </p>
*
* <p>
* Dynamic resolution is only supported on platforms where the time to render a frame can be
* measured accurately. Dynamic resolution is currently only supported on Android.
* </p>
*/
public static class DynamicResolutionOptions {
/**
* Enables or disables dynamic resolution on a View.
*/
public boolean enabled = false;
/**
* If false, the system scales the major axis first.
*/
public boolean homogeneousScaling = false;
/**
* Desired frame time in milliseconds.
*/
public float targetFrameTimeMilli = 1000.0f / 60.0f;
/**
* Additional headroom for the GPU as a ratio of the targetFrameTime.
*/
public float headRoomRatio = 0.0f;
/**
* Rate at which the scale will change to reach the target frame rate.
*/
public float scaleRate = 0.125f;
/**
* The minimum scale in X and Y this View should use.
*/
public float minScale = 0.5f;
/**
* The maximum scale in X and Y this View should use.
*/
public float maxScale = 1.0f;
/**
* History size. higher values, tend to filter more (clamped to 30).
*/
public int history = 9;
}
/**
* Options for Ambient Occlusion
* @see #setAmbientOcclusion
*/
public static class AmbientOcclusionOptions {
/**
* Ambient Occlusion radius in meters, between 0 and ~10.
*/
public float radius = 0.3f;
/**
* Self-occlusion bias in meters. Use to avoid self-occlusion. Between 0 and a few mm.
*/
public float bias = 0.005f;
/**
* Controls ambient occlusion's contrast. Between 0 (linear) and 1 (squared)
*/
public float power = 0.0f;
/**
* How each dimension of the AO buffer is scaled. Must be positive and <= 1.
*/
public float resolution = 0.5f;
}
/**
* Sets the quality of the HDR color buffer.
*
* <p>
* A quality of <code>HIGH</code> or <code>ULTRA</code> means using an RGB16F or RGBA16F color
* buffer. This means colors in the LDR range (0..1) have 10 bit precision. A quality of
* <code>LOW</code> or <code>MEDIUM</code> means using an R11G11B10F opaque color buffer or an
* RGBA16F transparent color buffer. With R11G11B10F colors in the LDR range have a precision of
* either 6 bits (red and green channels) or 5 bits (blue channel).
* </p>
*/
public enum QualityLevel {
LOW,
MEDIUM,
@@ -60,30 +169,73 @@ public class View {
ULTRA
}
/**
* Structure used to set the color precision for the rendering of a <code>View</code>.
*
* <p>
* This structure offers separate quality settings for different parts of the rendering
* pipeline.
* </p>
*
* @see #setRenderQuality
* @see #getRenderQuality
*/
public static class RenderQuality {
public QualityLevel hdrColorBuffer = QualityLevel.HIGH;
}
/**
* List of available ambient occlusion techniques.
*
* @see #setAmbientOcclusion
*/
public enum AmbientOcclusion {
NONE,
SSAO
}
/**
* List of available post-processing anti-aliasing techniques.
*
* @see #setAntiAliasing
* @see #getAntiAliasing
*/
public enum AntiAliasing {
/**
* No anti aliasing performed as part of post-processing.
*/
NONE,
/**
* FXAA is a low-quality but very efficient type of anti-aliasing. (default).
*/
FXAA
}
/**
* List of available tone-mapping operators
*/
public enum ToneMapping {
/**
* Equivalent to disabling tone-mapping.
*/
LINEAR,
/**
* The Academy Color Encoding System (ACES).
*/
ACES
}
/**
* List of available post-processing dithering techniques.
*/
public enum Dithering {
NONE,
TEMPORAL
}
/** @see #setDepthPrepass */
public enum DepthPrepass {
DEFAULT(-1),
DISABLED(0),
@@ -100,52 +252,135 @@ public class View {
mNativeObject = nativeView;
}
/**
* Sets the View's name. Only useful for debugging.
*/
public void setName(@NonNull String name) {
mName = name;
nSetName(getNativeObject(), name);
}
/**
* Returns the View's name.
*/
@Nullable
public String getName() {
return mName;
}
/**
* Sets this View instance's Scene.
*
* <p>
* This method associates the specified Scene with this View. Note that a particular scene can
* be associated with several View instances. To remove an existing association, simply pass
* null.
* </p>
*
* <p>
* The View does not take ownership of the Scene pointer. Before destroying a Scene, be sure
* to remove it from all assoicated Views.
* </p>
*
* @see #getScene
*/
public void setScene(@Nullable Scene scene) {
mScene = scene;
nSetScene(getNativeObject(), scene == null ? 0 : scene.getNativeObject());
}
/**
* Gets this View's associated Scene, or null if none has been assigned.
*
* @see #setScene
*/
@Nullable
public Scene getScene() {
return mScene;
}
/**
* Sets this View's Camera.
*
* <p>
* This method associates the specified Camera with this View. A Camera can be associated with
* several View instances. To remove an existing association, simply pass
* null.
* </p>
*
* <p>
* The View does not take ownership of the Scene pointer. Before destroying a Camera, be sure
* to remove it from all assoicated Views.
* </p>
*
* @see #getCamera
*/
public void setCamera(@Nullable Camera camera) {
mCamera = camera;
nSetCamera(getNativeObject(), camera == null ? 0 : camera.getNativeObject());
}
/**
* Gets this View's associated Camera, or null if none has been assigned.
*
* @see #setCamera
*/
@Nullable
public Camera getCamera() {
return mCamera;
}
/**
* Specifies the rectangular rendering area.
*
* <p>
* The viewport specifies where the content of the View (i.e. the Scene) is rendered in
* the render target. The render target is automatically clipped to the Viewport.
* </p>
*
* <p>
* If you wish subsequent changes to take effect please call this method again in order to
* propagate the changes down to the native layer.
* </p>
*
* @param viewport The Viewport to render the Scene into.
*/
public void setViewport(@NonNull Viewport viewport) {
mViewport = viewport;
nSetViewport(getNativeObject(),
mViewport.left, mViewport.bottom, mViewport.width, mViewport.height);
}
/**
* Returns the rectangular rendering area.
*
* @see #setViewport
*/
@NonNull
public Viewport getViewport() {
return mViewport;
}
/**
* Sets the color used to clear the Viewport when rendering this View.
*
* <p>This is ignored if a {@link Skybox} is present or if clearing has been disabled
* via {@link #setClearTargets}. Defaults to black.</p>
*
* @see #getClearColor
*/
public void setClearColor(
@LinearColor float r, @LinearColor float g, @LinearColor float b, float a) {
nSetClearColor(getNativeObject(), r, g, b, a);
}
/**
* Returns the View clear color in a provided 4-tuple.
*
* @return A reference to the passed-in array.
*
* @see #setClearColor
*/
@NonNull @Size(min = 4)
public float[] getClearColor(@NonNull @Size(min = 4) float[] out) {
out = Asserts.assertFloat4(out);
@@ -153,65 +388,178 @@ public class View {
return out;
}
/**
* Sets which targets to clear (default: true, true, false)
*
* @see #setClearColor
*/
public void setClearTargets(boolean color, boolean depth, boolean stencil) {
nSetClearTargets(getNativeObject(), color, depth, stencil);
}
/**
* Sets which layers are visible.
*
* <p>
* Renderable objects can have one or several layers associated to them. Layers are
* represented with an 8-bits bitmask, where each bit corresponds to a layer.
* By default all layers are visible.
* </p>
*
* @see RenderableManager#setLayerMask
*
* @param select a bitmask specifying which layer to set or clear using <code>values</code>.
* @param values a bitmask where each bit sets the visibility of the corresponding layer
* (1: visible, 0: invisible), only layers in <code>select</code> are affected.
*/
public void setVisibleLayers(
@IntRange(from = 0, to = 255) int select,
@IntRange(from = 0, to = 255) int values) {
nSetVisibleLayers(getNativeObject(), select & 0xFF, values & 0xFF);
}
/**
* Enables or disables shadow mapping. Enabled by default.
*
* @see LightManager.Builder#castShadows
* @see RenderableManager.Builder#receiveShadows
* @see RenderableManager.Builder#castShadows
*/
public void setShadowsEnabled(boolean enabled) {
nSetShadowsEnabled(getNativeObject(), enabled);
}
/**
* Specifies an offscreen render target to render into.
*
* <p>
* By default, the view's associated render target is null, which corresponds to the
* SwapChain associated with the engine.
* </p>
*
* @param target render target associated with view, or null for the swap chain
*/
public void setRenderTarget(@Nullable RenderTarget target) {
mRenderTarget = target;
nSetRenderTarget(getNativeObject(), target != null ? target.getNativeObject() : 0);
}
/**
* Gets the offscreen render target associated with this view.
*
* Returns null if the render target is the swap chain (which is default).
*
* @see #setRenderTarget
*/
@Nullable
public RenderTarget getRenderTarget() {
return mRenderTarget;
}
/**
* Sets how many samples are to be used for MSAA in the post-process stage.
* Default is 1 and disables MSAA.
*
* <p>
* Note that anti-aliasing can also be performed in the post-processing stage, generally at
* lower cost. See the FXAA option in {@link #setAntiAliasing}.
* </p>
*
* @param count number of samples to use for multi-sampled anti-aliasing.
*/
public void setSampleCount(int count) {
nSetSampleCount(getNativeObject(), count);
}
/**
* Returns the effective MSAA sample count.
*
* <p>
* A value of 0 or 1 means MSAA is disabled.
* </p>
*
* @return value set by {@link #setSampleCount}
*/
public int getSampleCount() {
return nGetSampleCount(getNativeObject());
}
/**
* Enables or disables anti-aliasing in the post-processing stage. Enabled by default.
*
* <p>
* For MSAA anti-aliasing, see {@link #setSampleCount}.
* </p>
*
* @param type FXAA for enabling, NONE for disabling anti-aliasing.
*/
public void setAntiAliasing(@NonNull AntiAliasing type) {
nSetAntiAliasing(getNativeObject(), type.ordinal());
}
/**
* Queries whether anti-aliasing is enabled during the post-processing stage. To query
* whether MSAA is enabled, see {@link #getSampleCount}.
*
* @return The post-processing anti-aliasing method.
*/
@NonNull
public AntiAliasing getAntiAliasing() {
return AntiAliasing.values()[nGetAntiAliasing(getNativeObject())];
}
/**
* Enables or disables tone-mapping in the post-processing stage. Enabled by default.
*
* @param type Tone-mapping function.
*/
public void setToneMapping(@NonNull ToneMapping type) {
nSetToneMapping(getNativeObject(), type.ordinal());
}
/**
* Returns the tone-mapping function.
* @return tone-mapping function.
*/
@NonNull
public ToneMapping getToneMapping() {
return ToneMapping.values()[nGetToneMapping(getNativeObject())];
}
/**
* Enables or disables dithering in the post-processing stage. Enabled by default.
*
* @param dithering dithering type
*/
public void setDithering(@NonNull Dithering dithering) {
nSetDithering(getNativeObject(), dithering.ordinal());
}
/**
* Queries whether dithering is enabled during the post-processing stage.
*
* @return the current dithering type for this view.
*/
@NonNull
public Dithering getDithering() {
return Dithering.values()[nGetDithering(getNativeObject())];
}
/**
* Sets the dynamic resolution options for this view.
*
* <p>
* Dynamic resolution options controls whether dynamic resolution is enabled, and if it is,
* how it behaves.
* </p>
*
* <p>
* If you wish subsequent changes to take effect please call this method again in order to
* propagate the changes down to the native layer.
* </p>
*
* @param options The dynamic resolution options to use on this view
*/
public void setDynamicResolutionOptions(@NonNull DynamicResolutionOptions options) {
mDynamicResolution = options;
nSetDynamicResolutionOptions(getNativeObject(),
@@ -225,6 +573,10 @@ public class View {
options.history);
}
/**
* Returns the dynamic resolution options associated with this view.
* @return value set by {@link #setDynamicResolutionOptions}.
*/
@NonNull
public DynamicResolutionOptions getDynamicResolutionOptions() {
if (mDynamicResolution == null) {
@@ -233,11 +585,20 @@ public class View {
return mDynamicResolution;
}
/**
* Sets the rendering quality for this view (e.g. color precision).
*
* @param renderQuality The render quality to use on this view
*/
public void setRenderQuality(@NonNull RenderQuality renderQuality) {
mRenderQuality = renderQuality;
nSetRenderQuality(getNativeObject(), renderQuality.hdrColorBuffer.ordinal());
}
/**
* Returns the render quality used by this view.
* @return value set by {@link #setRenderQuality}.
*/
@NonNull
public RenderQuality getRenderQuality() {
if (mRenderQuality == null) {
@@ -246,50 +607,168 @@ public class View {
return mRenderQuality;
}
/**
* Checks if this view is rendered with a depth-only prepass.
*
* @return the value set by {@link #setDepthPrepass}.
*/
@NonNull
public DepthPrepass getDepthPrepass() {
return mDepthPrepass;
}
/**
* Sets whether this view is rendered with or without a depth pre-pass.
*
* <p>
* By default, the system picks the most appropriate strategy for your platform; this method
* lets you override that strategy.
* </p>
*
* <p>
* When the depth pre-pass is enabled, the renderer will first draw all objects in the
* depth buffer from front to back, and then draw the objects again but sorted to minimize
* state changes. With the depth pre-pass disabled, objects are drawn only once, but it may
* result in more state changes or more overdraw.
* </p>
*
* <p>
* The best strategy may depend on the scene and/or GPU.
* </p>
*
* <ul>
* <li>DepthPrepass::DEFAULT uses the most appropriate strategy</li>
* <li>DepthPrepass::DISABLED disables the depth pre-pass</li>
* <li>DepthPrepass::ENABLE enables the depth pre-pass</li>
* </ul>
*/
public void setDepthPrepass(@NonNull DepthPrepass depthPrepass) {
mDepthPrepass = depthPrepass;
nSetDepthPrepass(getNativeObject(), depthPrepass.value);
}
/**
* Returns true if post-processing is enabled.
*
* @see #setPostProcessingEnabled
*/
public boolean isPostProcessingEnabled() {
return nIsPostProcessingEnabled(getNativeObject());
}
/**
* Enables or disables post processing. Enabled by default.
*
* <p>Post-processing includes:</p>
* <ul>
* <li>Tone-mapping & gamma encoding</li>
* <li>Dithering</li>
* <li>MSAA</li>
* <li>FXAA</li>
* <li>Dynamic scaling</li>
* </ul>
*
* <p>
* Disabling post-processing forgoes color correctness as well as anti-aliasing and
* should only be used experimentally (e.g., for UI overlays).
* </p>
*
* @param enabled true enables post processing, false disables it
*
* @see #setToneMapping
* @see #setAntiAliasing
* @see #setDithering
* @see #setSampleCount
*/
public void setPostProcessingEnabled(boolean enabled) {
nSetPostProcessingEnabled(getNativeObject(), enabled);
}
/**
* Returns true if post-processing is enabled.
*
* @see #setPostProcessingEnabled
*/
public boolean isFrontFaceWindingInverted() {
return nIsFrontFaceWindingInverted(getNativeObject());
}
/**
* Inverts the winding order of front faces. By default front faces use a counter-clockwise
* winding order. When the winding order is inverted, front faces are faces with a clockwise
* winding order.
*
* Changing the winding order will directly affect the culling mode in materials
* (see Material#getCullingMode).
*
* Inverting the winding order of front faces is useful when rendering mirrored reflections
* (water, mirror surfaces, front camera in AR, etc.).
*
* @param inverted True to invert front faces, false otherwise.
*/
public void setFrontFaceWindingInverted(boolean inverted) {
nSetFrontFaceWindingInverted(getNativeObject(), inverted);
}
/**
* Sets options relative to dynamic lighting for this view.
*
* <p>
* Together <code>zLightNear</code> and <code>zLightFar</code> must be chosen so that the
* visible influence of lights is spread between these two values.
* </p>
*
* @param zLightNear Distance from the camera where the lights are expected to shine.
* This parameter can affect performance and is useful because depending
* on the scene, lights that shine close to the camera may not be
* visible -- in this case, using a larger value can improve performance.
* e.g. when standing and looking straight, several meters of the ground
* isn't visible and if lights are expected to shine there, there is no
* point using a short zLightNear. (Default 5m).
*
* @param zLightFar Distance from the camera after which lights are not expected to be visible.
* Similarly to zLightNear, setting this value properly can improve
* performance. (Default 100m).
*
*/
public void setDynamicLightingOptions(float zLightNear, float zLightFar) {
nSetDynamicLightingOptions(getNativeObject(), zLightNear, zLightFar);
}
/**
* Activates or deactivates ambient occlusion.
*
* @param ao Type of ambient occlusion to use.
*/
public void setAmbientOcclusion(@NonNull AmbientOcclusion ao) {
nSetAmbientOcclusion(getNativeObject(), ao.ordinal());
}
/**
* Queries the type of ambient occlusion active for this View.
*
* @return ambient occlusion type.
*/
@NonNull
public AmbientOcclusion getAmbientOcclusion() {
return AmbientOcclusion.values()[nGetAmbientOcclusion(getNativeObject())];
}
/**
* Sets ambient occlusion options.
*
* @param options Options for ambient occlusion.
*/
public void setAmbientOcclusionOptions(@NonNull AmbientOcclusionOptions options) {
mAmbientOcclusionOptions = options;
nSetAmbientOcclusionOptions(getNativeObject(), options.radius, options.bias, options.power, options.resolution);
}
/**
* Gets the ambient occlusion options.
*
* @return ambient occlusion options currently set.
*/
@NonNull
public AmbientOcclusionOptions getAmbientOcclusionOptions() {
if (mAmbientOcclusionOptions == null) {
@@ -298,7 +777,7 @@ public class View {
return mAmbientOcclusionOptions;
}
long getNativeObject() {
public long getNativeObject() {
if (mNativeObject == 0) {
throw new IllegalStateException("Calling method on destroyed View");
}

View File

@@ -18,6 +18,16 @@ package com.google.android.filament;
import android.support.annotation.IntRange;
/**
* Specifies a rectangular region within a render target in terms of pixel coordinates.
*
* <p>
* The rectangle spans from <code>(left,bottom)</code> to <code>(left+width-1, top+height-1)</code>,
* inclusive. Width and height must be non-negative.
* </p>
*
* @see View#setViewport
*/
public class Viewport {
public Viewport(int left, int bottom, @IntRange(from = 0) int width, @IntRange(from = 0) int height) {
this.left = left;

View File

@@ -34,21 +34,6 @@ public final class TextureHelper {
private static final int BITMAP_CONFIG_RGBA_F16 = 4;
private static final int BITMAP_CONFIG_HARDWARE = 5;
private static Method sEngineGetNativeObject;
private static Method sTextureGetNativeObject;
static {
try {
sEngineGetNativeObject = Engine.class.getDeclaredMethod("getNativeObject");
sTextureGetNativeObject = Texture.class.getDeclaredMethod("getNativeObject");
sEngineGetNativeObject.setAccessible(true);
sTextureGetNativeObject.setAccessible(true);
} catch (NoSuchMethodException e) {
// Cannot happen
}
}
private TextureHelper() {
}
@@ -69,14 +54,10 @@ public final class TextureHelper {
throw new IllegalArgumentException("Unsupported config: ARGB_4444 or HARDWARE");
}
try {
long nativeTexture = (Long) sTextureGetNativeObject.invoke(texture);
long nativeEngine = (Long) sEngineGetNativeObject.invoke(engine);
nSetBitmap(nativeTexture, nativeEngine, level, xoffset, yoffset, width, height,
bitmap, format);
} catch (Exception e) {
// Ignored
}
long nativeTexture = texture.getNativeObject();
long nativeEngine = engine.getNativeObject();
nSetBitmap(nativeTexture, nativeEngine, level, xoffset, yoffset, width, height,
bitmap, format);
}
private static int toNativeFormat(Bitmap.Config config) {

View File

@@ -18,7 +18,6 @@ package com.google.android.filament.android;
import android.graphics.PixelFormat;
import android.graphics.SurfaceTexture;
import android.os.Build;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
@@ -27,6 +26,7 @@ import android.view.Surface;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.TextureView;
import com.google.android.filament.SwapChain;
/**
@@ -99,10 +99,6 @@ import com.google.android.filament.SwapChain;
* // Always detach the surface before destroying the engine
* mUiHelper.detach();
*
* // This ensures that all the commands we've sent to Filament have
* // been processed before we attempt to destroy anything
* Fence.waitAndDestroy(mEngine.createFence(Fence.Type.SOFT), Fence.Mode.FLUSH);
*
* mEngine.destroy();
* }
*

View File

@@ -26,16 +26,14 @@ import java.lang.annotation.Target;
*
* Note that adding this annotation to a method is not enough to guarantee that
* it is kept - either its class must be referenced elsewhere in the program, or
* the class must be annotated with this as well.
* the class must be annotated with this as well. Usage example:
* <pre>
*
* Usage example:<br />
* {@code
* @UsedByNative("NativeCrashHandler.cpp")
public static void reportCrash(int signal, int code, int address) {
...
}
}
* &commat;UsedByNative("NativeCrashHandler.cpp")
* public static void reportCrash(int signal, int code, int address) {
* ...
* }
* </pre>
*/
@Target({
ElementType.METHOD,

View File

@@ -26,16 +26,14 @@ import java.lang.annotation.Target;
*
* Note that adding this annotation to a method is not enough to guarantee that
* it is kept - either its class must be referenced elsewhere in the program, or
* the class must be annotated with this as well.
* the class must be annotated with this as well. Usage example:
* <pre>
*
* Usage example:<br />
* {@code
* @UsedByReflection("PeopleListItemView.java")
public PeopleListItemViewV11(Context context) {
super(context);
}
}
* &commat;UsedByReflection("PeopleListItemView.java")
* public PeopleListItemViewV11(Context context) {
* super(context);
* }
* </pre>
*/
@Target({
ElementType.METHOD,

View File

@@ -9,3 +9,4 @@
/build
/captures
.externalNativeBuild
/.cxx

View File

@@ -2,19 +2,22 @@ cmake_minimum_required(VERSION 3.6)
set(FILAMENT_DIR ${FILAMENT_DIST_DIR})
add_library(filament SHARED IMPORTED)
set(DISABLE_FILAMENT_JNI TRUE)
add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/../filament-android ${CMAKE_CURRENT_BINARY_DIR}/filament-android)
add_library(filament STATIC IMPORTED)
set_target_properties(filament PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libfilament.a)
add_library(backend SHARED IMPORTED)
add_library(backend STATIC IMPORTED)
set_target_properties(backend PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libbackend.a)
add_library(utils SHARED IMPORTED)
add_library(utils STATIC IMPORTED)
set_target_properties(utils PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libutils.a)
add_library(filaflat SHARED IMPORTED)
add_library(filaflat STATIC IMPORTED)
set_target_properties(filaflat PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libfilaflat.a)
@@ -26,27 +29,27 @@ add_library(ibl STATIC IMPORTED)
set_target_properties(ibl PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libibl.a)
add_library(geometry SHARED IMPORTED)
add_library(geometry STATIC IMPORTED)
set_target_properties(geometry PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libgeometry.a)
add_library(filabridge SHARED IMPORTED)
add_library(filabridge STATIC IMPORTED)
set_target_properties(filabridge PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libfilabridge.a)
add_library(gltfio SHARED IMPORTED)
add_library(gltfio STATIC IMPORTED)
set_target_properties(gltfio PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libgltfio_core.a)
add_library(gltfio_resources SHARED IMPORTED)
add_library(gltfio_resources STATIC IMPORTED)
set_target_properties(gltfio_resources PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libgltfio_resources.a)
add_library(bluevk SHARED IMPORTED)
add_library(bluevk STATIC IMPORTED)
set_target_properties(bluevk PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libbluevk.a)
add_library(smol-v SHARED IMPORTED)
add_library(smol-v STATIC IMPORTED)
set_target_properties(smol-v PROPERTIES IMPORTED_LOCATION
${FILAMENT_DIR}/lib/${ANDROID_ABI}/libsmol-v.a)
@@ -54,25 +57,20 @@ include_directories(${FILAMENT_DIR}/include
..
../../libs/utils/include)
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fno-stack-protector")
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fno-exceptions -fno-unwind-tables -fno-asynchronous-unwind-tables -fno-rtti")
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -ffast-math -ffp-contract=fast")
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fvisibility-inlines-hidden")
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fvisibility=hidden")
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -fomit-frame-pointer -ffunction-sections -fdata-sections")
set(CMAKE_SHARED_LINKER_FLAGS " ${CMAKE_SHARED_LINKER_FLAGS} -Wl,--gc-sections")
set(CMAKE_SHARED_LINKER_FLAGS " ${CMAKE_SHARED_LINKER_FLAGS} -Wl,-Bsymbolic-functions")
set(CMAKE_SHARED_LINKER_FLAGS_RELEASE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE} -Wl,--version-script=${CMAKE_SOURCE_DIR}/libgltfio-jni.map")
add_library(gltfio-jni SHARED
src/main/cpp/Animator.cpp
src/main/cpp/AssetLoader.cpp
src/main/cpp/FilamentAsset.cpp
src/main/cpp/KtxLoader.cpp
src/main/cpp/MaterialProvider.cpp
src/main/cpp/ResourceLoader.cpp
../common/CallbackUtils.cpp
../common/NioUtils.cpp
$<TARGET_OBJECTS:filament-jni-obj>
)
set_target_properties(gltfio-jni PROPERTIES LINK_DEPENDS
@@ -80,21 +78,22 @@ set_target_properties(gltfio-jni PROPERTIES LINK_DEPENDS
# The ordering in the following list is important because CMake does not have dependency information.
target_link_libraries(gltfio-jni
android
gltfio
gltfio_resources
filament
filabridge
backend
filaflat
filabridge
geometry
image
ibl
utils
log
GLESv3
EGL
android
jnigraphics
gltfio_resources
m
log
)
if (FILAMENT_SUPPORTS_VULKAN)

View File

@@ -13,7 +13,7 @@ buildscript {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:3.4.1'
classpath 'com.android.tools.build:gradle:3.5.0'
}
}
@@ -25,7 +25,7 @@ allprojects {
}
group = "com.google.android.filament"
version = "0.1"
version = "1.3"
apply plugin: 'com.android.library'
@@ -35,10 +35,10 @@ if (project.hasProperty("filament_dist_dir")) {
}
android {
compileSdkVersion 28
compileSdkVersion 29
defaultConfig {
minSdkVersion 14
targetSdkVersion 28
targetSdkVersion 29
versionCode 1
versionName "1.0"
@@ -47,7 +47,7 @@ android {
externalNativeBuild {
cmake {
arguments.add("-DANDROID_PIE=ON")
arguments.add("-DANDROID_PLATFORM=android-21")
arguments.add("-DANDROID_PLATFORM=android-19")
arguments.add("-DANDROID_STL=c++_static")
arguments.add("-DFILAMENT_DIST_DIR=${filament_path}".toString())
cppFlags.add("-std=c++14")

View File

@@ -1,5 +1,5 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4-bin.zip
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

View File

@@ -0,0 +1,59 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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 <gltfio/Animator.h>
using namespace filament;
using namespace filament::math;
using namespace gltfio;
using namespace utils;
extern "C" JNIEXPORT void JNICALL
Java_com_google_android_filament_gltfio_Animator_nApplyAnimation(JNIEnv*, jclass, jlong nativeAnimator,
jint index, jfloat time) {
Animator* animator = (Animator*) nativeAnimator;
animator->applyAnimation(index, time);
}
extern "C" JNIEXPORT void JNICALL
Java_com_google_android_filament_gltfio_Animator_nUpdateBoneMatrices(JNIEnv*, jclass, jlong nativeAnimator) {
Animator* animator = (Animator*) nativeAnimator;
animator->updateBoneMatrices();
}
extern "C" JNIEXPORT jint JNICALL
Java_com_google_android_filament_gltfio_Animator_nGetAnimationCount(JNIEnv*, jclass, jlong nativeAnimator) {
Animator* animator = (Animator*) nativeAnimator;
return animator->getAnimationCount();
}
extern "C" JNIEXPORT float JNICALL
Java_com_google_android_filament_gltfio_Animator_nGetAnimationDuration(JNIEnv*, jclass,
jlong nativeAnimator, jint index) {
Animator* animator = (Animator*) nativeAnimator;
return animator->getAnimationDuration(index);
}
extern "C" JNIEXPORT jstring JNICALL
Java_com_google_android_filament_gltfio_Animator_nGetAnimationName(JNIEnv* env, jclass,
jlong nativeAnimator, jint index) {
Animator* animator = (Animator*) nativeAnimator;
const char* val = animator->getAnimationName(index);
return val ? env->NewStringUTF(val) : nullptr;
}

View File

@@ -71,3 +71,10 @@ Java_com_google_android_filament_gltfio_FilamentAsset_nGetName(JNIEnv* env, jcla
const char* val = asset->getName(*entity);
return val ? env->NewStringUTF(val) : nullptr;
}
extern "C" JNIEXPORT jlong JNICALL
Java_com_google_android_filament_gltfio_FilamentAsset_nGetAnimator(JNIEnv* env, jclass,
jlong nativeAsset) {
FilamentAsset* asset = (FilamentAsset*) nativeAsset;
return (jlong) asset->getAnimator();
}

View File

@@ -34,7 +34,7 @@ Java_com_google_android_filament_gltfio_KtxLoader_nCreateTexture(JNIEnv* env, jc
Engine* engine = (Engine*) nativeEngine;
AutoBuffer buffer(env, javaBuffer, remaining);
KtxBundle* bundle = new KtxBundle((const uint8_t*) buffer.getData(), buffer.getSize());
return (jlong) KtxUtility::createTexture(engine, *bundle, srgb, [](void* userdata) {
return (jlong) ktx::createTexture(engine, *bundle, srgb, [](void* userdata) {
KtxBundle* bundle = (KtxBundle*) userdata;
delete bundle;
}, bundle);
@@ -46,7 +46,7 @@ Java_com_google_android_filament_gltfio_KtxLoader_nCreateIndirectLight(JNIEnv* e
Engine* engine = (Engine*) nativeEngine;
AutoBuffer buffer(env, javaBuffer, remaining);
KtxBundle* bundle = new KtxBundle((const uint8_t*) buffer.getData(), buffer.getSize());
Texture* cubemap = KtxUtility::createTexture(engine, *bundle, srgb, [](void* userdata) {
Texture* cubemap = ktx::createTexture(engine, *bundle, srgb, [](void* userdata) {
KtxBundle* bundle = (KtxBundle*) userdata;
delete bundle;
}, bundle);
@@ -69,7 +69,7 @@ Java_com_google_android_filament_gltfio_KtxLoader_nCreateSkybox(JNIEnv* env, jcl
Engine* engine = (Engine*) nativeEngine;
AutoBuffer buffer(env, javaBuffer, remaining);
KtxBundle* bundle = new KtxBundle((const uint8_t*) buffer.getData(), buffer.getSize());
Texture* cubemap = KtxUtility::createTexture(engine, *bundle, srgb, [](void* userdata) {
Texture* cubemap = ktx::createTexture(engine, *bundle, srgb, [](void* userdata) {
KtxBundle* bundle = (KtxBundle*) userdata;
delete bundle;
}, bundle);

View File

@@ -0,0 +1,103 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.google.android.filament.gltfio;
import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
/**
* Updates matrices according to glTF <code>animation</code> and <code>skin</code> definitions.
*
* <p>Animator can be used for two things:
* <ul>
* <li>Updating matrices in <code>TransformManager</code> components according to glTF <code>animation</code> definitions.</li>
* <li>Updating bone matrices in <code>RenderableManager</code> components according to glTF <code>skin</code> definitions.</li>
* </ul>
* </p>
*
* @see AssetLoader
* @see FilamentAsset
* @see ResourceLoader
*/
public class Animator {
private long mNativeObject;
Animator(long nativeObject) {
mNativeObject = nativeObject;
}
/**
* Applies rotation, translation, and scale to entities that have been targeted by the given
* animation definition. Uses <code>TransformManager</code>.
*
* @param animationIndex Zero-based index for the <code>animation</code> of interest.
* @param time Elapsed time of interest in seconds.
*
* @see #getAnimationCount
*/
public void applyAnimation(@IntRange(from = 0) int animationIndex, float time) {
nApplyAnimation(mNativeObject, animationIndex, time);
}
/**
* Computes root-to-node transforms for all bone nodes, then passes
* the results into {@see RenderableManager#setBones}.
* Uses <code>TransformManager</code> and <code>RenderableManager</code>.
*
* <p>NOTE: this operation is independent of <code>animation</code>.</p>
*/
public void updateBoneMatrices() {
nUpdateBoneMatrices(mNativeObject);
}
/**
* Returns the number of <code>animation</code> definitions in the glTF asset.
*/
public int getAnimationCount() {
return nGetAnimationCount(mNativeObject);
}
/**
* Returns the duration of the specified glTF <code>animation</code> in seconds.
*
* @param animationIndex Zero-based index for the <code>animation</code> of interest.
*
* @see #getAnimationCount
* */
public float getAnimationDuration(@IntRange(from = 0) int animationIndex) {
return nGetAnimationDuration(mNativeObject, animationIndex);
}
/**
* Returns a weak reference to the string name of the specified <code>animation</code>, or an
* empty string if none was specified.
*
* @param animationIndex Zero-based index for the <code>animation</code> of interest.
*
* @see #getAnimationCount
*/
public String getAnimationName(@IntRange(from = 0) int animationIndex) {
return nGetAnimationName(mNativeObject, animationIndex);
}
private static native void nApplyAnimation(long nativeAnimator, int index, float time);
private static native void nUpdateBoneMatrices(long nativeAnimator);
private static native int nGetAnimationCount(long nativeAnimator);
private static native float nGetAnimationDuration(long nativeAnimator, int index);
private static native String nGetAnimationName(long nativeAnimator, int index);
}

View File

@@ -21,81 +21,115 @@ import android.support.annotation.Nullable;
import com.google.android.filament.Engine;
import com.google.android.filament.EntityManager;
import com.google.android.filament.IndirectLight;
import com.google.android.filament.Skybox;
import com.google.android.filament.Texture;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.nio.Buffer;
/**
* Consumes a blob of glTF 2.0 content (either JSON or GLB) and produces {@link FilamentAsset}
* objects, which are bundles of Filament entities, material instances, textures, vertex buffers,
* and index buffers.
*
* <p>AssetLoader does not fetch external buffer data or create textures on its own. Clients can use
* the provided {@link ResourceLoader} class for this, which obtains the URI list from the asset.
* This is demonstrated in the Kotlin snippet below.</p>
*
* <pre>
*
* companion object {
* init {
* Filament.init()
* AssetLoader.init()
* }
* }
*
* override fun onCreate(savedInstanceState: Bundle?) {
*
* ...
*
* assetLoader = AssetLoader(engine, MaterialProvider(engine), EntityManager.get())
*
* filamentAsset = assets.open("models/lucy.glb").use { input ->
* val bytes = ByteArray(input.available())
* input.read(bytes)
* assetLoader.createAssetFromBinary(ByteBuffer.wrap(bytes))!!
* }
*
* ResourceLoader(engine).loadResources(filamentAsset).destroy()
* animator = asset.getAnimator()
*
* scene.addEntities(filamentAsset.entities)
* }
* </pre>
*
* @see Animator
* @see FilamentAsset
* @see ResourceLoader
*/
public class AssetLoader {
private long mNativeObject;
static Method sEngineGetNativeObject;
static Method sEntityManagerGetNativeObject;
static Constructor<Texture> sTextureConstructor;
static Constructor<IndirectLight> sIndirectLightConstructor;
static Constructor<Skybox> sSkyboxConstructor;
/**
* Initializes the gltfio JNI layer. Must be called before using any gltfio functionality.
*/
public static void init() {
System.loadLibrary("gltfio-jni");
try {
sEngineGetNativeObject = Engine.class.getDeclaredMethod("getNativeObject");
sEngineGetNativeObject.setAccessible(true);
sEntityManagerGetNativeObject = EntityManager.class.getDeclaredMethod("getNativeObject");
sEntityManagerGetNativeObject.setAccessible(true);
sTextureConstructor = Texture.class.getDeclaredConstructor(long.class);
sTextureConstructor.setAccessible(true);
sIndirectLightConstructor = IndirectLight.class.getDeclaredConstructor(long.class);
sIndirectLightConstructor.setAccessible(true);
sSkyboxConstructor = Skybox.class.getDeclaredConstructor(long.class);
sSkyboxConstructor.setAccessible(true);
} catch (NoSuchMethodException e) {
// Cannot happen
}
}
/**
* Constructs an <code>AssetLoader </code>that can be used to create and destroy instances of
* {@link FilamentAsset}.
*
* @param engine the engine that the loader should pass to builder objects
* @param generator specifies if materials should be generated or loaded from a pre-built set
* @param entities the EntityManager that should be used to create entities
*/
public AssetLoader(@NonNull Engine engine, @NonNull MaterialProvider generator,
@NonNull EntityManager entities) {
try {
long nativeEngine = (long) sEngineGetNativeObject.invoke(engine);
long nativeMaterials = generator.getNativeObject();
long nativeEntities = (long) sEntityManagerGetNativeObject.invoke(entities);
mNativeObject = nCreateAssetLoader(nativeEngine, nativeMaterials, nativeEntities);
} catch (Exception e) {
// Ignored
}
long nativeEngine = engine.getNativeObject();
long nativeMaterials = generator.getNativeObject();
long nativeEntities = entities.getNativeObject();
mNativeObject = nCreateAssetLoader(nativeEngine, nativeMaterials, nativeEntities);
if (mNativeObject == 0) {
throw new IllegalStateException("Unable to parse glTF asset.");
}
}
/**
* Frees all memory consumed by the native <code>AssetLoader</code> and its material cache.
*/
public void destroy() {
nDestroyAssetLoader(mNativeObject);
mNativeObject = 0;
}
/**
* Creates a {@link FilamentAsset} from the contents of a GLB file.
*/
@Nullable
public FilamentAsset createAssetFromBinary(@NonNull Buffer buffer) {
long nativeAsset = nCreateAssetFromBinary(mNativeObject, buffer, buffer.remaining());
return new FilamentAsset(nativeAsset);
}
/**
* Allows clients to enable diagnostic shading on newly-loaded assets.
*/
public void enableDiagnostics(boolean enable) {
nEnableDiagnostics(mNativeObject, enable);
}
/**
* Frees all memory associated with the given {@link FilamentAsset}.
*/
public void destroyAsset(@Nullable FilamentAsset asset) {
nDestroyAsset(mNativeObject, asset.getNativeObject());
asset.clearNativeObject();
}
private static native long nCreateAssetLoader(long nativeEngine, long nativeGenerator, long nativeEntities);
private static native long nCreateAssetLoader(long nativeEngine, long nativeGenerator,
long nativeEntities);
private static native void nDestroyAssetLoader(long nativeLoader);
private static native long nCreateAssetFromBinary(long nativeLoader, Buffer buffer, int remaining);
private static native void nEnableDiagnostics(long nativeLoader, boolean enable);

View File

@@ -21,37 +21,88 @@ import android.support.annotation.NonNull;
import com.google.android.filament.Box;
import com.google.android.filament.Entity;
/**
* Owns a bundle of Filament objects that have been created by <code>AssetLoader</code>.
*
* <p>For usage instructions, see the documentation for {@link AssetLoader}.</p>
*
* <p>This class owns a hierarchy of entities that have been loaded from a glTF asset. Every entity has
* a <code>TransformManager</code> component, and some entities also have
* <code>NameComponentManager</code> and/or <code>RenderableManager</code> components.</p>
*
* <p>In addition to the aforementioned entities, an asset has strong ownership over a list of
* <code>VertexBuffer</code>, <code>IndexBuffer</code>, <code>MaterialInstance</code>, and
* <code>Texture</code>.</p>
*
* <p>Clients can use {@link ResourceLoader} to create textures, compute tangent quaternions, and
* upload data into vertex buffers and index buffers.</p>
*
* @see ResourceLoader
* @see Animator
* @see AssetLoader
*/
public class FilamentAsset {
private long mNativeObject;
private Animator mAnimator;
FilamentAsset(long nativeObject) {
mNativeObject = nativeObject;
mAnimator = null;
}
long getNativeObject() {
return mNativeObject;
}
/**
* Gets the transform root for the asset, which has no matching glTF node.
*/
public @Entity int getRoot() {
return nGetRoot(mNativeObject);
}
/**
* Gets the list of entities, one for each glTF node.
*
* <p>All of these have a transform component. Some of the returned entities may also have a
* renderable component.</p>
*/
public @Entity int[] getEntities() {
int[] result = new int[nGetEntityCount(mNativeObject)];
nGetEntities(mNativeObject, result);
return result;
}
/**
* Gets the bounding box computed from the supplied min / max values in glTF accessors.
*/
public @NonNull Box getBoundingBox() {
float[] box = new float[6];
nGetBoundingBox(mNativeObject, box);
return new Box(box[0], box[1], box[2], box[3], box[4], box[5]);
}
/**
* Gets the <code>NameComponentManager</code> label for the given entity, if it exists.
*/
public String getName(@Entity int entity) {
return nGetName(getNativeObject(), entity);
}
/**
* Creates or retrieves the <code>Animator</code> for this asset.
*
* <p>When calling this for the first time, this must be called after
* {@see ResourceLoader#loadResources}.</p>
*/
public Animator getAnimator() {
if (mAnimator != null) {
return mAnimator;
}
mAnimator = new Animator(nGetAnimator(getNativeObject()));
return mAnimator;
}
void clearNativeObject() {
mNativeObject = 0;
}
@@ -61,4 +112,5 @@ public class FilamentAsset {
private static native void nGetEntities(long nativeAsset, int[] result);
private static native void nGetBoundingBox(long nativeAsset, float[] box);
private static native String nGetName(long nativeAsset, int entity);
private static native long nGetAnimator(long nativeAsset);
}

View File

@@ -24,42 +24,92 @@ import com.google.android.filament.IndirectLight;
import com.google.android.filament.Skybox;
import com.google.android.filament.Texture;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.nio.Buffer;
/**
* Utilities for consuming KTX files and producing Filament textures, IBLs, and sky boxes.
*
* <p>KTX is a simple container format that makes it easy to bundle miplevels and cubemap faces
* into a single file.</p>
*/
public class KtxLoader {
private static Constructor<Texture> sTextureConstructor;
private static Constructor<IndirectLight> sIndirectLightConstructor;
private static Constructor<Skybox> sSkyboxConstructor;
static {
try {
sTextureConstructor = Texture.class.getDeclaredConstructor(long.class);
sTextureConstructor.setAccessible(true);
sIndirectLightConstructor = IndirectLight.class.getDeclaredConstructor(long.class);
sIndirectLightConstructor.setAccessible(true);
sSkyboxConstructor = Skybox.class.getDeclaredConstructor(long.class);
sSkyboxConstructor.setAccessible(true);
} catch (NoSuchMethodException e) {
// Cannot happen
}
}
public static class Options {
public boolean srgb;
}
/**
* Consumes the content of a KTX file and produces a {@link Texture} object.
*
* @param engine Gets passed to the builder.
* @param buffer The content of the KTX File.
* @param options Loader options.
* @return The resulting Filament texture, or null on failure.
*/
@Nullable
public static Texture createTexture(@NonNull Engine engine, @NonNull Buffer buffer, @NonNull Options options) {
try {
long nativeEngine = (long) AssetLoader.sEngineGetNativeObject.invoke(engine);
long nativeEngine = engine.getNativeObject();
long nativeTexture = nCreateTexture(nativeEngine, buffer, buffer.remaining(), options.srgb);
return AssetLoader.sTextureConstructor.newInstance(nativeTexture);
return sTextureConstructor.newInstance(nativeTexture);
} catch (Exception e) {
return null;
}
}
/**
* Consumes the content of a KTX file and produces an {@link IndirectLight} object.
*
* @param engine Gets passed to the builder.
* @param buffer The content of the KTX File.
* @param options Loader options.
* @return The resulting Filament texture, or null on failure.
*/
@Nullable
public static IndirectLight createIndirectLight(@NonNull Engine engine, @NonNull Buffer buffer, @NonNull Options options) {
try {
long nativeEngine = (long) AssetLoader.sEngineGetNativeObject.invoke(engine);
long nativeEngine = engine.getNativeObject();
long nativeIndirectLight = nCreateIndirectLight(nativeEngine, buffer, buffer.remaining(), options.srgb);
return AssetLoader.sIndirectLightConstructor.newInstance(nativeIndirectLight);
return sIndirectLightConstructor.newInstance(nativeIndirectLight);
} catch (Exception e) {
return null;
}
}
/**
* Consumes the content of a KTX file and produces a {@link Skybox} object.
*
* @param engine Gets passed to the builder.
* @param buffer The content of the KTX File.
* @param options Loader options.
* @return The resulting Filament texture, or null on failure.
*/
@Nullable
public static Skybox createSkybox(@NonNull Engine engine, @NonNull Buffer buffer, @NonNull Options options) {
try {
long nativeEngine = (long) AssetLoader.sEngineGetNativeObject.invoke(engine);
long nativeEngine = engine.getNativeObject();
long nativeSkybox = nCreateSkybox(nativeEngine, buffer, buffer.remaining(), options.srgb);
return AssetLoader.sSkyboxConstructor.newInstance(nativeSkybox);
return sSkyboxConstructor.newInstance(nativeSkybox);
} catch (Exception e) {
return null;
}

View File

@@ -20,29 +20,28 @@ import com.google.android.filament.Engine;
import java.lang.reflect.Method;
/**
* Loads pre-generated ubershader materials that fulfill glTF requirements.
*
* <p>This class is used by {@link AssetLoader} to create Filament materials.
* Client applications do not need to call methods on it.</p>
*/
public class MaterialProvider {
private long mNativeObject;
private static Method sEngineGetNativeObject;
static {
try {
sEngineGetNativeObject = Engine.class.getDeclaredMethod("getNativeObject");
sEngineGetNativeObject.setAccessible(true);
} catch (NoSuchMethodException e) {
// Cannot happen
}
}
/**
* Constructs an ubershader loader using the supplied {@link Engine}.
*
* @param engine the engine used to create materials
*/
public MaterialProvider(Engine engine) {
try {
long nativeEngine = (long) sEngineGetNativeObject.invoke(engine);
mNativeObject = nCreateMaterialProvider(nativeEngine);
} catch (Exception e) {
// Ignored
}
long nativeEngine = engine.getNativeObject();
mNativeObject = nCreateMaterialProvider(nativeEngine);
}
/**
* Frees memory associated with the native material provider.
* */
public void destroy() {
nDestroyMaterialProvider(mNativeObject);
mNativeObject = 0;

View File

@@ -25,35 +25,66 @@ import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.nio.Buffer;
/**
* Uploads vertex buffers and textures to the GPU and computes tangents.
*
* <p>For a usage example, see the documentation for {@link AssetLoader}.</p>
*
* @see AssetLoader
* @see FilamentAsset
*/
public class ResourceLoader {
private final long mNativeObject;
private static Method sEngineGetNativeObject;
static {
try {
sEngineGetNativeObject = Engine.class.getDeclaredMethod("getNativeObject");
sEngineGetNativeObject.setAccessible(true);
} catch (NoSuchMethodException e) {
// Cannot happen
}
}
public ResourceLoader(@NonNull Engine engine) throws IllegalAccessException, InvocationTargetException {
long nativeEngine = (long) sEngineGetNativeObject.invoke(engine);
/**
* Constructs a resource loader tied to the given Filament engine.
*
* @param engine the engine that gets passed to all builder methods
* @throws IllegalAccessException
* @throws InvocationTargetException
*/
public ResourceLoader(@NonNull Engine engine) {
long nativeEngine = engine.getNativeObject();
mNativeObject = nCreateResourceLoader(nativeEngine);
}
/**
* Frees all memory associated with the native resource loader.
*/
public void destroy() {
nDestroyResourceLoader(mNativeObject);
}
/**
* Feeds the binary content of an external resource into the loader's URI cache.
*
* <p><code>ResourceLoader</code> does not know how to download external resources on its own
* (for example, external resources might come from a filesystem, a database, or the internet)
* so this method allows clients to download external resources and push them to the loader.</p>
*
* <p>When loading GLB files (as opposed to JSON-based glTF files), clients typically do not
* need to call this method.</p>
*
* @param uri the string path that matches an image URI or buffer URI in the glTF
* @param buffer the binary blob corresponding to the given URI
* @return self (for daisy chaining)
*/
@NonNull
public ResourceLoader addResourceData(@NonNull String url, @NonNull Buffer buffer) {
nAddResourceData(mNativeObject, url, buffer, buffer.remaining());
public ResourceLoader addResourceData(@NonNull String uri, @NonNull Buffer buffer) {
nAddResourceData(mNativeObject, uri, buffer, buffer.remaining());
return this;
}
/**
* Iterates through all external buffers and images and creates corresponding Filament objects
* (vertex buffers, textures, etc), which become owned by the asset.
*
* <p>This is the main entry point for <code>ResourceLoader</code>, and only needs to be called
* once.</p>
*
* @param asset the Filament asset that contains URI-based resources
* @return self (for daisy chaining)
*/
@NonNull
public ResourceLoader loadResources(@NonNull FilamentAsset asset) {
nLoadResources(mNativeObject, asset.getNativeObject());

View File

@@ -52,6 +52,12 @@ Demonstrates how to load glTF models and render to an offscreen buffer:
![glTF Bloom](../../docs/images/samples/sample_gltf_bloom.jpg)
### `hello-camera`
Demonstrates how to use `Stream` with Android's Camera2 API:
![Hello Camera](../../docs/images/samples/sample_hello_camera.jpg)
## Prerequisites
Before you start, make sure to read [Filament's README](../../README.md). You need to be able to

View File

@@ -17,31 +17,29 @@ task copyMesh(type: Copy) {
into file("src/main/assets/models")
}
task copySky(type: Copy) {
from file("../../../../samples/envs/venetian_crossroads/venetian_crossroads_ibl.ktx")
into file("src/main/assets/envs")
}
generateIbl {
group 'Filament'
description 'Generate IBL'
task copyIbl(type: Copy) {
from file("../../../../samples/envs/venetian_crossroads/venetian_crossroads_skybox.ktx")
into file("src/main/assets/envs")
cmgenArgs = "--format=ktx --size=256 --extract-blur=0.1 --deploy=src/main/assets/envs"
inputFile = file("../../../../third_party/environments/venetian_crossroads_2k.hdr")
outputDir = file("src/main/assets/envs")
}
preBuild.dependsOn compileMaterials
preBuild.dependsOn copyMesh
preBuild.dependsOn copySky
preBuild.dependsOn copyIbl
preBuild.dependsOn generateIbl
clean.doFirst {
delete "src/main/assets"
}
android {
compileSdkVersion 28
compileSdkVersion 29
defaultConfig {
applicationId "com.google.android.filament.gltf"
minSdkVersion 26
targetSdkVersion 28
minSdkVersion 19
targetSdkVersion 29
versionCode 1
versionName "1.0"
}

View File

@@ -43,10 +43,9 @@ data class Framebuffer(
class MainActivity : Activity() {
// Be sure to initialize not only Filament, but also gltfio (via AssetLoader)
// We are using the gltfio library, so init the AssetLoader rather than Filament.
companion object {
init {
Filament.init()
AssetLoader.init()
}
}
@@ -155,12 +154,14 @@ class MainActivity : Activity() {
// IndirectLight and SkyBox
// ------------------------
readUncompressedAsset("envs/venetian_crossroads_ibl.ktx").let {
val ibl = "venetian_crossroads_2k"
readUncompressedAsset("envs/$ibl/${ibl}_ibl.ktx").let {
primary.scene.indirectLight = KtxLoader.createIndirectLight(engine, it, KtxLoader.Options())
primary.scene.indirectLight!!.intensity = 50_000.0f
}
readUncompressedAsset("envs/venetian_crossroads_skybox.ktx").let {
readUncompressedAsset("envs/$ibl/${ibl}_skybox.ktx").let {
primary.scene.skybox = KtxLoader.createSkybox(engine, it, KtxLoader.Options())
}
@@ -207,7 +208,7 @@ class MainActivity : Activity() {
}
}
// Punctual Light Sources
// Light Sources
// ----------------------
light = EntityManager.get().create()
@@ -234,12 +235,19 @@ class MainActivity : Activity() {
tm.setTransform(tm.getInstance(filamentAsset.root),
floatArrayOf(
cos(v), 0.0f, -sin(v), 0.0f,
0.0f, 1.0f, 0.0f, 0.0f,
sin(v), 0.0f, cos(v), 0.0f,
0.0f, -1.7f, 0.0f, 1.0f
cos(v), 0.0f, -sin(v), 0.0f,
0.0f, 1.0f, 0.0f, 0.0f,
sin(v), 0.0f, cos(v), 0.0f,
0.0f, -1.7f, 0.0f, 1.0f
))
// The lucy asset does not have animation but we invoke the asset animator for demonstration purposes.
if (filamentAsset.animator.animationCount > 0) {
val elapsedTimeInSeconds = a.currentPlayTime.toFloat() / 1000.0f
filamentAsset.animator.applyAnimation(0, elapsedTimeInSeconds)
filamentAsset.animator.updateBoneMatrices()
}
}
animator.start()
}
@@ -424,13 +432,14 @@ class MainActivity : Activity() {
override fun onDestroy() {
super.onDestroy()
// Stop the animation and any pending frame
choreographer.removeFrameCallback(frameScheduler)
animator.cancel()
// Always detach the surface before destroying the engine
uiHelper.detach()
// This ensures that all the commands we've sent to Filament have
// been processed before we attempt to destroy anything
Fence.waitAndDestroy(engine.createFence(Fence.Type.SOFT), Fence.Mode.FLUSH)
assetLoader.destroyAsset(filamentAsset)
assetLoader.destroy()

View File

@@ -7,7 +7,7 @@ buildscript {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:3.4.1'
classpath 'com.android.tools.build:gradle:3.5.0'
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
// NOTE: Do not place your application dependencies here; they belong

View File

@@ -3,4 +3,4 @@ distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-5.1.1-all.zip
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip

12
android/samples/hello-camera/.gitignore vendored Normal file
View File

@@ -0,0 +1,12 @@
*.iml
.gradle
/local.properties
/.idea/workspace.xml
/.idea/libraries
/.idea/caches
/.idea/gradle.xml
.DS_Store
/build
/captures
/app/src/main/assets/materials/*.filamat
.externalNativeBuild

View File

@@ -0,0 +1 @@
/build

View File

@@ -0,0 +1,101 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.
*/
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply from: '../../../build/filament-tasks.gradle'
compileMaterials {
group 'Filament'
description 'Compile materials'
inputDir = file("src/main/materials")
outputDir = file("src/main/assets/materials")
}
preBuild.dependsOn compileMaterials
clean.doFirst {
delete "src/main/assets"
}
android {
compileSdkVersion 29
defaultConfig {
applicationId "com.google.android.filament.hellocam"
minSdkVersion 23 // 21 is required for CameraDevice.StateCallback, 23 is required for shouldShowRequestPermissionRationale
targetSdkVersion 29
versionCode 1
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
// Filament comes with native code, the following declarations
// can be used to generate architecture specific APKs
flavorDimensions 'cpuArch'
productFlavors {
arm8 {
dimension 'cpuArch'
ndk {
abiFilters 'arm64-v8a'
}
}
arm7 {
dimension 'cpuArch'
ndk {
abiFilters 'armeabi-v7a'
}
}
x86_64 {
dimension 'cpuArch'
ndk {
abiFilters 'x86_64'
}
}
x86 {
dimension 'cpuArch'
ndk {
abiFilters 'x86'
}
}
universal {
dimension 'cpuArch'
}
}
// We use the .filamat extension for materials compiled with matc
// Telling aapt to not compress them allows to load them efficiently
aaptOptions {
noCompress 'filamat'
}
}
dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar'])
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
// Depend on Filament
implementation 'com.google.android.filament:filament-android'
implementation 'com.android.support:support-compat:28.0.0'
}

View File

@@ -0,0 +1,21 @@
# Add project specific ProGuard rules here.
# You can control the set of applied configuration files using the
# proguardFiles setting in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
# Uncomment this to preserve the line number information for
# debugging stack traces.
#-keepattributes SourceFile,LineNumberTable
# If you keep the line number information, uncomment this to
# hide the original source file name.
#-renamesourcefileattribute SourceFile

View File

@@ -0,0 +1,43 @@
<?xml version="1.0" encoding="utf-8"?>
<!--
Copyright (C) 2019 The Android Open Source Project
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.
-->
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.google.android.filament.hellocam">
<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application>
</manifest>

View File

@@ -0,0 +1,219 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.google.android.filament.hellocam
import android.app.Activity
import android.content.Context
import android.content.pm.PackageManager
import android.graphics.SurfaceTexture
import android.hardware.camera2.*
import android.os.Handler
import android.os.HandlerThread
import android.support.v4.content.ContextCompat
import android.util.Log
import android.util.Size
import android.view.Surface
import android.Manifest
import android.opengl.Matrix
import android.view.Display
import com.google.android.filament.*
import java.util.concurrent.Semaphore
import java.util.concurrent.TimeUnit
/**
* Toy class that handles all interaction with the Android camera2 API.
* Sets the "textureTransform" and "videoTexture" parameters on the given Filament material.
*/
class CameraHelper(val activity: Activity, private val filamentEngine: Engine, private val filamentMaterial: MaterialInstance, private val display: Display) {
private lateinit var cameraId: String
private lateinit var captureRequest: CaptureRequest
private val cameraOpenCloseLock = Semaphore(1)
private var backgroundHandler: Handler? = null
private var backgroundThread: HandlerThread? = null
private var cameraDevice: CameraDevice? = null
private var captureSession: CameraCaptureSession? = null
private var resolution = Size(640, 480)
private var filamentTexture: Texture? = null
private var filamentStream: Stream? = null
private var surfaceTexture: SurfaceTexture? = null
private val cameraCallback = object : CameraDevice.StateCallback() {
override fun onOpened(cameraDevice: CameraDevice) {
cameraOpenCloseLock.release()
this@CameraHelper.cameraDevice = cameraDevice
createCaptureSession()
}
override fun onDisconnected(cameraDevice: CameraDevice) {
cameraOpenCloseLock.release()
cameraDevice.close()
this@CameraHelper.cameraDevice = null
}
override fun onError(cameraDevice: CameraDevice, error: Int) {
onDisconnected(cameraDevice)
this@CameraHelper.activity.finish()
}
}
/**
* Finds the front-facing Android camera, requests permission, and sets up a listener that will
* start a capture session as soon as the camera is ready.
*/
fun openCamera() {
val manager = activity.getSystemService(Context.CAMERA_SERVICE) as CameraManager
try {
for (cameraId in manager.cameraIdList) {
val characteristics = manager.getCameraCharacteristics(cameraId)
val cameraDirection = characteristics.get(CameraCharacteristics.LENS_FACING)
if (cameraDirection != null && cameraDirection == CameraCharacteristics.LENS_FACING_FRONT) {
continue
}
this.cameraId = cameraId
Log.i(kLogTag, "Selected camera $cameraId.")
val map = characteristics.get(CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP) ?: continue
resolution = map.getOutputSizes(SurfaceTexture::class.java)[0]
Log.i(kLogTag, "Highest resolution is $resolution.")
}
} catch (e: CameraAccessException) {
Log.e(kLogTag, e.toString())
} catch (e: NullPointerException) {
Log.e(kLogTag, "Camera2 API is not supported on this device.")
}
val permission = ContextCompat.checkSelfPermission(this.activity, Manifest.permission.CAMERA)
if (permission != PackageManager.PERMISSION_GRANTED) {
activity.requestPermissions(arrayOf(Manifest.permission.CAMERA), kRequestCameraPermission)
return
}
if (!cameraOpenCloseLock.tryAcquire(2500, TimeUnit.MILLISECONDS)) {
throw RuntimeException("Time out waiting to lock camera opening.")
}
manager.openCamera(cameraId, cameraCallback, backgroundHandler)
}
fun onResume() {
backgroundThread = HandlerThread("CameraBackground").also { it.start() }
backgroundHandler = Handler(backgroundThread?.looper!!)
}
fun onPause() {
backgroundThread?.quitSafely()
try {
backgroundThread?.join()
backgroundThread = null
backgroundHandler = null
} catch (e: InterruptedException) {
Log.e(kLogTag, e.toString())
}
}
fun onRequestPermissionsResult(requestCode: Int, grantResults: IntArray): Boolean {
if (requestCode == kRequestCameraPermission) {
if (grantResults.size != 1 || grantResults[0] != PackageManager.PERMISSION_GRANTED) {
Log.e(kLogTag, "Unable to obtain camera position.")
}
return true
}
return false
}
private fun createCaptureSession() {
if (surfaceTexture != null) {
surfaceTexture!!.release()
filamentEngine.destroyStream(filamentStream!!)
}
// Create the Android surface that will hold the camera image.
surfaceTexture = SurfaceTexture(0)
surfaceTexture!!.setDefaultBufferSize(resolution.width, resolution.height)
surfaceTexture!!.detachFromGLContext()
val surface = Surface(surfaceTexture)
// [Re]create the Filament Stream object that gets bound to the Texture.
filamentStream = Stream.Builder()
.stream(surfaceTexture!!)
.build(filamentEngine)
// Create the Filament Texture object if we haven't done so already.
if (filamentTexture == null) {
filamentTexture = Texture.Builder()
.sampler(Texture.Sampler.SAMPLER_EXTERNAL)
.format(Texture.InternalFormat.RGB8)
.build(filamentEngine)
}
// We are texturing a front-facing square shape so we need to generate a matrix that transforms (u, v, 0, 1)
// into a new UV coordinate according to the screen rotation and the aspect ratio of the camera image.
val aspectRatio = resolution.width.toFloat() / resolution.height.toFloat()
val textureTransform = FloatArray(16)
Matrix.setIdentityM(textureTransform, 0)
when (display.rotation) {
Surface.ROTATION_0 -> {
Matrix.translateM(textureTransform, 0, 1.0f, 0.0f, 0.0f)
Matrix.rotateM(textureTransform, 0, 90.0f, 0.0f, 0.0f, 1.0f)
Matrix.translateM(textureTransform, 0, 1.0f, 0.0f, 0.0f)
Matrix.scaleM(textureTransform, 0, -1.0f, 1.0f / aspectRatio, 1.0f)
}
Surface.ROTATION_90 -> {
Matrix.translateM(textureTransform, 0, 1.0f, 1.0f, 0.0f)
Matrix.rotateM(textureTransform, 0, 180.0f, 0.0f, 0.0f, 1.0f)
Matrix.translateM(textureTransform, 0, 1.0f, 0.0f, 0.0f)
Matrix.scaleM(textureTransform, 0, -1.0f / aspectRatio, 1.0f, 1.0f)
}
Surface.ROTATION_270 -> {
Matrix.translateM(textureTransform, 0, 1.0f, 0.0f, 0.0f)
Matrix.scaleM(textureTransform, 0, -1.0f / aspectRatio, 1.0f, 1.0f)
}
}
// Connect the Stream to the Texture and the Texture to the MaterialInstance.
val sampler = TextureSampler(TextureSampler.MinFilter.LINEAR, TextureSampler.MagFilter.LINEAR, TextureSampler.WrapMode.CLAMP_TO_EDGE)
filamentTexture!!.setExternalStream(filamentEngine, filamentStream!!)
filamentMaterial.setParameter("videoTexture", filamentTexture!!, sampler)
filamentMaterial.setParameter("textureTransform", MaterialInstance.FloatElement.MAT4, textureTransform, 0, 1)
// Start the capture session. You could also use TEMPLATE_PREVIEW here.
val captureRequestBuilder = cameraDevice!!.createCaptureRequest(CameraDevice.TEMPLATE_RECORD)
captureRequestBuilder.addTarget(surface)
cameraDevice?.createCaptureSession(listOf(surface),
object : CameraCaptureSession.StateCallback() {
override fun onConfigured(cameraCaptureSession: CameraCaptureSession) {
if (cameraDevice == null) return
captureSession = cameraCaptureSession
captureRequestBuilder.set(CaptureRequest.CONTROL_AF_MODE, CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE)
captureRequest = captureRequestBuilder.build()
captureSession!!.setRepeatingRequest(captureRequest, null, backgroundHandler)
Log.i(kLogTag, "Created CaptureRequest.")
}
override fun onConfigureFailed(session: CameraCaptureSession) {
Log.e(kLogTag, "onConfigureFailed")
}
}, null)
}
companion object {
private const val kLogTag = "CameraHelper"
private const val kRequestCameraPermission = 1
}
}

View File

@@ -0,0 +1,417 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.google.android.filament.hellocam
import android.animation.ValueAnimator
import android.app.Activity
import android.opengl.Matrix
import android.os.Bundle
import android.support.v4.app.ActivityCompat
import android.util.Log
import android.view.Choreographer
import android.view.Display
import android.view.Surface
import android.view.SurfaceView
import android.view.animation.LinearInterpolator
import com.google.android.filament.*
import com.google.android.filament.RenderableManager.*
import com.google.android.filament.VertexBuffer.*
import com.google.android.filament.android.UiHelper
import java.nio.ByteBuffer
import java.nio.ByteOrder
import java.nio.channels.Channels
import kotlin.math.*
class MainActivity : Activity(), ActivityCompat.OnRequestPermissionsResultCallback {
companion object {
init {
Filament.init()
}
}
private lateinit var surfaceView: SurfaceView
private lateinit var uiHelper: UiHelper
private lateinit var choreographer: Choreographer
private lateinit var engine: Engine
private lateinit var renderer: Renderer
private lateinit var scene: Scene
private lateinit var view: View
// This helper wraps the Android camera2 API and connects it to a Filament material.
private lateinit var cameraHelper: CameraHelper
// This is the Filament camera, not the phone camera. :)
private lateinit var camera: Camera
// Other Filament objects:
private lateinit var material: Material
private lateinit var materialInstance: MaterialInstance
private lateinit var vertexBuffer: VertexBuffer
private lateinit var indexBuffer: IndexBuffer
// Filament entity representing a renderable object
@Entity private var renderable = 0
@Entity private var light = 0
// A swap chain is Filament's representation of a surface
private var swapChain: SwapChain? = null
// Performs the rendering and schedules new frames
private val frameScheduler = FrameCallback()
private val animator = ValueAnimator.ofFloat(0.0f, 50.0f)
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
surfaceView = SurfaceView(this)
setContentView(surfaceView)
choreographer = Choreographer.getInstance()
setupSurfaceView()
setupFilament()
setupView()
setupScene()
cameraHelper = CameraHelper(this, engine, materialInstance, windowManager.defaultDisplay)
cameraHelper.openCamera()
}
private fun setupSurfaceView() {
uiHelper = UiHelper(UiHelper.ContextErrorPolicy.DONT_CHECK)
uiHelper.renderCallback = SurfaceCallback()
uiHelper.attachTo(surfaceView)
}
private fun setupFilament() {
engine = Engine.create()
renderer = engine.createRenderer()
scene = engine.createScene()
view = engine.createView()
camera = engine.createCamera()
}
private fun setupView() {
view.setClearColor(0.035f, 0.035f, 0.035f, 1.0f)
view.camera = camera
view.scene = scene
}
private fun setupScene() {
loadMaterial()
setupMaterial()
createMesh()
// To create a renderable we first create a generic entity
renderable = EntityManager.get().create()
// We then create a renderable component on that entity
// A renderable is made of several primitives; in this case we declare only 1
// If we wanted each face of the cube to have a different material, we could
// declare 6 primitives (1 per face) and give each of them a different material
// instance, setup with different parameters
RenderableManager.Builder(1)
// Overall bounding box of the renderable
.boundingBox(Box(0.0f, 0.0f, 0.0f, 1.0f, 1.0f, 1.0f))
// Sets the mesh data of the first primitive, 6 faces of 6 indices each
.geometry(0, PrimitiveType.TRIANGLES, vertexBuffer, indexBuffer, 0, 6 * 6)
// Sets the material of the first primitive
.material(0, materialInstance)
.build(engine, renderable)
// Add the entity to the scene to render it
scene.addEntity(renderable)
// We now need a light, let's create a directional light
light = EntityManager.get().create()
// Create a color from a temperature (5,500K)
val (r, g, b) = Colors.cct(5_500.0f)
LightManager.Builder(LightManager.Type.DIRECTIONAL)
.color(r, g, b)
// Intensity of the sun in lux on a clear day
.intensity(110_000.0f)
// The direction is normalized on our behalf
.direction(0.0f, -0.5f, -1.0f)
.castShadows(true)
.build(engine, light)
// Add the entity to the scene to light it
scene.addEntity(light)
// Set the exposure on the camera, this exposure follows the sunny f/16 rule
// Since we've defined a light that has the same intensity as the sun, it
// guarantees a proper exposure
camera.setExposure(16.0f, 1.0f / 125.0f, 100.0f)
// Move the camera back to see the object
camera.lookAt(0.0, 0.0, 6.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0)
startAnimation()
}
private fun loadMaterial() {
readUncompressedAsset("materials/lit.filamat").let {
material = Material.Builder().payload(it, it.remaining()).build(engine)
}
}
private fun setupMaterial() {
materialInstance = material.createInstance()
materialInstance.setParameter("baseColor", Colors.RgbType.SRGB, 1.0f, 0.85f, 0.57f)
materialInstance.setParameter("roughness", 0.3f)
}
private fun createMesh() {
val floatSize = 4
val shortSize = 2
// A vertex is a position + a tangent frame:
// 3 floats for XYZ position, 4 floats for normal+tangents (quaternion)
val vertexSize = 3 * floatSize + 4 * floatSize
// Define a vertex and a function to put a vertex in a ByteBuffer
@Suppress("ArrayInDataClass")
data class Vertex(val x: Float, val y: Float, val z: Float, val tangents: FloatArray)
fun ByteBuffer.put(v: Vertex): ByteBuffer {
putFloat(v.x)
putFloat(v.y)
putFloat(v.z)
v.tangents.forEach { putFloat(it) }
return this
}
// 6 faces, 4 vertices per face
val vertexCount = 6 * 4
// Create tangent frames, one per face
val tfPX = FloatArray(4)
val tfNX = FloatArray(4)
val tfPY = FloatArray(4)
val tfNY = FloatArray(4)
val tfPZ = FloatArray(4)
val tfNZ = FloatArray(4)
MathUtils.packTangentFrame( 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, -1.0f, 1.0f, 0.0f, 0.0f, tfPX)
MathUtils.packTangentFrame( 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, -1.0f, -1.0f, 0.0f, 0.0f, tfNX)
MathUtils.packTangentFrame(-1.0f, 0.0f, 0.0f, 0.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f, tfPY)
MathUtils.packTangentFrame(-1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, -1.0f, 0.0f, tfNY)
MathUtils.packTangentFrame( 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f, tfPZ)
MathUtils.packTangentFrame( 0.0f, -1.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, -1.0f, tfNZ)
val vertexData = ByteBuffer.allocate(vertexCount * vertexSize)
// It is important to respect the native byte order
.order(ByteOrder.nativeOrder())
// Face -Z
.put(Vertex(-1.5f, -1.5f, -1.0f, tfNZ))
.put(Vertex(-1.5f, 1.5f, -1.0f, tfNZ))
.put(Vertex( 1.5f, 1.5f, -1.0f, tfNZ))
.put(Vertex( 1.5f, -1.5f, -1.0f, tfNZ))
// Face +X
.put(Vertex( 1.5f, -1.5f, -1.0f, tfPX))
.put(Vertex( 1.5f, 1.5f, -1.0f, tfPX))
.put(Vertex( 1.0f, 1.0f, 1.0f, tfPX))
.put(Vertex( 1.0f, -1.0f, 1.0f, tfPX))
// Face +Z
.put(Vertex(-1.0f, -1.0f, 1.0f, tfPZ))
.put(Vertex( 1.0f, -1.0f, 1.0f, tfPZ))
.put(Vertex( 1.0f, 1.0f, 1.0f, tfPZ))
.put(Vertex(-1.0f, 1.0f, 1.0f, tfPZ))
// Face -X
.put(Vertex(-1.0f, -1.0f, 1.0f, tfNX))
.put(Vertex(-1.0f, 1.0f, 1.0f, tfNX))
.put(Vertex(-1.5f, 1.5f, -1.0f, tfNX))
.put(Vertex(-1.5f, -1.5f, -1.0f, tfNX))
// Face -Y
.put(Vertex(-1.0f, -1.0f, 1.0f, tfNY))
.put(Vertex(-1.5f, -1.5f, -1.0f, tfNY))
.put(Vertex( 1.5f, -1.5f, -1.0f, tfNY))
.put(Vertex( 1.0f, -1.0f, 1.0f, tfNY))
// Face +Y
.put(Vertex(-1.5f, 1.5f, -1.0f, tfPY))
.put(Vertex(-1.0f, 1.0f, 1.0f, tfPY))
.put(Vertex( 1.0f, 1.0f, 1.0f, tfPY))
.put(Vertex( 1.5f, 1.5f, -1.0f, tfPY))
// Make sure the cursor is pointing in the right place in the byte buffer
.flip()
// Declare the layout of our mesh
vertexBuffer = VertexBuffer.Builder()
.bufferCount(1)
.vertexCount(vertexCount)
// Because we interleave position and color data we must specify offset and stride
// We could use de-interleaved data by declaring two buffers and giving each
// attribute a different buffer index
.attribute(VertexAttribute.POSITION, 0, AttributeType.FLOAT3, 0, vertexSize)
.attribute(VertexAttribute.TANGENTS, 0, AttributeType.FLOAT4, 3 * floatSize, vertexSize)
.build(engine)
// Feed the vertex data to the mesh
// We only set 1 buffer because the data is interleaved
vertexBuffer.setBufferAt(engine, 0, vertexData)
// Create the indices
val indexData = ByteBuffer.allocate(6 * 2 * 3 * shortSize)
.order(ByteOrder.nativeOrder())
repeat(6) {
val i = (it * 4).toShort()
indexData
.putShort(i).putShort((i + 1).toShort()).putShort((i + 2).toShort())
.putShort(i).putShort((i + 2).toShort()).putShort((i + 3).toShort())
}
indexData.flip()
// 6 faces, 2 triangles per face,
indexBuffer = IndexBuffer.Builder()
.indexCount(vertexCount * 2)
.bufferType(IndexBuffer.Builder.IndexType.USHORT)
.build(engine)
indexBuffer.setBuffer(engine, indexData)
}
private fun startAnimation() {
// Animate the triangle
animator.interpolator = LinearInterpolator()
animator.duration = 6000
animator.repeatMode = ValueAnimator.RESTART
animator.repeatCount = ValueAnimator.INFINITE
animator.addUpdateListener(object : ValueAnimator.AnimatorUpdateListener {
val transformMatrix = FloatArray(16)
override fun onAnimationUpdate(animator: ValueAnimator) {
val t = animator.animatedValue as Float
val radians = sin(t) * 3.0f * PI.toFloat()
Matrix.setRotateM(transformMatrix, 0, radians, 0.0f, 1.0f, 0.0f)
val tcm = engine.transformManager
tcm.setTransform(tcm.getInstance(renderable), transformMatrix)
}
})
animator.start()
}
override fun onResume() {
super.onResume()
choreographer.postFrameCallback(frameScheduler)
animator.start()
cameraHelper.onResume()
}
override fun onPause() {
super.onPause()
choreographer.removeFrameCallback(frameScheduler)
animator.cancel()
cameraHelper.onPause()
}
override fun onDestroy() {
super.onDestroy()
// Stop the animation and any pending frame
choreographer.removeFrameCallback(frameScheduler)
animator.cancel()
// Always detach the surface before destroying the engine
uiHelper.detach()
// Cleanup all resources
engine.destroyEntity(light)
engine.destroyEntity(renderable)
engine.destroyRenderer(renderer)
engine.destroyVertexBuffer(vertexBuffer)
engine.destroyIndexBuffer(indexBuffer)
engine.destroyMaterialInstance(materialInstance)
engine.destroyMaterial(material)
engine.destroyView(view)
engine.destroyScene(scene)
engine.destroyCamera(camera)
// Engine.destroyEntity() destroys Filament related resources only
// (components), not the entity itself
val entityManager = EntityManager.get()
entityManager.destroy(light)
entityManager.destroy(renderable)
// Destroying the engine will free up any resource you may have forgotten
// to destroy, but it's recommended to do the cleanup properly
engine.destroy()
}
inner class FrameCallback : Choreographer.FrameCallback {
override fun doFrame(frameTimeNanos: Long) {
// Schedule the next frame
choreographer.postFrameCallback(this)
// This check guarantees that we have a swap chain
if (uiHelper.isReadyToRender) {
// If beginFrame() returns false you should skip the frame
// This means you are sending frames too quickly to the GPU
if (renderer.beginFrame(swapChain!!)) {
renderer.render(view)
renderer.endFrame()
}
}
}
}
inner class SurfaceCallback : UiHelper.RendererCallback {
override fun onNativeWindowChanged(surface: Surface) {
swapChain?.let { engine.destroySwapChain(it) }
swapChain = engine.createSwapChain(surface)
}
override fun onDetachedFromSurface() {
swapChain?.let {
engine.destroySwapChain(it)
// Required to ensure we don't return before Filament is done executing the
// destroySwapChain command, otherwise Android might destroy the Surface
// too early
engine.flushAndWait()
swapChain = null
}
}
override fun onResized(width: Int, height: Int) {
val aspect = width.toDouble() / height.toDouble()
camera.setProjection(45.0, aspect, 0.1, 20.0, Camera.Fov.VERTICAL)
view.viewport = Viewport(0, 0, width, height)
}
}
private fun readUncompressedAsset(@Suppress("SameParameterValue") assetName: String): ByteBuffer {
assets.openFd(assetName).use { fd ->
val input = fd.createInputStream()
val dst = ByteBuffer.allocate(fd.length.toInt())
val src = Channels.newChannel(input)
src.read(dst)
src.close()
return dst.apply { rewind() }
}
}
override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<String>, grantResults: IntArray) {
if (!cameraHelper.onRequestPermissionsResult(requestCode, grantResults)) {
this.onRequestPermissionsResult(requestCode, permissions, grantResults)
}
}
}

View File

@@ -0,0 +1,73 @@
// Simple lit material that defines 3 parameters:
// - baseColor
// - roughness
// - metallic
//
// These parameters can be used by the application to change the appearance of the material.
//
// This source material must be compiled to a binary material using the matc tool.
// The command used to compile this material is:
// matc -p mobile -a opengl -o app/src/main/assets/lit.filamat app/src/materials/lit.mat
//
// See build.gradle for an example of how to compile materials automatically
// Please refer to the documentation for more information about matc and the materials system.
material {
name : lit,
// Dynamic lighting is enabled on this material
shadingModel : lit,
// We don't need to declare a "requires" array, lit materials
// always requires the "tangents" vertex attribute (the normal
// is required for lighting, tangent/bitangent for normal mapping
// and anisotropy)
// Custom vertex shader outputs
variables : [
uv
],
// List of parameters exposed by this material
parameters : [
// The color must be passed in linear space, not sRGB
{
type : float3,
name : baseColor
},
{
type : float,
name : roughness
},
{
type : samplerExternal,
name : videoTexture
},
{
type : mat4,
name : textureTransform
}
],
}
vertex {
void materialVertex(inout MaterialVertexInputs material) {
material.uv = 0.5 * (getPosition() + vec4(1));
}
}
fragment {
void material(inout MaterialInputs material) {
prepareMaterial(material);
material.roughness = materialParams.roughness;
material.metallic = 0.0;
// Apply the video stream to the +Z face on the cube.
if (variable_uv.z >= 1.0) {
vec2 uv = (materialParams.textureTransform * vec4(variable_uv.xy, 0, 1)).xy;
material.baseColor.rgb = inverseTonemapSRGB(texture(materialParams_videoTexture, uv).rgb);
} else {
material.baseColor.rgb = materialParams.baseColor;
}
}
}

View File

@@ -0,0 +1,34 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:aapt="http://schemas.android.com/aapt"
android:width="108dp"
android:height="108dp"
android:viewportHeight="108"
android:viewportWidth="108">
<path
android:fillType="evenOdd"
android:pathData="M32,64C32,64 38.39,52.99 44.13,50.95C51.37,48.37 70.14,49.57 70.14,49.57L108.26,87.69L108,109.01L75.97,107.97L32,64Z"
android:strokeColor="#00000000"
android:strokeWidth="1">
<aapt:attr name="android:fillColor">
<gradient
android:endX="78.5885"
android:endY="90.9159"
android:startX="48.7653"
android:startY="61.0927"
android:type="linear">
<item
android:color="#44000000"
android:offset="0.0"/>
<item
android:color="#00000000"
android:offset="1.0"/>
</gradient>
</aapt:attr>
</path>
<path
android:fillColor="#FFFFFF"
android:fillType="nonZero"
android:pathData="M66.94,46.02L66.94,46.02C72.44,50.07 76,56.61 76,64L32,64C32,56.61 35.56,50.11 40.98,46.06L36.18,41.19C35.45,40.45 35.45,39.3 36.18,38.56C36.91,37.81 38.05,37.81 38.78,38.56L44.25,44.05C47.18,42.57 50.48,41.71 54,41.71C57.48,41.71 60.78,42.57 63.68,44.05L69.11,38.56C69.84,37.81 70.98,37.81 71.71,38.56C72.44,39.3 72.44,40.45 71.71,41.19L66.94,46.02ZM62.94,56.92C64.08,56.92 65,56.01 65,54.88C65,53.76 64.08,52.85 62.94,52.85C61.8,52.85 60.88,53.76 60.88,54.88C60.88,56.01 61.8,56.92 62.94,56.92ZM45.06,56.92C46.2,56.92 47.13,56.01 47.13,54.88C47.13,53.76 46.2,52.85 45.06,52.85C43.92,52.85 43,53.76 43,54.88C43,56.01 43.92,56.92 45.06,56.92Z"
android:strokeColor="#00000000"
android:strokeWidth="1"/>
</vector>

View File

@@ -0,0 +1,171 @@
<?xml version="1.0" encoding="utf-8"?>
<vector
xmlns:android="http://schemas.android.com/apk/res/android"
android:width="108dp"
android:height="108dp"
android:viewportHeight="108"
android:viewportWidth="108">
<path
android:fillColor="#26A69A"
android:pathData="M0,0h108v108h-108z"/>
<path
android:fillColor="#00000000"
android:pathData="M9,0L9,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,0L19,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M29,0L29,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M39,0L39,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M49,0L49,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M59,0L59,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M69,0L69,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M79,0L79,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M89,0L89,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M99,0L99,108"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,9L108,9"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,19L108,19"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,29L108,29"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,39L108,39"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,49L108,49"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,59L108,59"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,69L108,69"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,79L108,79"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,89L108,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M0,99L108,99"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,29L89,29"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,39L89,39"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,49L89,49"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,59L89,59"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,69L89,69"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M19,79L89,79"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M29,19L29,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M39,19L39,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M49,19L49,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M59,19L59,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M69,19L69,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
<path
android:fillColor="#00000000"
android:pathData="M79,19L79,89"
android:strokeColor="#33FFFFFF"
android:strokeWidth="0.8"/>
</vector>

View File

@@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background"/>
<foreground android:drawable="@drawable/ic_launcher_foreground"/>
</adaptive-icon>

View File

@@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background"/>
<foreground android:drawable="@drawable/ic_launcher_foreground"/>
</adaptive-icon>

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 15 KiB

View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="colorPrimary">#3F51B5</color>
<color name="colorPrimaryDark">#303F9F</color>
<color name="colorAccent">#FF4081</color>
</resources>

View File

@@ -0,0 +1,3 @@
<resources>
<string name="app_name">Hello Camera</string>
</resources>

View File

@@ -0,0 +1,8 @@
<resources>
<!-- Base application theme. -->
<style name="AppTheme" parent="android:Theme.Material.Light.DarkActionBar">
<!-- Customize your theme here. -->
</style>
</resources>

View File

@@ -0,0 +1,41 @@
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.
*/
buildscript {
ext.kotlin_version = '1.3.50'
repositories {
google()
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:3.5.0'
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
// NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files
}
}
allprojects {
repositories {
google()
jcenter()
}
}
task clean(type: Delete) {
delete rootProject.buildDir
}

View File

@@ -0,0 +1,13 @@
# Project-wide Gradle settings.
# IDE (e.g. Android Studio) users:
# Gradle settings configured through the IDE *will override*
# any settings specified in this file.
# For more details on how to configure your build environment visit
# http://www.gradle.org/docs/current/userguide/build_environment.html
# Specifies the JVM arguments used for the daemon process.
# The setting is particularly useful for tweaking memory settings.
org.gradle.jvmargs=-Xmx1536m
# When configured, Gradle will run in incubating parallel mode.
# This option should only be used with decoupled projects. More details, visit
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
# org.gradle.parallel=true

Binary file not shown.

View File

@@ -0,0 +1,6 @@
#Mon Jan 14 11:08:15 PST 2019
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-5.4.1-all.zip

172
android/samples/hello-camera/gradlew vendored Executable file
View File

@@ -0,0 +1,172 @@
#!/usr/bin/env sh
##############################################################################
##
## Gradle start up script for UN*X
##
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
PRG="$0"
# Need this for relative symlinks.
while [ -h "$PRG" ] ; do
ls=`ls -ld "$PRG"`
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
PRG="$link"
else
PRG=`dirname "$PRG"`"/$link"
fi
done
SAVED="`pwd`"
cd "`dirname \"$PRG\"`/" >/dev/null
APP_HOME="`pwd -P`"
cd "$SAVED" >/dev/null
APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"`
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS=""
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD="maximum"
warn () {
echo "$*"
}
die () {
echo
echo "$*"
echo
exit 1
}
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "`uname`" in
CYGWIN* )
cygwin=true
;;
Darwin* )
darwin=true
;;
MINGW* )
msys=true
;;
NONSTOP* )
nonstop=true
;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD="$JAVA_HOME/jre/sh/java"
else
JAVACMD="$JAVA_HOME/bin/java"
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD="java"
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
# Increase the maximum file descriptors if we can.
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
MAX_FD_LIMIT=`ulimit -H -n`
if [ $? -eq 0 ] ; then
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
MAX_FD="$MAX_FD_LIMIT"
fi
ulimit -n $MAX_FD
if [ $? -ne 0 ] ; then
warn "Could not set maximum file descriptor limit: $MAX_FD"
fi
else
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
fi
fi
# For Darwin, add options to specify how the application appears in the dock
if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi
# For Cygwin, switch paths to Windows format before running java
if $cygwin ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
JAVACMD=`cygpath --unix "$JAVACMD"`
# We build the pattern for arguments to be converted via cygpath
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
SEP=""
for dir in $ROOTDIRSRAW ; do
ROOTDIRS="$ROOTDIRS$SEP$dir"
SEP="|"
done
OURCYGPATTERN="(^($ROOTDIRS))"
# Add a user-defined pattern to the cygpath arguments
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
fi
# Now convert the arguments - kludge to limit ourselves to /bin/sh
i=0
for arg in "$@" ; do
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
else
eval `echo args$i`="\"$arg\""
fi
i=$((i+1))
done
case $i in
(0) set -- ;;
(1) set -- "$args0" ;;
(2) set -- "$args0" "$args1" ;;
(3) set -- "$args0" "$args1" "$args2" ;;
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac
fi
# Escape application args
save () {
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
echo " "
}
APP_ARGS=$(save "$@")
# Collect all arguments for the java command, following the shell quoting and substitution rules
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
cd "$(dirname "$0")"
fi
exec "$JAVACMD" "$@"

Some files were not shown because too many files have changed in this diff Show More