forked from Beckhoff/CCAT
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sram.c
110 lines (88 loc) · 2.59 KB
/
sram.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
// SPDX-License-Identifier: MIT
/**
Network Driver for Beckhoff CCAT communication controller
Copyright (C) 2015 - 2018 Beckhoff Automation GmbH & Co. KG
Author: Patrick Bruenn <[email protected]>
*/
// vi: set noexpandtab:
#include "module.h"
#include "sram.h"
#include <asm/io.h>
#include <linux/fs.h>
#include <linux/module.h>
#include <linux/uaccess.h>
MODULE_DESCRIPTION(DRV_DESCRIPTION);
MODULE_AUTHOR("Patrick Bruenn <[email protected]>");
MODULE_LICENSE("GPL and additional rights");
MODULE_VERSION(DRV_VERSION);
#define CCAT_SRAM_DEVICES_MAX 4
static ssize_t __sram_read(struct cdev_buffer *buffer, char __user * buf,
size_t len, loff_t * off)
{
memcpy_fromio(buffer->data, buffer->ccdev->ioaddr + *off, len);
if (copy_to_user(buf, buffer->data, len))
return -EFAULT;
*off += len;
return len;
}
ssize_t ccat_sram_read(struct file *const f, char __user * buf,
size_t len, loff_t * off)
{
struct cdev_buffer *buffer = f->private_data;
const size_t iosize = buffer->ccdev->iosize;
if (*off >= iosize) {
return 0;
}
len = min(len, (size_t) (iosize - *off));
return __sram_read(buffer, buf, len, off);
}
EXPORT_SYMBOL(ccat_sram_read);
ssize_t ccat_sram_write(struct file *const f, const char __user * buf,
size_t len, loff_t * off)
{
struct cdev_buffer *const buffer = f->private_data;
if (*off + len > buffer->ccdev->iosize) {
return 0;
}
if (copy_from_user(buffer->data, buf, len)) {
return -EFAULT;
}
memcpy_toio(buffer->ccdev->ioaddr + *off, buffer->data, len);
*off += len;
return len;
}
EXPORT_SYMBOL(ccat_sram_write);
static struct ccat_cdev dev_table[CCAT_SRAM_DEVICES_MAX];
static struct ccat_class cdev_class = {
.instances = {0},
.count = CCAT_SRAM_DEVICES_MAX,
.devices = dev_table,
.name = "ccat_sram",
.fops = {
.owner = THIS_MODULE,
.llseek = ccat_cdev_llseek,
.open = ccat_cdev_open,
.release = ccat_cdev_release,
.read = ccat_sram_read,
.write = ccat_sram_write,
},
};
static int ccat_sram_probe(struct platform_device *pdev)
{
struct ccat_function *const func = pdev->dev.platform_data;
static const u8 NO_SRAM_CONNECTED = 0;
const u8 type = func->info.sram_width & 0x3;
const size_t iosize = (1 << func->info.sram_size);
pr_info("%s: 0x%04x rev: 0x%04x\n", __FUNCTION__, func->info.type,
func->info.rev);
if (type == NO_SRAM_CONNECTED) {
return -ENODEV;
}
return ccat_cdev_probe(func, &cdev_class, iosize, NULL);
}
static struct platform_driver sram_driver = {
.driver = {.name = "ccat_sram"},
.probe = ccat_sram_probe,
.remove = ccat_cdev_remove,
};
module_platform_driver(sram_driver);