|
@@ -1226,41 +1226,66 @@ int enable_cmf(struct ccw_device *cdev)
|
|
{
|
|
{
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
|
|
+ device_lock(&cdev->dev);
|
|
ret = cmbops->alloc(cdev);
|
|
ret = cmbops->alloc(cdev);
|
|
- cmbops->reset(cdev);
|
|
|
|
if (ret)
|
|
if (ret)
|
|
- return ret;
|
|
|
|
|
|
+ goto out;
|
|
|
|
+ cmbops->reset(cdev);
|
|
|
|
+ ret = sysfs_create_group(&cdev->dev.kobj, cmbops->attr_group);
|
|
|
|
+ if (ret) {
|
|
|
|
+ cmbops->free(cdev);
|
|
|
|
+ goto out;
|
|
|
|
+ }
|
|
ret = cmbops->set(cdev, 2);
|
|
ret = cmbops->set(cdev, 2);
|
|
if (ret) {
|
|
if (ret) {
|
|
|
|
+ sysfs_remove_group(&cdev->dev.kobj, cmbops->attr_group);
|
|
cmbops->free(cdev);
|
|
cmbops->free(cdev);
|
|
- return ret;
|
|
|
|
}
|
|
}
|
|
- ret = sysfs_create_group(&cdev->dev.kobj, cmbops->attr_group);
|
|
|
|
- if (!ret)
|
|
|
|
- return 0;
|
|
|
|
- cmbops->set(cdev, 0); //FIXME: this can fail
|
|
|
|
- cmbops->free(cdev);
|
|
|
|
|
|
+out:
|
|
|
|
+ device_unlock(&cdev->dev);
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
- * disable_cmf() - switch off the channel measurement for a specific device
|
|
|
|
|
|
+ * __disable_cmf() - switch off the channel measurement for a specific device
|
|
* @cdev: The ccw device to be disabled
|
|
* @cdev: The ccw device to be disabled
|
|
*
|
|
*
|
|
* Returns %0 for success or a negative error value.
|
|
* Returns %0 for success or a negative error value.
|
|
*
|
|
*
|
|
* Context:
|
|
* Context:
|
|
- * non-atomic
|
|
|
|
|
|
+ * non-atomic, device_lock() held.
|
|
*/
|
|
*/
|
|
-int disable_cmf(struct ccw_device *cdev)
|
|
|
|
|
|
+int __disable_cmf(struct ccw_device *cdev)
|
|
{
|
|
{
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
ret = cmbops->set(cdev, 0);
|
|
ret = cmbops->set(cdev, 0);
|
|
if (ret)
|
|
if (ret)
|
|
return ret;
|
|
return ret;
|
|
- cmbops->free(cdev);
|
|
|
|
|
|
+
|
|
sysfs_remove_group(&cdev->dev.kobj, cmbops->attr_group);
|
|
sysfs_remove_group(&cdev->dev.kobj, cmbops->attr_group);
|
|
|
|
+ cmbops->free(cdev);
|
|
|
|
+
|
|
|
|
+ return ret;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * disable_cmf() - switch off the channel measurement for a specific device
|
|
|
|
+ * @cdev: The ccw device to be disabled
|
|
|
|
+ *
|
|
|
|
+ * Returns %0 for success or a negative error value.
|
|
|
|
+ *
|
|
|
|
+ * Context:
|
|
|
|
+ * non-atomic
|
|
|
|
+ */
|
|
|
|
+int disable_cmf(struct ccw_device *cdev)
|
|
|
|
+{
|
|
|
|
+ int ret;
|
|
|
|
+
|
|
|
|
+ device_lock(&cdev->dev);
|
|
|
|
+ ret = __disable_cmf(cdev);
|
|
|
|
+ device_unlock(&cdev->dev);
|
|
|
|
+
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
|