libexplain  1.4.D001
libexplain/buffer/v4l2_frmivalenum.c
Go to the documentation of this file.
00001 /*
00002  * libexplain - a library of system-call-specific strerror replacements
00003  * Copyright (C) 2011, 2013 Peter Miller
00004  *
00005  * This program is free software; you can redistribute it and/or modify
00006  * it under the terms of the GNU Lesser General Public License as
00007  * published by the Free Software Foundation; either version 3 of the
00008  * License, or (at your option) any later version.
00009  *
00010  * This program is distributed in the hope that it will be useful,
00011  * but WITHOUT ANY WARRANTY; without even the implied warranty of
00012  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
00013  * Lesser General Public License for more details.
00014  *
00015  * You should have received a copy of the GNU Lesser General Public License
00016  * along with this program. If not, see <http://www.gnu.org/licenses/>.
00017  */
00018 
00019 #include <libexplain/ac/errno.h>
00020 #include <libexplain/ac/linux/videodev2.h>
00021 #include <libexplain/ac/string.h>
00022 #include <libexplain/ac/sys/ioctl.h>
00023 
00024 #include <libexplain/buffer/int32_t.h>
00025 #include <libexplain/buffer/pointer.h>
00026 #include <libexplain/buffer/v4l2_fract.h>
00027 #include <libexplain/buffer/v4l2_frmivalenum.h>
00028 #include <libexplain/buffer/v4l2_frmivaltypes.h>
00029 #include <libexplain/buffer/v4l2_frmival_stepwise.h>
00030 #include <libexplain/buffer/v4l2_pixel_format.h>
00031 #include <libexplain/is_efault.h>
00032 #include <libexplain/sizeof.h>
00033 
00034 
00035 #ifdef HAVE_LINUX_VIDEODEV2_H
00036 
00037 void
00038 explain_buffer_v4l2_frmivalenum(explain_string_buffer_t *sb,
00039     const struct v4l2_frmivalenum *data, int extra)
00040 {
00041     if (explain_is_efault_pointer(data, sizeof(*data)))
00042     {
00043         explain_buffer_pointer(sb, data);
00044         return;
00045     }
00046 
00047     explain_string_buffer_puts(sb, "{ index = ");
00048     explain_buffer_uint32_t(sb, data->index);
00049     explain_string_buffer_puts(sb, ", pixel_format = ");
00050     explain_buffer_v4l2_pixel_format(sb, data->pixel_format);
00051     if (extra)
00052     {
00053         explain_string_buffer_puts(sb, ", width = ");
00054         explain_buffer_uint32_t(sb, data->width);
00055         explain_string_buffer_puts(sb, ", height = ");
00056         explain_buffer_uint32_t(sb, data->height);
00057         explain_string_buffer_puts(sb, ", type = ");
00058         explain_buffer_v4l2_frmivaltypes(sb, data->type);
00059         switch (data->type)
00060         {
00061         case V4L2_FRMIVAL_TYPE_DISCRETE:
00062             explain_string_buffer_puts(sb, ", discrete = ");
00063             explain_buffer_v4l2_fract(sb, &data->discrete);
00064             break;
00065 
00066         case V4L2_FRMIVAL_TYPE_STEPWISE:
00067             explain_string_buffer_puts(sb, ", stepwise = ");
00068             explain_buffer_v4l2_frmival_stepwise(sb, &data->stepwise);
00069             break;
00070 
00071         case V4L2_FRMIVAL_TYPE_CONTINUOUS:
00072         default:
00073             break;
00074         }
00075         if (!explain_uint32_array_all_zero(data->reserved,
00076             SIZEOF(data->reserved)))
00077         {
00078             explain_string_buffer_puts(sb, ", reserved = ");
00079             explain_buffer_uint32_array(sb, data->reserved,
00080                 SIZEOF(data->reserved));
00081         }
00082     }
00083     explain_string_buffer_puts(sb, " }");
00084 }
00085 
00086 
00087 int
00088 explain_v4l2_frmivalenum_get_n(int fildes, int pixfmt)
00089 {
00090     int lo = 0;
00091     int hi = 200;
00092     for (;;)
00093     {
00094         int mid = (lo + hi) / 2;
00095         struct v4l2_frmivalenum qry;
00096         memset(&qry, 0, sizeof(qry));
00097         qry.index = mid;
00098         qry.pixel_format = pixfmt;
00099         if (ioctl(fildes, VIDIOC_ENUM_FRAMEINTERVALS, &qry) >= 0)
00100         {
00101             if (hi <= 0 && lo <= 0)
00102                 return -1;
00103             /* mid < ninputs */
00104             lo = mid + 1;
00105             if (lo > hi)
00106                 return lo;
00107         }
00108         else if (errno != EINVAL)
00109         {
00110             return -1;
00111         }
00112         else
00113         {
00114             if (hi <= 0 && lo <= 0)
00115                 return -1;
00116             /* mid >= ninputs */
00117             hi = mid - 1;
00118             if (lo >= hi)
00119                 return lo;
00120         }
00121     }
00122 }
00123 
00124 #endif
00125 
00126 /* vim: set ts=8 sw=4 et : */