<p dir="ltr">Pekka,<br>
I've got just a comple of general comments below.</p>
<p dir="ltr">On Nov 8, 2013 8:46 AM, "Pekka Paalanen" <<a href="mailto:ppaalanen@gmail.com">ppaalanen@gmail.com</a>> wrote:<br>
><br>
> Hi all,<br>
><br>
> this is the v2 of the crop and scale extension, as RFC.<br>
><br>
> The v1 was in:<br>
> <a href="http://lists.freedesktop.org/archives/wayland-devel/2013-April/008927.html">http://lists.freedesktop.org/archives/wayland-devel/2013-April/008927.html</a><br>
><br>
> Based on v1, Jonny Lamb has been working on a Weston implementation,<br>
> starting from the protocol side, and working towards the renderer<br>
> implementations. That work is still very much in progress.<br>
><br>
><br>
> Introduction:<br>
><br>
> The primary use case for the crop & scale extension is for hardware<br>
> accelerated, zero-copy video display. Hardware video decoders produce<br>
> complete frames or ultimately wl_buffers that are attached to<br>
> a wl_surface. The crop & scale extension allows the client to<br>
> request the compositor to crop and scale the frame so it fits the<br>
> client's purposes. Optimally, a compositor implements this by<br>
> programming a hardware overlay unit to do the requested cropping and<br>
> scaling. The major benefit is that the CPU never has to even see the<br>
> video pixels.<br>
><br>
> Probably a common case is to have a video in a sub-surface, so that<br>
> window decorations and overlaid GUI elements can be in other<br>
> (sub-)surfaces, and avoid touching the pixels in the video frame<br>
> buffers. Video as a browser element will need cropping when the element<br>
> is partially off-view. Scaling should be useful in general, e.g.<br>
> showing a video scaled up (or down) but still in a window.<br>
><br>
> However, I also see that crop & scale can be used to present videos<br>
> with non-square pixels in fullscreen (e.g. without sub-surfaces). Crop<br>
> & scale is needed in this case, because the fullscreening methods in<br>
> wl_shell do not allow changing the aspect ratio, or cropping parts of<br>
> the video out to fill the whole output area when video and output<br>
> aspect ratios differ. The fullscreening methods support only adding<br>
> black borders, but not scale-to-fill-and-crop.<br>
><br>
><br>
> Changes since v1:<br>
><br>
> The changes I have made are very small, and can be seen in patch form<br>
> at:<br>
> <a href="http://cgit.collabora.com/git/user/pq/weston.git/log/?h=clipscale-wip">http://cgit.collabora.com/git/user/pq/weston.git/log/?h=clipscale-wip</a><br>
><br>
> The changes are:<br>
> - improve wording, add missing details<br>
> - take buffer_scale into account<br>
> - rewrite the coordinate transformations more clearly<br>
><br>
> In the end, I did not get much else out from the discussions of v1.<br>
><br>
> I think some people do not like the structure of the protocol<br>
> additions, namely adding yet another global factory interface and a<br>
> sub-interface to wl_surface. If the concensus really is to move this<br>
> into a single wl_surface request, that is fine.<br>
><br>
> But, to me this would not make sense as a request in wl_subsurface. The<br>
> crop & scale state is supposed to be driven by the application<br>
> component that produces the wl_buffers and attaches them to a<br>
> wl_surface. The wl_subsurface interface OTOH is to be used by the<br>
> parent component, for e.g. setting the sub-surface position. The<br>
> situation is similar to a compositor vs. clients: clients define the<br>
> surface size and content, compositor the position. Also, the crop &<br>
> scale state is not well suited to be "forced on" by a parent software<br>
> component, as it changes how e.g. input event coordinates relate to the<br>
> wl_buffer contents. Finally, there is the fullscreen video use case I<br>
> described above.</p>
<p dir="ltr">In the design of both the subsurfaces and crop&scale, I fear that you may be trying too hard to avoid out-of-band communication between components with little to no benefit.  I don't want to start a bikeshedding landslide here and you are free to disagree.  However, I'm concerned that we're overcomplicating things without gaining anything.</p>

<p dir="ltr">In some simple use-cases, you will be able to write one component that has runs on a subsurface and another that uses it and the two components are basically unaware of each other beyond the initial handshake.  However, I think in practice I feel that what is more likely to happen is that the primary tookit will handle everything.  If this is the case, it makes no difference whether it's separate or inside wl_subsurface except that the toolkit has more extensions to wrangle.</p>

<p dir="ltr">Concerning full-screen video:  After thinking about it a bit more, I think this is a failure of wl_subsurface more than a need for scaling regular surfaces.  I think the primary issue there is that wl_subsurface provides no real way to make a surface that, itself, is empty but its subsurfaces have content.  If this were possible, then it would be no problem putting crop&scale in the wl_subsurface protocol.  It's worth noting that this same issue makes putting a video (from an external source) in a SHM subsurface frame really awkward.  See also this e-mail from April:</p>

<p dir="ltr"><a href="http://lists.freedesktop.org/archives/wayland-devel/2013-April/008875.html">http://lists.freedesktop.org/archives/wayland-devel/2013-April/008875.html</a></p>
<p dir="ltr">Hope it's useful.  Thanks,<br>
--Jason Ekstrand</p>
<p dir="ltr">><br>
> The interface names are still so-and-so, I haven't come up with<br>
> anything better. wl_scaler is pretty ok, but I feel something better<br>
> would be in place for wl_surface_scaler. How would wl_viewport sound<br>
> like?<br>
><br>
> A minor benefit of keeping crop & scale as a separate interface is that<br>
> it could be made optional, a per-backend or renderer thing.<br>
><br>
><br>
> For your reviewing pleasure, here is the v2:<br>
><br>
> <?xml version="1.0" encoding="UTF-8"?><br>
> <protocol name="scaler"><br>
><br>
>   <copyright><br>
>     Copyright Â© 2013 Collabora, Ltd.<br>
><br>
>     Permission to use, copy, modify, distribute, and sell this<br>
>     software and its documentation for any purpose is hereby granted<br>
>     without fee, provided that the above copyright notice appear in<br>
>     all copies and that both that copyright notice and this permission<br>
>     notice appear in supporting documentation, and that the name of<br>
>     the copyright holders not be used in advertising or publicity<br>
>     pertaining to distribution of the software without specific,<br>
>     written prior permission.  The copyright holders make no<br>
>     representations about the suitability of this software for any<br>
>     purpose.  It is provided "as is" without express or implied<br>
>     warranty.<br>
><br>
>     THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS<br>
>     SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND<br>
>     FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY<br>
>     SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES<br>
>     WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN<br>
>     AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,<br>
>     ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF<br>
>     THIS SOFTWARE.<br>
>   </copyright><br>
><br>
>   <interface name="wl_scaler" version="1"><br>
>     <description summary="surface cropping and scaling"><br>
>       The global interface exposing surface cropping and scaling<br>
>       capabilities is used to instantiate an interface extension for a<br>
>       wl_surface object. This extended interface will then allow<br>
>       cropping and scaling the surface contents, effectively<br>
>       disconnecting the direct relationship between the buffer and the<br>
>       surface size.<br>
>     </description><br>
><br>
>     <request name="destroy" type="destructor"><br>
>       <description summary="unbind from the cropping and scaling interface"><br>
>         Informs the server that the client will not be using this<br>
>         protocol object anymore. This does not affect any other objects,<br>
>         wl_surface_scaler objects included.<br>
>       </description><br>
>     </request><br>
><br>
>     <enum name="error"><br>
>       <entry name="scaler_exists" value="0"<br>
>              summary="the surface already has a scaler object associated"/><br>
>     </enum><br>
><br>
>     <request name="get_surface_scaler"><br>
>       <description summary="extend surface interface for crop and scale"><br>
>         Instantiate an interface extension for the given wl_surface to<br>
>         crop and scale its content. If the given wl_surface already has<br>
>         a wl_surface_scaler object associated, the scaler_exists<br>
>         protocol error is raised.<br>
>       </description><br>
><br>
>       <arg name="id" type="new_id" interface="wl_surface_scaler"<br>
>            summary="the new scaler interface id"/><br>
>       <arg name="surface" type="object" interface="wl_surface"<br>
>            summary="the surface"/><br>
>     </request><br>
>   </interface><br>
><br>
>   <interface name="wl_surface_scaler" version="1"><br>
>     <description summary="crop and scale interface to a wl_surface"><br>
>       An additional interface to a wl_surface object, which allows the<br>
>       client to specify the cropping and scaling of the surface<br>
>       contents.<br>
><br>
>       This interface allows to define the source rectangle (src_x,<br>
>       src_y, src_width, src_height) from where to take the wl_buffer<br>
>       contents, and scale that to destination size (dst_width,<br>
>       dst_height). This state is double-buffered, and is applied on the<br>
>       next wl_surface.commit.<br>
><br>
>       Before the first set request, the wl_surface still behaves as if<br>
>       there was no crop and scale state. That is, no scaling is applied,<br>
>       and the surface size is as defined in wl_surface.attach.<br>
><br>
>       The crop and scale state causes the surface size to become<br>
>       dst_width, dst_height. This overrides whatever the attached<br>
>       wl_buffer size is, unless the wl_buffer is NULL. If the wl_buffer is<br>
>       NULL, the surface has no content and therefore no size.<br>
><br>
>       The coordinate transformations from buffer pixel coordinates up to<br>
>       the surface-local coordinates happen in the following order:<br>
>         1. buffer_transform (wl_surface.set_buffer_transform)<br>
>         2. buffer_scale (wl_surface.set_buffer_scale)<br>
>         3. crop and scale (wl_surface_scaler.set)<br>
>       This means, that the source rectangle coordinates of crop and scale<br>
>       are given in the coordinates after the buffer transform and scale,<br>
>       i.e. in the coordinates that would be the surface-local coordinates<br>
>       if the crop and scale was not applied.<br>
><br>
>       If the source rectangle is partially or completely outside of the<br>
>       wl_buffer, then the surface contents are undefined (not void), and<br>
>       the surface size is still dst_width, dst_height.<br>
><br>
>       The x, y arguments of wl_surface.attach are applied as normal to<br>
>       the surface. They indicate how many pixels to remove from the<br>
>       surface size from the left and the top. In other words, they are<br>
>       still in the surface-local coordinate system, just like dst_width<br>
>       and dst_height are.<br>
><br>
>       If the wl_surface associated with the wl_surface_scaler is<br>
>       destroyed, the wl_surface_scaler object becomes inert.<br>
><br>
>       If the wl_surface_scaler object is destroyed, the crop and scale<br>
>       state is removed from the wl_surface. The change will be applied<br>
>       on the next wl_surface.commit.<br>
>     </description><br>
><br>
>     <request name="destroy" type="destructor"><br>
>       <description summary="remove scaling and cropping from the surface"><br>
>         The associated wl_surface's crop and scale state is removed.<br>
>         The change is applied on the next wl_surface.commit.<br>
>       </description><br>
>     </request><br>
><br>
>     <enum name="error"><br>
>       <entry name="bad_value" value="0"<br>
>              summary="negative values in width or height"/><br>
>     </enum><br>
><br>
>     <request name="set"><br>
>       <description summary="set the crop and scale state"><br>
>         Set the crop and scale state of the associated wl_surface. See<br>
>         wl_surface_scaler for the description, and relation to the<br>
>         wl_buffer size.<br>
><br>
>         If any of src_width, src_height, dst_width, and dst_height is<br>
>         negative, the bad_value protocol error is raised.<br>
><br>
>         The crop and scale state is double-buffered state, and will be<br>
>         applied on the next wl_surface.commit.<br>
><br>
>         Arguments dst_x and dst_y do not exist here, use the x and y<br>
>         arguments to wl_surface.attach. The x, y, dst_width, and dst_height<br>
>         define the surface-local coordinate system irrespective of the<br>
>         attached wl_buffer size.<br>
>       </description><br>
><br>
>       <arg name="src_x" type="fixed" summary="source rectangle x"/><br>
>       <arg name="src_y" type="fixed" summary="source rectangle y"/><br>
>       <arg name="src_width" type="fixed" summary="source rectangle width"/><br>
>       <arg name="src_height" type="fixed" summary="source rectangle height"/><br>
>       <arg name="dst_width" type="int" summary="surface width"/><br>
>       <arg name="dst_height" type="int" summary="surface height"/><br>
>     </request><br>
><br>
>   </interface><br>
> </protocol><br>
><br>
><br>
> Thanks,<br>
> pq<br>
> _______________________________________________<br>
> wayland-devel mailing list<br>
> <a href="mailto:wayland-devel@lists.freedesktop.org">wayland-devel@lists.freedesktop.org</a><br>
> <a href="http://lists.freedesktop.org/mailman/listinfo/wayland-devel">http://lists.freedesktop.org/mailman/listinfo/wayland-devel</a><br>
</p>