/* * v4l-test: Test environment for Video For Linux Two API * * 4 Apr 2009 0.5 Test case for NULL parameter reworked * 22 Mar 2009 0.4 Cleanup dprintf() outputs and ret and errno names * 9 Feb 2009 0.3 Modify test cases to support drivers without any inputs * 22 Dec 2008 0.2 Test case with NULL parameter added * 18 Dec 2008 0.1 First release * * Written by Márton Németh * Released under GPL */ #include #include #include #include #include #include #include #include #include #include #include #include "v4l2_test.h" #include "dev_video.h" #include "video_limits.h" #include "test_VIDIOC_INPUT.h" int valid_input_index(int f, __u32 index) { __u32 i; struct v4l2_input input; int ret_enum, errno_enum; int valid = 0; /* Search for index with VIDIOC_ENUMINPUT. Do not just * check the given index with VIDIOC_ENUMINPUT because * we could miss the end of the enumeration. After the * end of enumeration no more indexes are allowed. */ i = 0; do { memset(&input, 0xff, sizeof(input)); input.index = i; ret_enum = ioctl(f, VIDIOC_ENUMINPUT, &input); errno_enum = errno; if (ret_enum == 0 && index == i) { valid = 1; break; } i++; } while (ret_enum == 0 && i != 0); return valid; } void test_VIDIOC_G_INPUT() { int ret_get, errno_get; __u32 index; int f; f = get_video_fd(); memset(&index, 0xff, sizeof(index)); ret_get = ioctl(f, VIDIOC_G_INPUT, &index); errno_get = errno; dprintf("\tVIDIOC_G_INPUT, ret_get=%i, errno_get=%i\n", ret_get, errno_get); if (ret_get == 0) { CU_ASSERT_EQUAL(ret_get, 0); CU_ASSERT(valid_input_index(f, index)); dprintf("\tindex=0x%X\n", index); } else { CU_ASSERT_EQUAL(ret_get, -1); CU_ASSERT_EQUAL(errno_get, EINVAL); } } void test_VIDIOC_S_INPUT_from_enum() { int ret_get, errno_get; int ret_enum, errno_enum; int ret_set, errno_set; __u32 input_index_orig; struct v4l2_input input; __u32 i; int f; f = get_video_fd(); memset(&input_index_orig, 0xff, sizeof(input_index_orig)); ret_get = ioctl(f, VIDIOC_G_INPUT, &input_index_orig); errno_get = errno; if (ret_get == 0) { CU_ASSERT_EQUAL(ret_get, 0); i = 0; do { memset(&input, 0xff, sizeof(input)); input.index = i; ret_enum = ioctl(f, VIDIOC_ENUMINPUT, &input); errno_enum = errno; dprintf("\tENUMINPUT: i=%u, ret_enum=%i, errno=%i\n", i, ret_enum, errno); if (ret_enum == 0) { ret_set = ioctl(f, VIDIOC_S_INPUT, &input.index); errno_set = errno; CU_ASSERT_EQUAL(ret_set, 0); dprintf ("\tinput.index=0x%X, ret_set=%i, errno_set=%i\n", input.index, ret_set, errno_set); } i++; } while (ret_enum == 0 && i != 0); /* Setting the original input_id should not fail */ ret_set = ioctl(f, VIDIOC_S_INPUT, &input_index_orig); CU_ASSERT_EQUAL(ret_set, 0); } else { CU_ASSERT_EQUAL(ret_get, -1); CU_ASSERT_EQUAL(errno_get, EINVAL); } } static void do_set_input(int f, __u32 first_wrong_input, __u32 index) { struct v4l2_input input; int ret_set, errno_set; if (first_wrong_input <= index) { dprintf("\tdo_set_input(f, 0x%X, 0x%X)\n", first_wrong_input, index); memset(&input, 0xff, sizeof(input)); input.index = index; ret_set = ioctl(f, VIDIOC_S_INPUT, &input.index); errno_set = errno; CU_ASSERT_EQUAL(ret_set, -1); CU_ASSERT_EQUAL(errno_set, EINVAL); dprintf("\t%s:%u: input.index=0x%X, ret_set=%i, errno_set=%i\n", __FILE__, __LINE__, input.index, ret_set, errno_set); } } void test_VIDIOC_S_INPUT_invalid_inputs() { int ret_get, errno_get; int ret_enum, errno_enum; int ret_set, errno_set; __u32 input_index_orig; struct v4l2_input input; __u32 i, first_wrong_input; int f; f = get_video_fd(); memset(&input_index_orig, 0xff, sizeof(input_index_orig)); ret_get = ioctl(f, VIDIOC_G_INPUT, &input_index_orig); errno_get = errno; if (ret_get == 0) { CU_ASSERT_EQUAL(ret_get, 0); i = 0; do { memset(&input, 0xff, sizeof(input)); input.index = i; ret_enum = ioctl(f, VIDIOC_ENUMINPUT, &input); errno_enum = errno; dprintf ("\t%s:%u: ENUMINPUT: i=%u, ret_enum=%i, errno_enum=%i\n", __FILE__, __LINE__, i, ret_enum, errno_enum); i++; } while (ret_enum == 0 && i != 0); if (i != 0) { first_wrong_input = i; /* The input index range 0..(i-1) are valid inputs. */ /* Try index values from range i..U32_MAX */ do_set_input(f, first_wrong_input, i); do_set_input(f, first_wrong_input, i + 1); /* Check for signed/unsigned mismatch near S32_MAX */ for (i = 0; i <= first_wrong_input + 1; i++) { do_set_input(f, first_wrong_input, ((__u32) S32_MAX) + i); } i = (U32_MAX - 1) - first_wrong_input; do { do_set_input(f, first_wrong_input, i); i++; } while (i != 0); } /* Setting the original input_id should not fail */ ret_set = ioctl(f, VIDIOC_S_INPUT, &input_index_orig); errno_set = errno; CU_ASSERT_EQUAL(ret_set, 0); } else { CU_ASSERT_EQUAL(ret_get, -1); CU_ASSERT_EQUAL(errno_get, EINVAL); } } void test_VIDIOC_G_INPUT_NULL() { int ret_get, errno_get; int ret_null, errno_null; __u32 index; memset(&index, 0xff, sizeof(index)); ret_get = ioctl(get_video_fd(), VIDIOC_G_INPUT, &index); errno_get = errno; dprintf("\t%s:%u: VIDIOC_G_INPUT, ret_get=%i, errno_get=%i\n", __FILE__, __LINE__, ret_get, errno_get); ret_null = ioctl(get_video_fd(), VIDIOC_G_INPUT, NULL); errno_null = errno; dprintf("\t%s:%u: VIDIOC_G_INPUT: ret_null=%i, errno_null=%i\n", __FILE__, __LINE__, ret_null, errno_null); if (ret_get == 0) { CU_ASSERT_EQUAL(ret_get, 0); CU_ASSERT_EQUAL(ret_null, -1); CU_ASSERT_EQUAL(errno_null, EFAULT); } else { CU_ASSERT_EQUAL(ret_get, -1); CU_ASSERT_EQUAL(errno_get, EINVAL); CU_ASSERT_EQUAL(ret_null, -1); CU_ASSERT_EQUAL(errno_null, EINVAL); } } void test_VIDIOC_S_INPUT_NULL() { int ret_orig, errno_orig; int ret_set, errno_set; int ret_null, errno_null; __u32 index_orig; __u32 index; /* save the original input */ memset(&index_orig, 0, sizeof(index_orig)); ret_orig = ioctl(get_video_fd(), VIDIOC_G_INPUT, &index_orig); errno_orig = errno; dprintf("\t%s:%u: VIDIOC_G_INPUT, ret_orig=%i, errno_orig=%i\n", __FILE__, __LINE__, ret_orig, errno_orig); memset(&index, 0xff, sizeof(index)); index = index_orig; ret_set = ioctl(get_video_fd(), VIDIOC_S_INPUT, &index); errno_set = errno; dprintf("\t%s:%u: VIDIOC_S_INPUT ret_set=%i, errno_set=%i\n", __FILE__, __LINE__, ret_set, errno_set); ret_null = ioctl(get_video_fd(), VIDIOC_S_INPUT, NULL); errno_null = errno; dprintf("\t%s:%u: VIDIOC_S_INPUT: ret_null=%i, errno_null=%i\n", __FILE__, __LINE__, ret_null, errno_null); if (ret_set == 0) { CU_ASSERT_EQUAL(ret_set, 0); CU_ASSERT_EQUAL(ret_null, -1); CU_ASSERT_EQUAL(errno_null, EFAULT); } else { CU_ASSERT_EQUAL(ret_set, -1); CU_ASSERT_EQUAL(errno_set, EINVAL); CU_ASSERT_EQUAL(ret_null, -1); CU_ASSERT_EQUAL(errno_null, EINVAL); } }