Skip to content

Commit

Permalink
Auto merge of #308 - ceyusa:render-android, r=ferjm
Browse files Browse the repository at this point in the history
Render android

Add GL render for android target, which is a unix type with only EGL.
  • Loading branch information
bors-servo authored Sep 16, 2019
2 parents f9aed4e + 9add8b7 commit d6f799e
Show file tree
Hide file tree
Showing 6 changed files with 310 additions and 9 deletions.
13 changes: 13 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 3 additions & 0 deletions backends/gstreamer/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -87,3 +87,6 @@ path = "render"

[target.'cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))'.dependencies]
servo-media-gstreamer-render-unix = { path = "render-unix", features = ["gl-egl", "gl-x11"] }

[target.'cfg(any(target_os = "android"))'.dependencies]
servo-media-gstreamer-render-android = { path = "render-android" }
27 changes: 27 additions & 0 deletions backends/gstreamer/render-android/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
[package]
name = "servo-media-gstreamer-render-android"
version = "0.1.0"
authors = ["The Servo Project Developers"]

[lib]
name = "servo_media_gstreamer_render_android"
path = "lib.rs"

[dependencies.glib]
version = "0.8"

[dependencies.gstreamer]
version = "0.14"

[dependencies.gstreamer-gl]
version = "0.14"
features = ["egl"]

[dependencies.gstreamer-video]
version = "0.14"

[dependencies.servo-media-player]
path = "../../../player"

[dependencies.servo-media-gstreamer-render]
path = "../render"
253 changes: 253 additions & 0 deletions backends/gstreamer/render-android/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,253 @@
//! `RenderAndroid` is a `Render` implementation for Android
//! platform. It only implements an OpenGLES mechanism.
//!
//! Internally it uses GStreamer's *glsinkbin* element as *videosink*
//! wrapping the *appsink* from the Player. And the shared frames are
//! mapped as texture IDs.
extern crate gstreamer as gst;
extern crate gstreamer_gl as gst_gl;
extern crate gstreamer_video as gst_video;

extern crate servo_media_gstreamer_render as sm_gst_render;
extern crate servo_media_player as sm_player;

use gst::prelude::*;
use gst_gl::prelude::*;
use sm_gst_render::Render;
use sm_player::context::{GlApi, GlContext, NativeDisplay, PlayerGLContext};
use sm_player::frame::{Buffer, Frame, FrameData};
use sm_player::PlayerError;
use std::sync::{Arc, Mutex};

struct GStreamerBuffer {
is_external_oes: bool,
frame: gst_video::VideoFrame<gst_video::video_frame::Readable>,
}

impl Buffer for GStreamerBuffer {
fn to_vec(&self) -> Result<FrameData, ()> {
// packed formats are guaranteed to be in a single plane
if self.frame.format() == gst_video::VideoFormat::Rgba {
let tex_id = self.frame.get_texture_id(0).ok_or_else(|| ())?;
Ok(if self.is_external_oes {
FrameData::OESTexture(tex_id)
} else {
FrameData::Texture(tex_id)
})
} else {
Err(())
}
}
}

pub struct RenderAndroid {
display: gst_gl::GLDisplay,
app_context: gst_gl::GLContext,
gst_context: Arc<Mutex<Option<gst_gl::GLContext>>>,
gl_upload: Arc<Mutex<Option<gst::Element>>>,
}

impl RenderAndroid {
/// Tries to create a new intance of the `RenderAndroid`
///
/// # Arguments
///
/// * `context` - is the PlayerContext trait object from
/// application.
pub fn new(app_gl_context: Box<dyn PlayerGLContext>) -> Option<RenderAndroid> {
// Check that we actually have the elements that we
// need to make this work.
if gst::ElementFactory::find("glsinkbin").is_none() {
return None;
}

let display_native = app_gl_context.get_native_display();
let gl_context = app_gl_context.get_gl_context();
let gl_api = match app_gl_context.get_gl_api() {
GlApi::OpenGL => gst_gl::GLAPI::OPENGL,
GlApi::OpenGL3 => gst_gl::GLAPI::OPENGL3,
GlApi::Gles1 => gst_gl::GLAPI::GLES1,
GlApi::Gles2 => gst_gl::GLAPI::GLES2,
GlApi::None => gst_gl::GLAPI::NONE,
};

let (wrapped_context, display) = match gl_context {
GlContext::Egl(context) => {
let display = match display_native {
NativeDisplay::Egl(display_native) => {
unsafe { gst_gl::GLDisplayEGL::new_with_egl_display(display_native) }
.and_then(|display| Some(display.upcast()))
}
_ => None,
};

if let Some(display) = display {
let wrapped_context = unsafe {
gst_gl::GLContext::new_wrapped(
&display,
context,
gst_gl::GLPlatform::EGL,
gl_api,
)
};
(wrapped_context, Some(display))
} else {
(None, None)
}
}
_ => (None, None),
};

if let Some(app_context) = wrapped_context {
Some(RenderAndroid {
display: display.unwrap(),
app_context,
gst_context: Arc::new(Mutex::new(None)),
gl_upload: Arc::new(Mutex::new(None)),
})
} else {
None
}
}
}

impl Render for RenderAndroid {
fn is_gl(&self) -> bool {
true
}

fn build_frame(&self, sample: gst::Sample) -> Result<Frame, ()> {
if self.gst_context.lock().unwrap().is_none() && self.gl_upload.lock().unwrap().is_some() {
*self.gst_context.lock().unwrap() =
if let Some(glupload) = self.gl_upload.lock().unwrap().as_ref() {
glupload
.get_property("context")
.or_else(|_| Err(()))?
.get::<gst_gl::GLContext>()
} else {
None
};
}

let buffer = sample.get_buffer_owned().ok_or_else(|| ())?;
let caps = sample.get_caps().ok_or_else(|| ())?;

let is_external_oes = caps
.get_structure(0)
.and_then(|s| {
s.get::<&str>("texture-target").and_then(|target| {
if target == "external-oes" {
Some(s)
} else {
None
}
})
})
.is_some();

let info = gst_video::VideoInfo::from_caps(caps).ok_or_else(|| ())?;

if self.gst_context.lock().unwrap().is_some() {
if let Some(sync_meta) = buffer.get_meta::<gst_gl::GLSyncMeta>() {
sync_meta.set_sync_point(self.gst_context.lock().unwrap().as_ref().unwrap());
}
}

let frame =
gst_video::VideoFrame::from_buffer_readable_gl(buffer, &info).or_else(|_| Err(()))?;

if self.gst_context.lock().unwrap().is_some() {
if let Some(sync_meta) = frame.buffer().get_meta::<gst_gl::GLSyncMeta>() {
sync_meta.wait(&self.app_context);
}
}

Frame::new(
info.width() as i32,
info.height() as i32,
Arc::new(GStreamerBuffer {
is_external_oes,
frame,
}),
)
}

fn build_video_sink(
&self,
appsink: &gst::Element,
pipeline: &gst::Element,
) -> Result<(), PlayerError> {
if self.gl_upload.lock().unwrap().is_some() {
return Err(PlayerError::Backend(
"render unix already setup the video sink".to_owned(),
));
}

let vsinkbin = gst::ElementFactory::make("glsinkbin", Some("servo-media-vsink"))
.ok_or(PlayerError::Backend("glupload creation failed".to_owned()))?;

let caps = gst::Caps::builder("video/x-raw")
.features(&[&gst_gl::CAPS_FEATURE_MEMORY_GL_MEMORY])
.field("format", &gst_video::VideoFormat::Rgba.to_string())
.field("texture-target", &gst::List::new(&[&"2D", &"external-oes"]))
.build();
appsink
.set_property("caps", &caps)
.expect("appsink doesn't have expected 'caps' property");

vsinkbin
.set_property("sink", &appsink)
.expect("glsinkbin doesn't have expected 'sink' property");

pipeline
.set_property("video-sink", &vsinkbin)
.expect("playbin doesn't have expected 'video-sink' property");

let bus = pipeline.get_bus().expect("pipeline with no bus");
let display_ = self.display.clone();
let context_ = self.app_context.clone();
bus.set_sync_handler(move |_, msg| {
match msg.view() {
gst::MessageView::NeedContext(ctxt) => {
if let Some(el) = msg.get_src().map(|s| s.downcast::<gst::Element>().unwrap()) {
let context_type = ctxt.get_context_type();
if context_type == *gst_gl::GL_DISPLAY_CONTEXT_TYPE {
let ctxt = gst::Context::new(context_type, true);
ctxt.set_gl_display(&display_);
el.set_context(&ctxt);
} else if context_type == "gst.gl.app_context" {
let mut ctxt = gst::Context::new(context_type, true);
{
let s = ctxt.get_mut().unwrap().get_mut_structure();
s.set_value("context", context_.to_send_value());
}
el.set_context(&ctxt);
}
}
}
_ => (),
}

gst::BusSyncReply::Pass
});

let mut iter = vsinkbin
.dynamic_cast::<gst::Bin>()
.unwrap()
.iterate_elements();
*self.gl_upload.lock().unwrap() = loop {
match iter.next() {
Ok(Some(element)) => {
if "glupload" == element.get_factory().unwrap().get_name() {
break Some(element);
}
}
Err(gst::IteratorError::Resync) => iter.resync(),
_ => break None,
}
};

Ok(())
}
}
8 changes: 0 additions & 8 deletions backends/gstreamer/render-unix/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,6 @@
//! wrapping the *appsink* from the Player. And the shared frames are
//! mapped as texture IDs.
#![cfg(any(
target_os = "linux",
target_os = "dragonfly",
target_os = "freebsd",
target_os = "netbsd",
target_os = "openbsd"
))]

#[macro_use]
extern crate gstreamer as gst;
extern crate gstreamer_gl as gst_gl;
Expand Down
15 changes: 14 additions & 1 deletion backends/gstreamer/render.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,12 +28,25 @@ mod platform {
}
}

#[cfg(target_os = "android")]
mod platform {
extern crate servo_media_gstreamer_render_android;
pub use self::servo_media_gstreamer_render_android::RenderAndroid as Render;

use super::*;

pub fn create_render(gl_context: Box<dyn PlayerGLContext>) -> Option<Render> {
Render::new(gl_context)
}
}

#[cfg(not(any(
target_os = "linux",
target_os = "dragonfly",
target_os = "freebsd",
target_os = "netbsd",
target_os = "openbsd"
target_os = "openbsd",
target_os = "android",
)))]
mod platform {
use servo_media_gstreamer_render::Render as RenderTrait;
Expand Down

0 comments on commit d6f799e

Please sign in to comment.