提交 8d2d3dd1 编写于 作者: A Andres Salomon 提交者: Greg Kroah-Hartman

staging: olpc_dcon: get rid of global i2c_client, create a dcon_priv struct

Rather than using the global i2c_client variable, create a dcon_priv
struct, store in the drvdata portion of the dev, and pass that around.

In order to access dcon struct from various callbacks, include
the reboot notifier and source switching work struct in the dcon struct.
Signed-off-by: NAndres Salomon <dilinger@queued.net>
Signed-off-by: NGreg Kroah-Hartman <gregkh@suse.de>
上级 fe2d5b43
...@@ -53,10 +53,16 @@ struct dcon_platform_data { ...@@ -53,10 +53,16 @@ struct dcon_platform_data {
static struct dcon_platform_data *pdata; static struct dcon_platform_data *pdata;
struct dcon_priv {
struct i2c_client *client;
struct work_struct switch_source;
struct notifier_block reboot_nb;
};
/* I2C structures */ /* I2C structures */
static struct i2c_driver dcon_driver; static struct i2c_driver dcon_driver;
static struct i2c_client *dcon_client;
/* Platform devices */ /* Platform devices */
static struct platform_device *dcon_device; static struct platform_device *dcon_device;
...@@ -93,16 +99,24 @@ static DECLARE_WAIT_QUEUE_HEAD(dcon_wait_queue); ...@@ -93,16 +99,24 @@ static DECLARE_WAIT_QUEUE_HEAD(dcon_wait_queue);
static unsigned short normal_i2c[] = { 0x0d, I2C_CLIENT_END }; static unsigned short normal_i2c[] = { 0x0d, I2C_CLIENT_END };
#define dcon_write(reg, val) i2c_smbus_write_word_data(dcon_client, reg, val) static s32 dcon_write(struct dcon_priv *dcon, u8 reg, u16 val)
#define dcon_read(reg) i2c_smbus_read_word_data(dcon_client, reg) {
return i2c_smbus_write_word_data(dcon->client, reg, val);
}
static s32 dcon_read(struct dcon_priv *dcon, u8 reg)
{
return i2c_smbus_read_word_data(dcon->client, reg);
}
/* The current backlight value - this saves us some smbus traffic */ /* The current backlight value - this saves us some smbus traffic */
static int bl_val = -1; static int bl_val = -1;
/* ===== API functions - these are called by a variety of users ==== */ /* ===== API functions - these are called by a variety of users ==== */
static int dcon_hw_init(struct i2c_client *client, int is_init) static int dcon_hw_init(struct dcon_priv *dcon, int is_init)
{ {
struct i2c_client *client = dcon->client;
uint16_t ver; uint16_t ver;
int rc = 0; int rc = 0;
...@@ -173,7 +187,7 @@ static int dcon_hw_init(struct i2c_client *client, int is_init) ...@@ -173,7 +187,7 @@ static int dcon_hw_init(struct i2c_client *client, int is_init)
* smbus. For newer models, we simply BUG(); we want to know if this * smbus. For newer models, we simply BUG(); we want to know if this
* still happens despite the power fixes that have been made! * still happens despite the power fixes that have been made!
*/ */
static int dcon_bus_stabilize(struct i2c_client *client, int is_powered_down) static int dcon_bus_stabilize(struct dcon_priv *dcon, int is_powered_down)
{ {
unsigned long timeout; unsigned long timeout;
int x; int x;
...@@ -194,7 +208,7 @@ static int dcon_bus_stabilize(struct i2c_client *client, int is_powered_down) ...@@ -194,7 +208,7 @@ static int dcon_bus_stabilize(struct i2c_client *client, int is_powered_down)
for (x = -1, timeout = 50; timeout && x < 0; timeout--) { for (x = -1, timeout = 50; timeout && x < 0; timeout--) {
msleep(1); msleep(1);
x = dcon_read(DCON_REG_ID); x = dcon_read(dcon, DCON_REG_ID);
} }
if (x < 0) { if (x < 0) {
printk(KERN_ERR "olpc-dcon: unable to stabilize dcon's " printk(KERN_ERR "olpc-dcon: unable to stabilize dcon's "
...@@ -208,51 +222,51 @@ static int dcon_bus_stabilize(struct i2c_client *client, int is_powered_down) ...@@ -208,51 +222,51 @@ static int dcon_bus_stabilize(struct i2c_client *client, int is_powered_down)
} }
if (is_powered_down) if (is_powered_down)
return dcon_hw_init(client, 0); return dcon_hw_init(dcon, 0);
return 0; return 0;
} }
static int dcon_get_backlight(void) static int dcon_get_backlight(struct dcon_priv *dcon)
{ {
if (dcon_client == NULL) if (!dcon || !dcon->client)
return 0; return 0;
if (bl_val == -1) if (bl_val == -1)
bl_val = dcon_read(DCON_REG_BRIGHT) & 0x0F; bl_val = dcon_read(dcon, DCON_REG_BRIGHT) & 0x0F;
return bl_val; return bl_val;
} }
static void dcon_set_backlight_hw(int level) static void dcon_set_backlight_hw(struct dcon_priv *dcon, int level)
{ {
bl_val = level & 0x0F; bl_val = level & 0x0F;
dcon_write(DCON_REG_BRIGHT, bl_val); dcon_write(dcon, DCON_REG_BRIGHT, bl_val);
/* Purposely turn off the backlight when we go to level 0 */ /* Purposely turn off the backlight when we go to level 0 */
if (bl_val == 0) { if (bl_val == 0) {
dcon_disp_mode &= ~MODE_BL_ENABLE; dcon_disp_mode &= ~MODE_BL_ENABLE;
dcon_write(DCON_REG_MODE, dcon_disp_mode); dcon_write(dcon, DCON_REG_MODE, dcon_disp_mode);
} else if (!(dcon_disp_mode & MODE_BL_ENABLE)) { } else if (!(dcon_disp_mode & MODE_BL_ENABLE)) {
dcon_disp_mode |= MODE_BL_ENABLE; dcon_disp_mode |= MODE_BL_ENABLE;
dcon_write(DCON_REG_MODE, dcon_disp_mode); dcon_write(dcon, DCON_REG_MODE, dcon_disp_mode);
} }
} }
static void dcon_set_backlight(int level) static void dcon_set_backlight(struct dcon_priv *dcon, int level)
{ {
if (dcon_client == NULL) if (!dcon || !dcon->client)
return; return;
if (bl_val == (level & 0x0F)) if (bl_val == (level & 0x0F))
return; return;
dcon_set_backlight_hw(level); dcon_set_backlight_hw(dcon, level);
} }
/* Set the output type to either color or mono */ /* Set the output type to either color or mono */
static int dcon_set_output(int arg) static int dcon_set_output(struct dcon_priv *dcon, int arg)
{ {
if (dcon_output == arg) if (dcon_output == arg)
return 0; return 0;
...@@ -269,7 +283,7 @@ static int dcon_set_output(int arg) ...@@ -269,7 +283,7 @@ static int dcon_set_output(int arg)
dcon_disp_mode |= MODE_COL_AA; dcon_disp_mode |= MODE_COL_AA;
} }
dcon_write(DCON_REG_MODE, dcon_disp_mode); dcon_write(dcon, DCON_REG_MODE, dcon_disp_mode);
return 0; return 0;
} }
...@@ -277,7 +291,7 @@ static int dcon_set_output(int arg) ...@@ -277,7 +291,7 @@ static int dcon_set_output(int arg)
* DCONLOAD works in a sleep and account for it accordingly * DCONLOAD works in a sleep and account for it accordingly
*/ */
static void dcon_sleep(int state) static void dcon_sleep(struct dcon_priv *dcon, int state)
{ {
int x; int x;
...@@ -301,7 +315,7 @@ static void dcon_sleep(int state) ...@@ -301,7 +315,7 @@ static void dcon_sleep(int state)
/* Only re-enable the backlight if the backlight value is set */ /* Only re-enable the backlight if the backlight value is set */
if (bl_val != 0) if (bl_val != 0)
dcon_disp_mode |= MODE_BL_ENABLE; dcon_disp_mode |= MODE_BL_ENABLE;
x = dcon_bus_stabilize(dcon_client, 1); x = dcon_bus_stabilize(dcon, 1);
if (x) if (x)
printk(KERN_WARNING "olpc-dcon: unable to reinit dcon" printk(KERN_WARNING "olpc-dcon: unable to reinit dcon"
" hardware: %d!\n", x); " hardware: %d!\n", x);
...@@ -309,7 +323,7 @@ static void dcon_sleep(int state) ...@@ -309,7 +323,7 @@ static void dcon_sleep(int state)
dcon_sleep_val = state; dcon_sleep_val = state;
/* Restore backlight */ /* Restore backlight */
dcon_set_backlight_hw(bl_val); dcon_set_backlight_hw(dcon, bl_val);
} }
/* We should turn off some stuff in the framebuffer - but what? */ /* We should turn off some stuff in the framebuffer - but what? */
...@@ -337,6 +351,8 @@ void dcon_load_holdoff(void) ...@@ -337,6 +351,8 @@ void dcon_load_holdoff(void)
static void dcon_source_switch(struct work_struct *work) static void dcon_source_switch(struct work_struct *work)
{ {
struct dcon_priv *dcon = container_of(work, struct dcon_priv,
switch_source);
DECLARE_WAITQUEUE(wait, current); DECLARE_WAITQUEUE(wait, current);
int source = dcon_pending; int source = dcon_pending;
...@@ -351,7 +367,8 @@ static void dcon_source_switch(struct work_struct *work) ...@@ -351,7 +367,8 @@ static void dcon_source_switch(struct work_struct *work)
case DCON_SOURCE_CPU: case DCON_SOURCE_CPU:
printk("dcon_source_switch to CPU\n"); printk("dcon_source_switch to CPU\n");
/* Enable the scanline interrupt bit */ /* Enable the scanline interrupt bit */
if (dcon_write(DCON_REG_MODE, dcon_disp_mode | MODE_SCAN_INT)) if (dcon_write(dcon, DCON_REG_MODE,
dcon_disp_mode | MODE_SCAN_INT))
printk(KERN_ERR printk(KERN_ERR
"olpc-dcon: couldn't enable scanline interrupt!\n"); "olpc-dcon: couldn't enable scanline interrupt!\n");
else { else {
...@@ -364,7 +381,7 @@ static void dcon_source_switch(struct work_struct *work) ...@@ -364,7 +381,7 @@ static void dcon_source_switch(struct work_struct *work)
printk(KERN_ERR "olpc-dcon: Timeout entering CPU mode; expect a screen glitch.\n"); printk(KERN_ERR "olpc-dcon: Timeout entering CPU mode; expect a screen glitch.\n");
/* Turn off the scanline interrupt */ /* Turn off the scanline interrupt */
if (dcon_write(DCON_REG_MODE, dcon_disp_mode)) if (dcon_write(dcon, DCON_REG_MODE, dcon_disp_mode))
printk(KERN_ERR "olpc-dcon: couldn't disable scanline interrupt!\n"); printk(KERN_ERR "olpc-dcon: couldn't disable scanline interrupt!\n");
/* /*
...@@ -454,40 +471,39 @@ static void dcon_source_switch(struct work_struct *work) ...@@ -454,40 +471,39 @@ static void dcon_source_switch(struct work_struct *work)
dcon_source = source; dcon_source = source;
} }
static DECLARE_WORK(dcon_work, dcon_source_switch); static void dcon_set_source(struct dcon_priv *dcon, int arg)
static void dcon_set_source(int arg)
{ {
if (dcon_pending == arg) if (dcon_pending == arg)
return; return;
dcon_pending = arg; dcon_pending = arg;
if ((dcon_source != arg) && !work_pending(&dcon_work)) if ((dcon_source != arg) && !work_pending(&dcon->switch_source))
schedule_work(&dcon_work); schedule_work(&dcon->switch_source);
} }
static void dcon_set_source_sync(int arg) static void dcon_set_source_sync(struct dcon_priv *dcon, int arg)
{ {
dcon_set_source(arg); dcon_set_source(dcon, arg);
flush_scheduled_work(); flush_scheduled_work();
} }
static int dconbl_set(struct backlight_device *dev) static int dconbl_set(struct backlight_device *dev)
{ {
struct dcon_priv *dcon = bl_get_data(dev);
int level = dev->props.brightness; int level = dev->props.brightness;
if (dev->props.power != FB_BLANK_UNBLANK) if (dev->props.power != FB_BLANK_UNBLANK)
level = 0; level = 0;
dcon_set_backlight(level); dcon_set_backlight(dcon, level);
return 0; return 0;
} }
static int dconbl_get(struct backlight_device *dev) static int dconbl_get(struct backlight_device *dev)
{ {
return dcon_get_backlight(); struct dcon_priv *dcon = bl_get_data(dev);
return dcon_get_backlight(dcon);
} }
static ssize_t dcon_mode_show(struct device *dev, static ssize_t dcon_mode_show(struct device *dev,
...@@ -548,7 +564,7 @@ static ssize_t dcon_output_store(struct device *dev, ...@@ -548,7 +564,7 @@ static ssize_t dcon_output_store(struct device *dev,
return -EINVAL; return -EINVAL;
if (output == DCON_OUTPUT_COLOR || output == DCON_OUTPUT_MONO) { if (output == DCON_OUTPUT_COLOR || output == DCON_OUTPUT_MONO) {
dcon_set_output(output); dcon_set_output(dev_get_drvdata(dev), output);
rc = count; rc = count;
} }
...@@ -558,6 +574,7 @@ static ssize_t dcon_output_store(struct device *dev, ...@@ -558,6 +574,7 @@ static ssize_t dcon_output_store(struct device *dev,
static ssize_t dcon_freeze_store(struct device *dev, static ssize_t dcon_freeze_store(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count) struct device_attribute *attr, const char *buf, size_t count)
{ {
struct dcon_priv *dcon = dev_get_drvdata(dev);
int output; int output;
if (_strtoul(buf, count, &output)) if (_strtoul(buf, count, &output))
...@@ -567,13 +584,13 @@ static ssize_t dcon_freeze_store(struct device *dev, ...@@ -567,13 +584,13 @@ static ssize_t dcon_freeze_store(struct device *dev,
switch (output) { switch (output) {
case 0: case 0:
dcon_set_source(DCON_SOURCE_CPU); dcon_set_source(dcon, DCON_SOURCE_CPU);
break; break;
case 1: case 1:
dcon_set_source_sync(DCON_SOURCE_DCON); dcon_set_source_sync(dcon, DCON_SOURCE_DCON);
break; break;
case 2: /* normally unused */ case 2: /* normally unused */
dcon_set_source(DCON_SOURCE_DCON); dcon_set_source(dcon, DCON_SOURCE_DCON);
break; break;
default: default:
return -EINVAL; return -EINVAL;
...@@ -592,7 +609,7 @@ static ssize_t dcon_resumeline_store(struct device *dev, ...@@ -592,7 +609,7 @@ static ssize_t dcon_resumeline_store(struct device *dev,
return rc; return rc;
resumeline = rl; resumeline = rl;
dcon_write(DCON_REG_SCAN_INT, resumeline); dcon_write(dev_get_drvdata(dev), DCON_REG_SCAN_INT, resumeline);
rc = count; rc = count;
return rc; return rc;
...@@ -606,7 +623,7 @@ static ssize_t dcon_sleep_store(struct device *dev, ...@@ -606,7 +623,7 @@ static ssize_t dcon_sleep_store(struct device *dev,
if (_strtoul(buf, count, &output)) if (_strtoul(buf, count, &output))
return -EINVAL; return -EINVAL;
dcon_sleep(output ? DCON_SLEEP : DCON_ACTIVE); dcon_sleep(dev_get_drvdata(dev), output ? DCON_SLEEP : DCON_ACTIVE);
return count; return count;
} }
...@@ -627,20 +644,17 @@ static const struct backlight_ops dcon_bl_ops = { ...@@ -627,20 +644,17 @@ static const struct backlight_ops dcon_bl_ops = {
static int dcon_reboot_notify(struct notifier_block *nb, static int dcon_reboot_notify(struct notifier_block *nb,
unsigned long foo, void *bar) unsigned long foo, void *bar)
{ {
if (dcon_client == NULL) struct dcon_priv *dcon = container_of(nb, struct dcon_priv, reboot_nb);
if (!dcon || !dcon->client)
return 0; return 0;
/* Turn off the DCON. Entirely. */ /* Turn off the DCON. Entirely. */
dcon_write(DCON_REG_MODE, 0x39); dcon_write(dcon, DCON_REG_MODE, 0x39);
dcon_write(DCON_REG_MODE, 0x32); dcon_write(dcon, DCON_REG_MODE, 0x32);
return 0; return 0;
} }
static struct notifier_block dcon_nb = {
.notifier_call = dcon_reboot_notify,
.priority = -1,
};
static int unfreeze_on_panic(struct notifier_block *nb, static int unfreeze_on_panic(struct notifier_block *nb,
unsigned long e, void *p) unsigned long e, void *p)
{ {
...@@ -660,11 +674,14 @@ static int fb_notifier_callback(struct notifier_block *self, ...@@ -660,11 +674,14 @@ static int fb_notifier_callback(struct notifier_block *self,
unsigned long event, void *data) unsigned long event, void *data)
{ {
struct fb_event *evdata = data; struct fb_event *evdata = data;
struct backlight_device *bl = container_of(self,
struct backlight_device, fb_notif);
struct dcon_priv *dcon = bl_get_data(bl);
int *blank = (int *) evdata->data; int *blank = (int *) evdata->data;
if (((event != FB_EVENT_BLANK) && (event != FB_EVENT_CONBLANK)) || if (((event != FB_EVENT_BLANK) && (event != FB_EVENT_CONBLANK)) ||
ignore_fb_events) ignore_fb_events)
return 0; return 0;
dcon_sleep((*blank) ? DCON_SLEEP : DCON_ACTIVE); dcon_sleep(dcon, (*blank) ? DCON_SLEEP : DCON_ACTIVE);
return 0; return 0;
} }
...@@ -681,12 +698,24 @@ static int dcon_detect(struct i2c_client *client, struct i2c_board_info *info) ...@@ -681,12 +698,24 @@ static int dcon_detect(struct i2c_client *client, struct i2c_board_info *info)
static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id) static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id)
{ {
struct dcon_priv *dcon;
int rc, i, j; int rc, i, j;
dcon = kzalloc(sizeof(*dcon), GFP_KERNEL);
if (!dcon)
return -ENOMEM;
dcon->client = client;
INIT_WORK(&dcon->switch_source, dcon_source_switch);
dcon->reboot_nb.notifier_call = dcon_reboot_notify;
dcon->reboot_nb.priority = -1;
i2c_set_clientdata(client, dcon);
if (num_registered_fb >= 1) if (num_registered_fb >= 1)
fbinfo = registered_fb[0]; fbinfo = registered_fb[0];
rc = dcon_hw_init(client, 1); rc = dcon_hw_init(dcon, 1);
if (rc) if (rc)
goto einit; goto einit;
...@@ -699,9 +728,8 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id) ...@@ -699,9 +728,8 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id)
rc = -ENOMEM; rc = -ENOMEM;
goto eirq; goto eirq;
} }
/* Place holder...*/
i2c_set_clientdata(client, dcon_device);
rc = platform_device_add(dcon_device); rc = platform_device_add(dcon_device);
platform_set_drvdata(dcon_device, dcon);
if (rc) { if (rc) {
printk(KERN_ERR "dcon: Unable to add the DCON device\n"); printk(KERN_ERR "dcon: Unable to add the DCON device\n");
...@@ -718,11 +746,8 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id) ...@@ -718,11 +746,8 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id)
} }
/* Add the backlight device for the DCON */ /* Add the backlight device for the DCON */
dcon_client = client;
dcon_bl_dev = backlight_device_register("dcon-bl", &dcon_device->dev, dcon_bl_dev = backlight_device_register("dcon-bl", &dcon_device->dev,
NULL, &dcon_bl_ops, NULL); dcon, &dcon_bl_ops, NULL);
if (IS_ERR(dcon_bl_dev)) { if (IS_ERR(dcon_bl_dev)) {
printk(KERN_ERR "Cannot register the backlight device (%ld)\n", printk(KERN_ERR "Cannot register the backlight device (%ld)\n",
...@@ -731,12 +756,12 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id) ...@@ -731,12 +756,12 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id)
} else { } else {
dcon_bl_dev->props.max_brightness = 15; dcon_bl_dev->props.max_brightness = 15;
dcon_bl_dev->props.power = FB_BLANK_UNBLANK; dcon_bl_dev->props.power = FB_BLANK_UNBLANK;
dcon_bl_dev->props.brightness = dcon_get_backlight(); dcon_bl_dev->props.brightness = dcon_get_backlight(dcon);
backlight_update_status(dcon_bl_dev); backlight_update_status(dcon_bl_dev);
} }
register_reboot_notifier(&dcon_nb); register_reboot_notifier(&dcon->reboot_nb);
atomic_notifier_chain_register(&panic_notifier_list, &dcon_panic_nb); atomic_notifier_chain_register(&panic_notifier_list, &dcon_panic_nb);
fb_register_client(&fb_nb); fb_register_client(&fb_nb);
...@@ -751,15 +776,19 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id) ...@@ -751,15 +776,19 @@ static int dcon_probe(struct i2c_client *client, const struct i2c_device_id *id)
eirq: eirq:
free_irq(DCON_IRQ, &dcon_driver); free_irq(DCON_IRQ, &dcon_driver);
einit: einit:
i2c_set_clientdata(client, NULL);
kfree(dcon);
return rc; return rc;
} }
static int dcon_remove(struct i2c_client *client) static int dcon_remove(struct i2c_client *client)
{ {
dcon_client = NULL; struct dcon_priv *dcon = i2c_get_clientdata(client);
i2c_set_clientdata(client, NULL);
fb_unregister_client(&fb_nb); fb_unregister_client(&fb_nb);
unregister_reboot_notifier(&dcon_nb); unregister_reboot_notifier(&dcon->reboot_nb);
atomic_notifier_chain_unregister(&panic_notifier_list, &dcon_panic_nb); atomic_notifier_chain_unregister(&panic_notifier_list, &dcon_panic_nb);
free_irq(DCON_IRQ, &dcon_driver); free_irq(DCON_IRQ, &dcon_driver);
...@@ -769,7 +798,9 @@ static int dcon_remove(struct i2c_client *client) ...@@ -769,7 +798,9 @@ static int dcon_remove(struct i2c_client *client)
if (dcon_device != NULL) if (dcon_device != NULL)
platform_device_unregister(dcon_device); platform_device_unregister(dcon_device);
cancel_work_sync(&dcon_work); cancel_work_sync(&dcon->switch_source);
kfree(dcon);
return 0; return 0;
} }
...@@ -777,9 +808,11 @@ static int dcon_remove(struct i2c_client *client) ...@@ -777,9 +808,11 @@ static int dcon_remove(struct i2c_client *client)
#ifdef CONFIG_PM #ifdef CONFIG_PM
static int dcon_suspend(struct i2c_client *client, pm_message_t state) static int dcon_suspend(struct i2c_client *client, pm_message_t state)
{ {
struct dcon_priv *dcon = i2c_get_clientdata(client);
if (dcon_sleep_val == DCON_ACTIVE) { if (dcon_sleep_val == DCON_ACTIVE) {
/* Set up the DCON to have the source */ /* Set up the DCON to have the source */
dcon_set_source_sync(DCON_SOURCE_DCON); dcon_set_source_sync(dcon, DCON_SOURCE_DCON);
} }
return 0; return 0;
...@@ -787,9 +820,11 @@ static int dcon_suspend(struct i2c_client *client, pm_message_t state) ...@@ -787,9 +820,11 @@ static int dcon_suspend(struct i2c_client *client, pm_message_t state)
static int dcon_resume(struct i2c_client *client) static int dcon_resume(struct i2c_client *client)
{ {
struct dcon_priv *dcon = i2c_get_clientdata(client);
if (dcon_sleep_val == DCON_ACTIVE) { if (dcon_sleep_val == DCON_ACTIVE) {
dcon_bus_stabilize(client, 0); dcon_bus_stabilize(dcon, 0);
dcon_set_source(DCON_SOURCE_CPU); dcon_set_source(dcon, DCON_SOURCE_CPU);
} }
return 0; return 0;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册