All Courses
All Courses
Courses by Software
Courses by Semester
Courses by Domain
Tool-focused Courses
Machine learning
POPULAR COURSES
Success Stories
Overview of the Project: Device Driver Write the device driver for MCP9808 Temperature Sensor. Probe function: Register the Platform device under driver/misc. Read function: It should send the i2c message to read the data from sensor and copy it to the user space. Alert pin connected to BBB as gpio interrupt. The temp…
Omkar Madhavi
updated on 21 Feb 2024
Code:
mcp9808.c :
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define MCP9808_TEMP_REG 0x05
#define MCP9808_CONFIG_REG 0x01
#define MCP9808_T_LOWER_REG 0x02
#define MCP9808_T_UPPER_REG 0x03
struct mcp9808_data {
struct i2c_client *client;
struct platform_device *pdev;
int irq_gpio;
int threshold;
};
static irqreturn_t mcp9808_alert_handler(int irq, void *data) {
struct mcp9808_data *mcp = data;
int temp;
u8 buf[2];
i2c_smbus_read_i2c_block_data(mcp->client, MCP9808_TEMP_REG, 2, buf);
temp = ((buf[0] & 0x0F) << 8) | buf[1];
if (temp > mcp->threshold) {
gpio_set_value_cansleep(mcp->irq_gpio, 1);
} else {
gpio_set_value_cansleep(mcp->irq_gpio, 0);
}
return IRQ_HANDLED;
}
static int mcp9808_probe(struct i2c_client *client) {
struct mcp9808_data *mcp;
struct platform_device *pdev;
struct device *dev = &client->dev;
int ret;
mcp = devm_kzalloc(dev, sizeof(struct mcp9808_data), GFP_KERNEL);
if (!mcp) {
dev_err(dev, "Failed to allocate memory\n");
return -ENOMEM;
}
mcp->client = client;
mcp->threshold = 0;
// Set T_LOWER and T_UPPER from device tree
struct device_node *np = dev->of_node;
if (np) {
u32 t_lower, t_upper;
of_property_read_u32(np, "t-lower", &t_lower);
of_property_read_u32(np, "t-upper", &t_upper);
i2c_smbus_write_word_data(mcp->client, MCP9808_T_LOWER_REG, t_lower);
i2c_smbus_write_word_data(mcp->client, MCP9808_T_UPPER_REG, t_upper);
dev_info(dev, "Set T_LOWER = %d and T_UPPER = %d\n", t_lower, t_upper);
}
// Register the platform device
pdev = platform_device_alloc("mcp9808", -1);
if (!pdev) {
dev_err(dev, "Failed to allocate platform device\n");
return -ENOMEM;
}
mcp->pdev = pdev;
ret = platform_device_add_data(pdev, mcp, sizeof(struct mcp9808_data));
if (ret) {
dev_err(dev, "Failed to add platform device data\n");
platform_device_put(pdev);
return ret;
}
ret = platform_device_add(pdev);
if (ret) {
dev_err(dev, "Failed to add platform device\n");
platform_device_unregister(pdev);
return ret;
}
// Register the interrupt handler
mcp->irq_gpio = of_get_named_gpio(dev->of_node, "alert-gpio", 0);
if (gpio_is_valid(mcp->irq_gpio)) {
ret = gpio_request(mcp->irq_gpio, "mcp9808-alert");
if (ret) {
dev_err(dev, "Failed to request gpio %d\n", mcp->irq_gpio);
platform_device_unregister(mcp->pdev);
return ret;
}
ret = gpio_direction_output(mcp->irq_gpio, 0);
if (ret) {
dev_err(dev, "Failed to set gpio %d direction\n", mcp->irq_gpio);
gpio_free(mcp->irq_gpio);
platform_device_unregister(mcp->pdev);
return ret;
}
ret = gpio_to_irq(mcp->irq_gpio);
if (ret < 0) {
dev_err(dev, "Failed to get irq from gpio %d\n", mcp->irq_gpio);
gpio_free(mcp->irq_gpio);
platform_device_unregister(mcp->pdev);
return ret;
}
ret = request_irq(ret, mcp9808_alert_handler, IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, "mcp9808-alert", mcp);
if (ret) {
dev_err(dev, "Failed to request irq\n");
gpio_free(mcp->irq_gpio);
platform_device_unregister(mcp->pdev);
return ret;
}
dev_info(dev, "Registered interrupt handler\n");
}
dev_info(dev, "MCP9808 device detected\n");
return 0;
}
static void mcp9808_remove(struct i2c_client *client) {
struct mcp9808_data *mcp = i2c_get_clientdata(client);
if (gpio_is_valid(mcp->irq_gpio)) {
free_irq(gpio_to_irq(mcp->irq_gpio), mcp);
gpio_free(mcp->irq_gpio);
}
platform_device_unregister(mcp->pdev);
}
static const struct of_device_id mcp9808_of_match[] = {
{ .compatible = "microchip,mcp9808", },
{},
};
MODULE_DEVICE_TABLE(of, mcp9808_of_match);
static const struct i2c_device_id mcp9808_id[] = {
{ "mcp9808", 0 },
{ },
};
MODULE_DEVICE_TABLE(i2c, mcp9808_id);
static struct i2c_driver mcp9808_driver = {
.driver = {
.name = "mcp9808",
.of_match_table = mcp9808_of_match,
},
.probe = mcp9808_probe,
.remove = mcp9808_remove,
.id_table = mcp9808_id,
};
module_i2c_driver(mcp9808_driver);
MODULE_AUTHOR("Author name");
MODULE_DESCRIPTION("MCP9808 temperature sensor driver");
MODULE_LICENSE("GPL v2");
Makefile:
obj-m += mcp9808.o
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
clean:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
Leave a comment
Thanks for choosing to leave a comment. Please keep in mind that all the comments are moderated as per our comment policy, and your email will not be published for privacy reasons. Please leave a personal & meaningful conversation.
Other comments...
Project 2 - Design and develop the web based Temperature control system using Beagle Bone Black.
Overview of the Project: Device Driver Write the device driver for MCP9808 Temperature Sensor. Probe function: Register the Platform device under driver/misc. Read function: It should send the i2c message to read the data from sensor and copy it to the user space. Alert pin connected to BBB as gpio interrupt. The temp…
21 Feb 2024 06:40 PM IST
Project 1 - Develop the full featured char driver as a loaded module
Overview of the Project: Kconfig build system The module should be added to the Kernel configuration Kconfig build system with tristate option. Char driver should be of Open --> Only one user is allowed to open the device. Other users should be blocked. Read and write operation --> store the data to the dynamically…
15 Feb 2024 09:52 AM IST
Project 2 - V&V SW Analysis II
Perform Static Code Review Analysis for “C:\**\LDRA_workarea\Examples\C_Testbed_examples\Testrain\Testrain.c” Generate Code review report and upload them. Perform Integration testing by including all the .C files available in the folder location C:\**\LDRA_workarea\Examples\Toolsuite\Cashregister_6.0\…
25 Jan 2024 05:30 AM IST
Project 2 - Development of TFT Cluster Speedometer Software Component
1) Development of TFT Cluster Speedometer Software Component Cluster Instrument receives the signals from other ECU via CAN bus interface. It also receives commands from the driver via steering wheel buttons. The signals are then processed by the Cluster ECU, after which the Cluster ECU may send the vehicle information…
24 Jan 2024 10:46 AM IST
Related Courses
Skill-Lync offers industry relevant advanced engineering courses for engineering students by partnering with industry experts.
© 2025 Skill-Lync Inc. All Rights Reserved.