fw-device.h 4.7 KB
Newer Older
1
/*
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 * Copyright (C) 2005-2006  Kristian Hoegsberg <krh@bitplanet.net>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software Foundation,
 * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 */

#ifndef __fw_device_h
#define __fw_device_h

#include <linux/fs.h>
#include <linux/cdev.h>
24
#include <linux/rwsem.h>
25
#include <asm/atomic.h>
26 27 28 29

enum fw_device_state {
	FW_DEVICE_INITIALIZING,
	FW_DEVICE_RUNNING,
30
	FW_DEVICE_SHUTDOWN,
31 32
};

33 34 35 36 37 38
struct fw_attribute_group {
	struct attribute_group *groups[2];
	struct attribute_group group;
	struct attribute *attrs[11];
};

39 40 41 42 43 44 45 46 47 48 49
/*
 * Note, fw_device.generation always has to be read before fw_device.node_id.
 * Use SMP memory barriers to ensure this.  Otherwise requests will be sent
 * to an outdated node_id if the generation was updated in the meantime due
 * to a bus reset.
 *
 * Likewise, fw-core will take care to update .node_id before .generation so
 * that whenever fw_device.generation is current WRT the actual bus generation,
 * fw_device.node_id is guaranteed to be current too.
 *
 * The same applies to fw_device.card->node_id vs. fw_device.generation.
50 51 52 53 54
 *
 * fw_device.config_rom and fw_device.config_rom_length may be accessed during
 * the lifetime of any fw_unit belonging to the fw_device, before device_del()
 * was called on the last fw_unit.  Alternatively, they may be accessed while
 * holding fw_device_rwsem.
55
 */
56
struct fw_device {
57
	atomic_t state;
58 59 60
	struct fw_node *node;
	int node_id;
	int generation;
61
	unsigned max_speed;
62
	bool cmc;
63 64
	struct fw_card *card;
	struct device device;
65
	struct list_head client_list;
66
	u32 *config_rom;
67 68 69
	size_t config_rom_length;
	int config_rom_retries;
	struct delayed_work work;
70
	struct fw_attribute_group attribute_group;
71 72
};

73
static inline struct fw_device *fw_device(struct device *dev)
74
{
75
	return container_of(dev, struct fw_device, device);
76 77
}

78
static inline int fw_device_is_shutdown(struct fw_device *device)
79 80 81 82
{
	return atomic_read(&device->state) == FW_DEVICE_SHUTDOWN;
}

83
static inline struct fw_device *fw_device_get(struct fw_device *device)
84 85 86 87 88 89
{
	get_device(&device->device);

	return device;
}

90
static inline void fw_device_put(struct fw_device *device)
91 92 93 94
{
	put_device(&device->device);
}

95
struct fw_device *fw_device_get_by_devt(dev_t devt);
96 97
int fw_device_enable_phys_dma(struct fw_device *device);

98
void fw_device_cdev_update(struct fw_device *device);
99
void fw_device_cdev_remove(struct fw_device *device);
100

101
extern struct rw_semaphore fw_device_rwsem;
102 103
extern int fw_cdev_major;

104 105 106
/*
 * fw_unit.directory must not be accessed after device_del(&fw_unit.device).
 */
107 108 109
struct fw_unit {
	struct device device;
	u32 *directory;
110
	struct fw_attribute_group attribute_group;
111 112
};

113
static inline struct fw_unit *fw_unit(struct device *dev)
114
{
115
	return container_of(dev, struct fw_unit, device);
116 117
}

118 119 120 121 122 123 124 125 126 127 128 129
static inline struct fw_unit *fw_unit_get(struct fw_unit *unit)
{
	get_device(&unit->device);

	return unit;
}

static inline void fw_unit_put(struct fw_unit *unit)
{
	put_device(&unit->device);
}

130 131 132 133 134 135 136 137 138 139 140 141 142 143
#define CSR_OFFSET	0x40
#define CSR_LEAF	0x80
#define CSR_DIRECTORY	0xc0

#define CSR_DESCRIPTOR		0x01
#define CSR_VENDOR		0x03
#define CSR_HARDWARE_VERSION	0x04
#define CSR_NODE_CAPABILITIES	0x0c
#define CSR_UNIT		0x11
#define CSR_SPECIFIER_ID	0x12
#define CSR_VERSION		0x13
#define CSR_DEPENDENT_INFO	0x14
#define CSR_MODEL		0x17
#define CSR_INSTANCE		0x18
144
#define CSR_DIRECTORY_ID	0x20
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172

struct fw_csr_iterator {
	u32 *p;
	u32 *end;
};

void fw_csr_iterator_init(struct fw_csr_iterator *ci, u32 *p);
int fw_csr_iterator_next(struct fw_csr_iterator *ci,
			 int *key, int *value);

#define FW_MATCH_VENDOR		0x0001
#define FW_MATCH_MODEL		0x0002
#define FW_MATCH_SPECIFIER_ID	0x0004
#define FW_MATCH_VERSION	0x0008

struct fw_device_id {
	u32 match_flags;
	u32 vendor;
	u32 model;
	u32 specifier_id;
	u32 version;
	void *driver_data;
};

struct fw_driver {
	struct device_driver driver;
	/* Called when the parent device sits through a bus reset. */
	void (*update) (struct fw_unit *unit);
173
	const struct fw_device_id *id_table;
174 175 176 177 178
};

static inline struct fw_driver *
fw_driver(struct device_driver *drv)
{
179
	return container_of(drv, struct fw_driver, driver);
180 181
}

182
extern const struct file_operations fw_device_ops;
183

184
#endif /* __fw_device_h */