[vlc-devel] [PATCH 03/11] Add transform helper functions.

Matthias Keiser matthias at tristan-inc.com
Thu Mar 6 01:12:24 CET 2014


---
 include/vlc_es.h     |  48 ++++++++++++++
 src/libvlccore.sym   |   4 ++
 src/misc/es_format.c | 173 +++++++++++++++++++++++++++++++++++++++++++++++++++
 3 files changed, 225 insertions(+)

diff --git a/include/vlc_es.h b/include/vlc_es.h
index 38d63da..7bdd1fc 100644
--- a/include/vlc_es.h
+++ b/include/vlc_es.h
@@ -153,6 +153,8 @@ typedef enum video_orientation_t
     ORIENT_RIGHT_BOTTOM, /**< Anti-transposed */
 
     ORIENT_NORMAL      = ORIENT_TOP_LEFT,
+    ORIENT_TRANSPOSED  = ORIENT_LEFT_TOP,
+    ORIENT_ANTI_TRANSPOSED = ORIENT_RIGHT_BOTTOM,
     ORIENT_HFLIPPED    = ORIENT_TOP_RIGHT,
     ORIENT_VFLIPPED    = ORIENT_BOTTOM_LEFT,
     ORIENT_ROTATED_180 = ORIENT_BOTTOM_RIGHT,
@@ -174,6 +176,18 @@ typedef enum video_orientation_t
 /** Applies horizontal flip to an orientation */
 #define ORIENT_ROTATE_180(orient) ((orient) ^ 3)
 
+typedef enum video_transform_t
+{
+    TRANSFORM_IDENTIY        = ORIENT_NORMAL,
+    TRANSFORM_HFLIP          = ORIENT_HFLIPPED,
+    TRANSFORM_VFLIP          = ORIENT_VFLIPPED,
+    TRANSFORM_R180           = ORIENT_ROTATED_180,
+    TRANSFORM_R270           = ORIENT_ROTATED_270,
+    TRANSFORM_R90            = ORIENT_ROTATED_90,
+    TRANSFORM_TRANSPOSE      = ORIENT_TRANSPOSED,
+    TRANSFORM_ANTI_TRANSPOSE = ORIENT_ANTI_TRANSPOSED
+} video_transform_t;
+
 /**
  * video format description
  */
@@ -265,6 +279,28 @@ VLC_API void video_format_CopyCrop( video_format_t *, const video_format_t * );
 VLC_API void video_format_ScaleCropAr( video_format_t *, const video_format_t * );
 
 /**
+ * This function "normalizes" the formats orientation, by switching the a/r according to the orientation,
+ * producing a format whose orientation is ORIENT_NORMAL. It makes a shallow copy (pallette is not alloc'ed).
+ */
+VLC_API void video_format_ApplyRotation(const video_format_t * restrict in, video_format_t * restrict out);
+
+/**
+ * This function applies the transform operation to fmt.
+ */
+VLC_API void video_format_TransformBy(video_format_t *fmt, video_transform_t transform);
+
+/**
+ * This function applies the transforms necessary to fmt so that the resulting fmt
+ * has the dst_orientation.
+ */
+VLC_API void video_format_TransformTo(video_format_t *fmt, video_orientation_t dst_orientation);
+
+/**
+ * Returns the operation required to transform src into dst.
+ */
+VLC_API video_transform_t video_format_GetTransform(video_orientation_t src, video_orientation_t dst);
+
+/**
  * This function will check if the first video format is similar
  * to the second one.
  */
@@ -275,6 +311,18 @@ VLC_API bool video_format_IsSimilar( const video_format_t *, const video_format_
  */
 VLC_API void video_format_Print( vlc_object_t *, const char *, const video_format_t * );
 
+
+static inline video_transform_t transform_Inverse( video_transform_t transform )
+{
+    switch ( transform ) {
+        case TRANSFORM_R90:
+            return TRANSFORM_R270;
+        case TRANSFORM_R270:
+            return TRANSFORM_R90;
+        default:
+            return transform;
+    }
+}
 /**
  * subtitles format description
  */
diff --git a/src/libvlccore.sym b/src/libvlccore.sym
index 3858e43..e2c2008 100644
--- a/src/libvlccore.sym
+++ b/src/libvlccore.sym
@@ -467,6 +467,10 @@ var_LocationParse
 video_format_CopyCrop
 video_format_ScaleCropAr
 video_format_FixRgb
+video_format_ApplyRotation
+video_format_TransformBy
+video_format_TransformTo
+video_format_GetTransform
 video_format_IsSimilar
 video_format_Setup
 video_format_Print
diff --git a/src/misc/es_format.c b/src/misc/es_format.c
index 8936414..82961b5 100644
--- a/src/misc/es_format.c
+++ b/src/misc/es_format.c
@@ -244,6 +244,179 @@ void video_format_ScaleCropAr( video_format_t *p_dst, const video_format_t *p_sr
                 p_dst->i_sar_num, p_dst->i_sar_den, 65536);
 }
 
+//Simplify transforms to have something more managable. Order: angle, hflip.
+void transform_GetBasicOps( video_transform_t transform, int *angle, int *hflip ) {
+
+    *angle = 0;
+    *hflip = 0;
+
+    switch ( transform ) {
+
+        case TRANSFORM_R90:
+            *angle = 90;
+            break;
+        case TRANSFORM_R180:
+            *angle = 180;
+            break;
+        case TRANSFORM_R270:
+            *angle = 270;
+            break;
+        case TRANSFORM_HFLIP:
+            *hflip = 1;
+            break;
+        case TRANSFORM_VFLIP:
+            *angle = 180;
+            *hflip = 1;
+            break;
+        case TRANSFORM_TRANSPOSE:
+            *angle = 90;
+            *hflip = 1;
+            break;
+        case TRANSFORM_ANTI_TRANSPOSE:
+            *angle = 270;
+            *hflip = 1;
+            break;
+    }
+}
+
+video_transform_t transform_FromBasicOps( int angle, int hflip )
+{
+    switch ( angle ) {
+        case 0:
+            return hflip ? TRANSFORM_HFLIP : TRANSFORM_IDENTIY;
+        case 90:
+            return hflip ? TRANSFORM_TRANSPOSE : TRANSFORM_R90;
+        case 180:
+            return hflip ? TRANSFORM_VFLIP : TRANSFORM_R180;
+        case 270:
+            return hflip ? TRANSFORM_ANTI_TRANSPOSE : TRANSFORM_R270;
+        default:
+            return TRANSFORM_IDENTIY;
+    }
+}
+
+video_transform_t video_format_GetTransform( video_orientation_t src, video_orientation_t dst )
+{
+    int angle1 = 0;
+    int hflip1 = 0;
+
+    transform_GetBasicOps(  (video_transform_t)src, &angle1, &hflip1 );
+
+    int angle2 = 0;
+    int hflip2 = 0;
+
+    transform_GetBasicOps( transform_Inverse( (video_transform_t)dst ), &angle2, &hflip2 );
+
+    int angle = (angle1 + angle2) % 360;
+    int hflip = (hflip1 + hflip2) % 2;
+
+    return transform_FromBasicOps(angle, hflip);
+}
+
+void video_format_TransformTo( video_format_t *fmt, video_orientation_t dst_orientation )
+{
+    video_transform_t transform = video_format_GetTransform(fmt->orientation, dst_orientation);
+
+    video_format_TransformBy(fmt, transform);
+}
+
+void video_format_TransformBy( video_format_t *fmt, video_transform_t transform )
+{
+    /* Get destination orientation */
+
+    int angle1 = 0;
+    int hflip1 = 0;
+
+    transform_GetBasicOps( transform, &angle1, &hflip1 );
+
+    int angle2 = 0;
+    int hflip2 = 0;
+
+    transform_GetBasicOps( (video_transform_t)( fmt->orientation ), &angle2, &hflip2 );
+
+    int angle = (angle2 - angle1 + 360) % 360;
+    int hflip = (hflip2 - hflip1 + 2) % 2;
+
+    video_orientation_t dst_orient = ORIENT_NORMAL;
+
+    if( hflip ) {
+
+        if( angle == 0 )
+            dst_orient = ORIENT_HFLIPPED;
+        else if( angle == 90 )
+            dst_orient = ORIENT_ANTI_TRANSPOSED;
+        else if( angle == 180 )
+            dst_orient = ORIENT_VFLIPPED;
+        else if( angle == 270 )
+            dst_orient = ORIENT_TRANSPOSED;
+    }
+    else {
+
+        if( angle == 90 )
+            dst_orient = ORIENT_ROTATED_90;
+        else if( angle == 180 )
+            dst_orient = ORIENT_ROTATED_180;
+        else if( angle == 270 )
+            dst_orient = ORIENT_ROTATED_270;
+    }
+
+    /* Apply transform */
+
+    video_format_t scratch = *fmt;
+
+    if( ORIENT_IS_SWAP( fmt->orientation ) != ORIENT_IS_SWAP( dst_orient )) {
+
+        fmt->i_width = scratch.i_height;
+        fmt->i_visible_width = scratch.i_visible_height;
+        fmt->i_height = scratch.i_width;
+        fmt->i_visible_height = scratch.i_visible_width;
+        fmt->i_sar_num = scratch.i_sar_den;
+        fmt->i_sar_den = scratch.i_sar_num;
+    }
+
+    unsigned int delta_y = scratch.i_height - scratch.i_visible_height - scratch.i_y_offset;
+    unsigned int delta_x = scratch.i_width - scratch.i_visible_width - scratch.i_x_offset;
+
+    switch ( transform )
+    {
+        case TRANSFORM_R90:
+            fmt->i_x_offset = delta_y;
+            fmt->i_y_offset = scratch.i_x_offset;
+            break;
+        case TRANSFORM_R180:
+            fmt->i_x_offset = delta_x;
+            fmt->i_y_offset = delta_y;
+            break;
+        case TRANSFORM_R270:
+            fmt->i_x_offset = scratch.i_y_offset;
+            fmt->i_y_offset = delta_x;
+            break;
+        case TRANSFORM_HFLIP:
+            fmt->i_x_offset = delta_x;
+            break;
+        case TRANSFORM_VFLIP:
+            fmt->i_y_offset = delta_y;
+            break;
+        case TRANSFORM_TRANSPOSE:
+            fmt->i_x_offset = scratch.i_y_offset;
+            fmt->i_y_offset = scratch.i_x_offset;
+            break;
+        case TRANSFORM_ANTI_TRANSPOSE:
+            fmt->i_x_offset = delta_y;
+            fmt->i_y_offset = delta_x;
+            break;
+    }
+
+    fmt->orientation = dst_orient;
+}
+
+void video_format_ApplyRotation( const video_format_t * restrict in, video_format_t * restrict out )
+{
+    *out = *in;
+
+    video_format_TransformTo(out, ORIENT_NORMAL);
+}
+
 bool video_format_IsSimilar( const video_format_t *p_fmt1, const video_format_t *p_fmt2 )
 {
     video_format_t v1 = *p_fmt1;
-- 
1.8.3.4 (Apple Git-47)




More information about the vlc-devel mailing list