| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- #include <linux/module.h>
- #include <linux/kernel.h>
- #include <linux/init.h>
- #include <linux/fs.h>
- #include <linux/cdev.h>
- #include <linux/device.h>
- #include <linux/uaccess.h>
- #include <linux/slab.h>
- #include <linux/pci.h>
- #include <linux/i2c.h>
- #include <linux/acpi.h>
- #include <linux/interrupt.h>
- #include <linux/wait.h>
- #include <linux/sched.h>
- #include <linux/poll.h>
- #include <linux/mutex.h>
- #include <linux/delay.h>
- #include <asm/io.h>
- #include "gpioregs.h"
- extern struct kobject *vfiec_kobj;
- static int wait_ibf(void)
- {
- int i = 0;
- while (inb(EC_CMD_PORT) & EC_IBF)
- {
- if (++i > TIMEOUT_LOOPS)
- {
- return -1;
- }
- udelay(1);
- }
- return 0;
- }
- static int wait_obf(void)
- {
- int i = 0;
- while (!(inb(EC_CMD_PORT) & EC_OBF))
- {
- if (++i > TIMEOUT_LOOPS)
- {
- return -1;
- }
- udelay(1);
- }
- return 0;
- }
- static int oem_ec_read_ram(uint8_t page, uint8_t offset, uint8_t *data)
- {
- unsigned char WEC, REC;
- switch(page)
- {
- case 0:
- {
- WEC = 0x96;
- REC = 0x95;
- break;
- }
-
- case 1:
- {
- WEC = 0x98;
- REC = 0x97;
- break;
- }
-
- default:
- {
- WEC = EC_VERSION_WEC;
- REC = EC_VERSION_REC;
- break;
- }
- }
- if (wait_ibf() < 0)
- return -1;
- outb(REC, EC_CMD_PORT);
- if (wait_ibf() < 0)
- return -1;
- outb(offset, EC_DATA_PORT);
- if (wait_obf() < 0)
- return -1;
- *data = inb(EC_DATA_PORT);
- return 0;
- }
- /* ==================== mode ==================== */
- static ssize_t fw_version_show(struct kobject *kobj, struct kobj_attribute *attr,
- char *buf)
- {
- unsigned char data[4] = {0};
- int i = 0;
- for(i = 0; i < 4; i++)
- {
- oem_ec_read_ram(2, i, &data[i]);
- }
- // 这里待完善,获取真实的,版本号
- sprintf(buf, "%02x%02x%02x%02x", data[0], data[1], data[2], data[3]);
- return 8;
- }
- static ssize_t fw_version_store(struct kobject *kobj, struct kobj_attribute *attr,
- const char *buf, size_t count)
- {
- printk("fw_version_store kernel rev:%s\n", buf);
- return count;
- }
- static struct kobj_attribute ec_version_attr =
- __ATTR(fw_version, 0444, fw_version_show, fw_version_store);
- /* ==================== 属性组 ==================== */
- static struct attribute *ec_version_attrs[] = {
- &ec_version_attr.attr,
- NULL,
- };
- static struct attribute_group ec_version_attr_group = {
- .attrs = ec_version_attrs,
- };
- int ec_version_init(void)
- {
- int ret = 0;
- /* 创建属性文件 */
- ret = sysfs_create_group(vfiec_kobj, &ec_version_attr_group);
- if (ret)
- {
- pr_err("Faiec_version to create sysfs group: %d\n", ret);
- }
- return ret;
- }
- void ec_version_exit(void)
- {
- sysfs_remove_group(vfiec_kobj, &ec_version_attr_group);
- }
|