- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在制作一个每秒进行一次 acpi 调用的 Linux 模块(目前仅持续 20 秒)。我希望它继续每秒进行 acpi 调用,直到它被删除。正如我所拥有的,我将模块放入循环中,如果我确实设置了此循环永远重复,则我无法使用 rmmod 来删除模块。有没有办法为循环设置全局变量?
代码:acpi_call.ko
/* Copyright (c) 2010: Michal Kottman */
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/proc_fs.h>
#include <linux/slab.h>
#include <asm/uaccess.h>
#include <acpi/acpi.h>
#include <linux/jiffies.h>
#include <linux/sched.h>
MODULE_LICENSE("GPL");
#define BUFFER_SIZE 256
extern struct proc_dir_entry *acpi_root_dir;
char result_buffer[BUFFER_SIZE];
void do_acpi_call(void);
size_t get_avail_bytes(void)
{
return BUFFER_SIZE - strlen(result_buffer);
}
char *get_buffer_end(void)
{
return result_buffer + strlen(result_buffer);
}
int __init init_battcheck(void)
{
int i;
if( true )
{
for (i=0 ; i < 1 ; i++)
{
do_acpi_call();
if(result_buffer[3] == '1')
printk(KERN_INFO "Battery is discharging. %c\n", result_buffer[3]);
else if(result_buffer[3] == '2')
printk(KERN_INFO "Battery is charging. %c\n", result_buffer[3]);
else
printk(KERN_INFO "Battery is CRITICAL. %c\n", result_buffer[3]);
}
}
return 1;
}
/** Appends the contents of an acpi_object to the result buffer
@param result: An acpi object holding result data
@returns: 0 if the result could fully be saved, a higher value otherwise **/
int acpi_result_to_string(union acpi_object *result)
{
if (result->type == ACPI_TYPE_INTEGER)
{
snprintf(get_buffer_end(), get_avail_bytes(),"0x%x", (int)result->integer.value);
}
else if (result->type == ACPI_TYPE_STRING)
{
snprintf(get_buffer_end(), get_avail_bytes(), "\"%*s\"", result->string.length, result->string.pointer);
}
else if (result->type == ACPI_TYPE_BUFFER)
{
int i;
// do not store more than data if it does not fit. The first element is
// just 4 chars, but there is also two bytes from the curly brackets
int show_values = min(result->buffer.length, get_avail_bytes() / 6);
sprintf(get_buffer_end(), "{");
for (i = 0; i < show_values; i++)
sprintf(get_buffer_end(), i == 0 ? "0x%02x" : ", 0x%02x", result->buffer.pointer[i]);
if (result->buffer.length > show_values)
{
// if data was truncated, show a trailing comma if there is space
snprintf(get_buffer_end(), get_avail_bytes(), ",");
return 1;
}
else
{
// in case show_values == 0, but the buffer is too small to hold
// more values (i.e. the buffer cannot have anything more than "{")
snprintf(get_buffer_end(), get_avail_bytes(), "}");
}
}
else if (result->type == ACPI_TYPE_PACKAGE)
{
int i;
sprintf(get_buffer_end(), "[");
for (i=0; i < result->package.count; i++)
{
if (i > 0)
snprintf(get_buffer_end(), get_avail_bytes(), ", ");
// abort if there is no more space available
if (!get_avail_bytes() || acpi_result_to_string(&result->package.elements[i]))
return 1;
}
snprintf(get_buffer_end(), get_avail_bytes(), "]");
}
else
{
snprintf(get_buffer_end(), get_avail_bytes(), "Object type 0x%x\n", result->type);
}
// return 0 if there are still bytes available, 1 otherwise
return !get_avail_bytes();
}
void do_acpi_call(void)
{
acpi_status status;
acpi_handle handle;
struct acpi_object_list arg;
struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
printk(KERN_INFO "acpi_call: Calling \\_SB.PCI0.LPCB.EC0.BAT0._BST\n");
// get the handle of the method, must be a fully qualified path
status = acpi_get_handle(NULL, (acpi_string) "\\_SB.PCI0.LPCB.EC0.BAT0._BST", &handle);
if (ACPI_FAILURE(status))
{
snprintf(result_buffer, BUFFER_SIZE, "Error: %s", acpi_format_exception(status));
printk(KERN_ERR "acpi_call: Cannot get handle: %s\n", result_buffer);
return;
}
// prepare parameters
arg.count = 0;
arg.pointer = NULL;
// call the method
status = acpi_evaluate_object(handle, NULL, &arg, &buffer);
if (ACPI_FAILURE(status))
{
snprintf(result_buffer, BUFFER_SIZE,"Error: %s", acpi_format_exception(status));
printk(KERN_ERR "acpi_call: Method call failed: %s\n", result_buffer);
return;
}
// reset the result buffer
*result_buffer = '\0';
acpi_result_to_string(buffer.pointer);
kfree(buffer.pointer);
printk(KERN_INFO "acpi_call: Call successful: %s\n", result_buffer);
}
/** module initialization function */
int __init init_acpi_call(void)
{
struct proc_dir_entry *acpi_entry = create_proc_entry("call", 0660, acpi_root_dir);
strcpy(result_buffer, "not called");
if (acpi_entry == NULL)
{
printk(KERN_ERR "acpi_call: Couldn't create proc entry\n");
return -ENOMEM;
}
printk(KERN_INFO "acpi_call: Module loaded successfully\n");
init_battcheck();
return 0;
}
void __exit unload_acpi_call(void)
{
remove_proc_entry("call", acpi_root_dir);
printk(KERN_INFO "acpi_call: Module unloaded successfully\n");
}
module_init(init_acpi_call);
module_exit(unload_acpi_call);
编辑:我刚刚意识到每次调用 do_acpi_call 时我都不需要等待一秒钟,我让它这样做只是为了检查错误。这只是我正在制作的一个例子。我问是否有一种方法可以删除我所描述的循环中的模块?
最佳答案
设置并使用工作队列。这将安排您想要在未来一段时间内运行的工作。在 Linux 设备驱动程序中阅读相关内容: http://lwn.net/images/pdf/LDD3/ch07.pdf
当异步任务开始在您的内核模块中运行时,您需要非常小心地避免竞争条件并仔细清理。也就是说,在模块删除时,取消任何未完成的工作,等待任何正在进行的作业完成(确保作业不会重新安排自身),然后销毁工作队列,然后才关闭工作函数可能想要使用的任何资源。
关于Linux 模块每秒检查电池状态,更好的检查方式,当前方式 rmmod 不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12380924/
我正在通读 Windows Phone 7.5 Unleashed,有很多代码看起来像这样(在页面的代码隐藏中): bool loaded; protected override void OnNav
在cgi服务器中,我这样返回 print ('Status: 201 Created') print ('Content-Type: text/html') print ('Location: htt
我正在查看 esh(easy shell)的实现,无法理解在这种情况下什么是 22 和 9 信号。理想情况下,有一个更具描述性的常量,但我找不到列表。 最佳答案 信号列表及其编号(包括您看到的这两个)
我的Oozie Hive Action 永远处于运行模式。 oozie.log文件中没有显示错误。
我正在编写一个使用 RFCOMM 通过蓝牙连接到设备的 Android 应用程序。我使用 BluetoothChat 示例作为建立连接的基础,大部分时间一切正常。 但是,有时由于出现套接字已打开的消息
我有一个云调度程序作业,它应该每小时访问我的 API 以更新一些价格。这些作业大约需要 80 秒才能运行。 这是它的作用: POST https://www.example.com/api/jobs/
我正在 Tomcat 上访问一个简单的 JSP 页面: 但是当我使用 curl 测试此页面时,我得到了 200 响应代码而不是预期的 202: $ curl -i "http://localhos
有时 JAR-RS 客户端会发送错误的语法请求正文。服务器应响应 HTTP status 400 (Bad Request) , 但它以 HTTP status 500 (Internal Serve
我正在尝试通过 response.send() 发送一个整数,但我不断收到此错误 express deprecated res.send(status): Use res.sendStatus(sta
我已经用 Excel 和 Java 做过很多次了……这次我需要用 Stata 来做,因为保存变量更方便'labels .如何将 dataset_1 重组为下面的 dataset_2? 我需要转换以下
我正在创建一个应用程序,其中的对象具有状态查找功能。为了提供一些上下文,让我们使用以下示例。 帮助台应用程序,其中创建作业并通过以下工作流程移动: 新 - 工作已创建但未分配 进行中 - 分配给工作人
我想在 Keras 中运行 LSTM 并获得输出和状态。在 TF 中有这样的事情 with tf.variable_scope("RNN"): for time_step in range
有谁知道 Scala-GWT 的当前状态 项目? 那里的主要作者 Grzegorz Kossakowski 似乎退出了这个项目,在 Spring 中从事 scalac 的工作。 但是,在 interv
我正在尝试编写一个 super 简单的 applescript 来启动 OneDrive App , 或确保打开,当机器的电源设置为插入时,将退出,或确保关闭,当电源设置为电池时。 我无法找到如何访问
目前我正在做这样的事情 link.on('click', function () { if (link.attr('href') !== $route.current.originalPath
是否可以仅通过查看用户代理来检测浏览器上是否启用/禁用 Javascript。 如果是,我应该寻找什么。如果否,检测用户浏览器是否启用/禁用 JavaScript 的最佳方法是什么 最佳答案 不,没有
Spring 和 OSGi 目前的开发状况如何? 最近好像有点安静了。 文档的最新版本 ( http://docs.spring.io/osgi/ ) 来自 2009 年。 我看到一些声明 Sprin
我正在从主函数为此类创建一个线程,但即使使用 Thread.currentThread().interrupt() 中断它,输出仍然包含“Still Here”行。 public class Writ
为了满足并发要求,我想知道如何在 Godog 中的多个步骤之间传递参数或状态。 func FeatureContext(s *godog.Suite) { // This step is ca
我有一个UIButton子类,它不使用UIImage背景,仅使用背景色。我注意到的一件事是,当您设置按钮的背景图像时,有一个默认的突出显示状态,当按下按钮时,该按钮会稍微变暗。 这是我当前的代码。
我是一名优秀的程序员,十分优秀!