libexplain
1.4.D001
|
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_tuner.h> 00027 #include <libexplain/buffer/v4l2_tuner_audmode.h> 00028 #include <libexplain/buffer/v4l2_tuner_capability.h> 00029 #include <libexplain/buffer/v4l2_tuner_rxsubchans.h> 00030 #include <libexplain/buffer/v4l2_tuner_type.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_tuner(explain_string_buffer_t *sb, 00039 const struct v4l2_tuner *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 if (extra) 00050 { 00051 explain_string_buffer_puts(sb, ", name = "); 00052 explain_string_buffer_putsu_quoted_n(sb, data->name, 00053 sizeof(data->name)); 00054 explain_string_buffer_puts(sb, ", type = "); 00055 explain_buffer_v4l2_tuner_type(sb, data->type); 00056 explain_string_buffer_puts(sb, ", capability = "); 00057 explain_buffer_v4l2_tuner_capability(sb, data->capability); 00058 explain_string_buffer_puts(sb, ", rangelow = "); 00059 explain_buffer_uint32_t(sb, data->rangelow); 00060 explain_string_buffer_puts(sb, ", rangehigh = "); 00061 explain_buffer_uint32_t(sb, data->rangehigh); 00062 explain_string_buffer_puts(sb, ", rxsubchans = "); 00063 explain_buffer_v4l2_tuner_rxsubchans(sb, data->rxsubchans); 00064 explain_string_buffer_puts(sb, ", audmode = "); 00065 explain_buffer_v4l2_tuner_audmode(sb, data->audmode); 00066 explain_string_buffer_puts(sb, ", signal = "); 00067 explain_buffer_int32_t(sb, data->signal); 00068 explain_string_buffer_puts(sb, ", afc = "); 00069 explain_buffer_int32_t(sb, data->afc); 00070 if (!explain_uint32_array_all_zero(data->reserved, 00071 SIZEOF(data->reserved))) 00072 { 00073 explain_string_buffer_puts(sb, ", reserved = "); 00074 explain_buffer_uint32_array(sb, data->reserved, 00075 SIZEOF(data->reserved)); 00076 } 00077 } 00078 explain_string_buffer_puts(sb, " }"); 00079 } 00080 00081 00082 int 00083 explain_v4l2_tuner_get_n(int fildes) 00084 { 00085 const int max = 200; 00086 00087 int lo = 0; 00088 int hi = max; 00089 for (;;) 00090 { 00091 int mid = (lo + hi) / 2; 00092 struct v4l2_tuner qry; 00093 memset(&qry, 0, sizeof(qry)); 00094 qry.index = mid; 00095 if (ioctl(fildes, VIDIOC_G_TUNER, &qry) >= 0) 00096 { 00097 if (hi <= 0 && lo <= 0) 00098 return -1; 00099 /* mid < ninputs */ 00100 lo = mid + 1; 00101 if (lo > hi) 00102 { 00103 done: 00104 if (lo >= max) 00105 { 00106 /* 00107 * Many drivers have the same bug: they do not check 00108 * data->index at all, with the result that they appear to 00109 * support an infinite number of tuners. 00110 */ 00111 return 1; 00112 } 00113 return lo; 00114 } 00115 } 00116 else if (errno != EINVAL) 00117 { 00118 return -1; 00119 } 00120 else 00121 { 00122 if (hi <= 0 && lo <= 0) 00123 return -1; 00124 /* mid >= ninputs */ 00125 hi = mid - 1; 00126 if (lo >= hi) 00127 goto done; 00128 } 00129 } 00130 } 00131 00132 #endif 00133 00134 /* vim: set ts=8 sw=4 et : */