Commit 5f3d2898 authored by malattia@linux.it's avatar malattia@linux.it Committed by Len Brown

sony-laptop: add camera enable/disable parameter, better handle possible infinite loop

Use a parameter to enable/disable motion eye camera (for C1VE/C1VN models)
controls and avoid entering an infinite loop if the camera is not present
and the HW doesn't answer as we expect on io commands.
Signed-off-by: default avatarMattia Dongili <malattia@linux.it>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent 7b153f36
...@@ -107,6 +107,12 @@ module_param(mask, ulong, 0644); ...@@ -107,6 +107,12 @@ module_param(mask, ulong, 0644);
MODULE_PARM_DESC(mask, MODULE_PARM_DESC(mask,
"set this to the mask of event you want to enable (see doc)"); "set this to the mask of event you want to enable (see doc)");
static int camera; /* = 0 */
module_param(camera, int, 0444);
MODULE_PARM_DESC(camera,
"set this to 1 to enable Motion Eye camera controls "
"(only use it if you have a C1VE or C1VN model)");
#ifdef CONFIG_SONY_LAPTOP_OLD #ifdef CONFIG_SONY_LAPTOP_OLD
static int minor = -1; static int minor = -1;
module_param(minor, int, 0); module_param(minor, int, 0);
...@@ -1226,29 +1232,39 @@ static int sony_pic_camera_ready(void) ...@@ -1226,29 +1232,39 @@ static int sony_pic_camera_ready(void)
return (v != 0xff && (v & SONYPI_CAMERA_STATUS_READY)); return (v != 0xff && (v & SONYPI_CAMERA_STATUS_READY));
} }
static void sony_pic_camera_off(void) static int sony_pic_camera_off(void)
{ {
if (!camera) {
printk(KERN_WARNING DRV_PFX "camera control not enabled\n");
return -ENODEV;
}
wait_on_command(sony_pic_call3(0x90, SONYPI_CAMERA_PICTURE, wait_on_command(sony_pic_call3(0x90, SONYPI_CAMERA_PICTURE,
SONYPI_CAMERA_MUTE_MASK), SONYPI_CAMERA_MUTE_MASK),
ITERATIONS_SHORT); ITERATIONS_SHORT);
if (!spic_dev.camera_power) if (spic_dev.camera_power) {
return;
sony_pic_call2(0x91, 0); sony_pic_call2(0x91, 0);
spic_dev.camera_power = 0; spic_dev.camera_power = 0;
}
return 0;
} }
static void sony_pic_camera_on(void) static int sony_pic_camera_on(void)
{ {
int i, j; int i, j, x;
if (!camera) {
printk(KERN_WARNING DRV_PFX "camera control not enabled\n");
return -ENODEV;
}
if (spic_dev.camera_power) if (spic_dev.camera_power)
return; return 0;
for (j = 5; j > 0; j--) { for (j = 5; j > 0; j--) {
while (sony_pic_call2(0x91, 0x1)) for (x = 0; x < 100 && sony_pic_call2(0x91, 0x1); x++)
msleep(10); msleep(10);
sony_pic_call1(0x93); sony_pic_call1(0x93);
...@@ -1262,8 +1278,8 @@ static void sony_pic_camera_on(void) ...@@ -1262,8 +1278,8 @@ static void sony_pic_camera_on(void)
} }
if (j == 0) { if (j == 0) {
printk(KERN_WARNING "sonypi: failed to power on camera\n"); printk(KERN_WARNING DRV_PFX "failed to power on camera\n");
return; return -ENODEV;
} }
wait_on_command(sony_pic_call3(0x90, SONYPI_CAMERA_CONTROL, wait_on_command(sony_pic_call3(0x90, SONYPI_CAMERA_CONTROL,
...@@ -1271,6 +1287,7 @@ static void sony_pic_camera_on(void) ...@@ -1271,6 +1287,7 @@ static void sony_pic_camera_on(void)
ITERATIONS_SHORT); ITERATIONS_SHORT);
spic_dev.camera_power = 1; spic_dev.camera_power = 1;
return 0;
} }
static ssize_t sony_pic_camerapower_store(struct device *dev, static ssize_t sony_pic_camerapower_store(struct device *dev,
...@@ -1278,14 +1295,18 @@ static ssize_t sony_pic_camerapower_store(struct device *dev, ...@@ -1278,14 +1295,18 @@ static ssize_t sony_pic_camerapower_store(struct device *dev,
const char *buffer, size_t count) const char *buffer, size_t count)
{ {
unsigned long value; unsigned long value;
int result;
if (count > 31) if (count > 31)
return -EINVAL; return -EINVAL;
value = simple_strtoul(buffer, NULL, 10); value = simple_strtoul(buffer, NULL, 10);
if (value) if (value)
sony_pic_camera_on(); result = sony_pic_camera_on();
else else
sony_pic_camera_off(); result = sony_pic_camera_off();
if (result)
return result;
return count; return count;
} }
...@@ -1662,7 +1683,7 @@ static int sonypi_compat_init(void) ...@@ -1662,7 +1683,7 @@ static int sonypi_compat_init(void)
goto err_free_kfifo; goto err_free_kfifo;
} }
if (minor == -1) if (minor == -1)
printk(KERN_INFO "sonypi: device allocated minor is %d\n", printk(KERN_INFO DRV_PFX "device allocated minor is %d\n",
sonypi_misc_device.minor); sonypi_misc_device.minor);
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment