libexplain  1.4.D001
libexplain/buffer/v4l2_modulator.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_modulator.h>
00027 #include <libexplain/buffer/v4l2_tuner_capability.h>
00028 #include <libexplain/buffer/v4l2_tuner_rxsubchans.h>
00029 #include <libexplain/is_efault.h>
00030 #include <libexplain/sizeof.h>
00031 
00032 
00033 #ifdef HAVE_LINUX_VIDEODEV2_H
00034 
00035 void
00036 explain_buffer_v4l2_modulator(explain_string_buffer_t *sb,
00037     const struct v4l2_modulator *data, int extra)
00038 {
00039     if (explain_is_efault_pointer(data, sizeof(*data)))
00040     {
00041         explain_buffer_pointer(sb, data);
00042         return;
00043     }
00044 
00045     explain_string_buffer_puts(sb, "{ index = ");
00046     explain_buffer_uint32_t(sb, data->index);
00047     if (extra)
00048     {
00049         explain_string_buffer_puts(sb, ", name = ");
00050         explain_string_buffer_putsu_quoted_n(sb, data->name,
00051             sizeof(data->name));
00052         explain_string_buffer_puts(sb, ", capability = ");
00053         explain_buffer_v4l2_tuner_capability(sb, data->capability);
00054         explain_string_buffer_puts(sb, ", rangelow = ");
00055         explain_buffer_uint32_t(sb, data->rangelow);
00056         explain_string_buffer_puts(sb, ", rangehigh = ");
00057         explain_buffer_uint32_t(sb, data->rangehigh);
00058         explain_string_buffer_puts(sb, ", txsubchans = ");
00059         explain_buffer_v4l2_tuner_rxsubchans(sb, data->txsubchans);
00060         if (!explain_uint32_array_all_zero(data->reserved,
00061             SIZEOF(data->reserved)))
00062         {
00063             explain_string_buffer_puts(sb, ", reserved = ");
00064             explain_buffer_uint32_array(sb, data->reserved,
00065                 SIZEOF(data->reserved));
00066         }
00067     }
00068     explain_string_buffer_puts(sb, " }");
00069 }
00070 
00071 
00072 int
00073 explain_v4l2_modulator_get_n(int fildes)
00074 {
00075     const int max = 200;
00076 
00077     int lo = 0;
00078     int hi = max;
00079     for (;;)
00080     {
00081         int mid = (lo + hi) / 2;
00082         struct v4l2_modulator qry;
00083         memset(&qry, 0, sizeof(qry));
00084         qry.index = mid;
00085         if (ioctl(fildes, VIDIOC_G_MODULATOR, &qry) >= 0)
00086         {
00087             if (hi <= 0 && lo <= 0)
00088                 return -1;
00089             /* mid < ninputs */
00090             lo = mid + 1;
00091             if (lo > hi)
00092             {
00093                 done:
00094                 if (lo >= max)
00095                 {
00096                     /*
00097                      * Many drivers have the same bug: they do not check
00098                      * data->index at all, with the result that they appear to
00099                      * support an infinite number of modulators.
00100                      */
00101                     return 1;
00102                 }
00103                 return lo;
00104             }
00105         }
00106         else if (errno != EINVAL)
00107         {
00108             return -1;
00109         }
00110         else
00111         {
00112             if (hi <= 0 && lo <= 0)
00113                 return -1;
00114             /* mid >= ninputs */
00115             hi = mid - 1;
00116             if (lo >= hi)
00117                 goto done;
00118         }
00119     }
00120 }
00121 
00122 #endif
00123 
00124 /* vim: set ts=8 sw=4 et : */