19c92ab61SThomas Gleixner /* SPDX-License-Identifier: GPL-2.0-only */
2400e64dfSOhad Ben-Cohen /*
3400e64dfSOhad Ben-Cohen * Remote processor framework
4400e64dfSOhad Ben-Cohen *
5400e64dfSOhad Ben-Cohen * Copyright (C) 2011 Texas Instruments, Inc.
6400e64dfSOhad Ben-Cohen * Copyright (C) 2011 Google, Inc.
7400e64dfSOhad Ben-Cohen *
8400e64dfSOhad Ben-Cohen * Ohad Ben-Cohen <ohad@wizery.com>
9400e64dfSOhad Ben-Cohen * Brian Swetland <swetland@google.com>
10400e64dfSOhad Ben-Cohen */
11400e64dfSOhad Ben-Cohen
12400e64dfSOhad Ben-Cohen #ifndef REMOTEPROC_INTERNAL_H
13400e64dfSOhad Ben-Cohen #define REMOTEPROC_INTERNAL_H
14400e64dfSOhad Ben-Cohen
15400e64dfSOhad Ben-Cohen #include <linux/irqreturn.h>
1672854fb0SSjur Brændeland #include <linux/firmware.h>
17400e64dfSOhad Ben-Cohen
18400e64dfSOhad Ben-Cohen struct rproc;
19400e64dfSOhad Ben-Cohen
20a987e6b9SLoic Pallardy struct rproc_debug_trace {
21a987e6b9SLoic Pallardy struct rproc *rproc;
22a987e6b9SLoic Pallardy struct dentry *tfile;
23a987e6b9SLoic Pallardy struct list_head node;
24a987e6b9SLoic Pallardy struct rproc_mem_entry trace_mem;
25a987e6b9SLoic Pallardy };
26a987e6b9SLoic Pallardy
27fd28f879SArnaud Pouliquen /**
28fd28f879SArnaud Pouliquen * struct rproc_vdev_data - remoteproc virtio device data
29fd28f879SArnaud Pouliquen * @rsc_offset: offset of the vdev's resource entry
30fd28f879SArnaud Pouliquen * @id: virtio device id (as in virtio_ids.h)
31fd28f879SArnaud Pouliquen * @index: vdev position versus other vdev declared in resource table
32fd28f879SArnaud Pouliquen * @rsc: pointer to the vdev resource entry. Valid only during vdev init as
33fd28f879SArnaud Pouliquen * the resource can be cached by rproc.
34fd28f879SArnaud Pouliquen */
35fd28f879SArnaud Pouliquen struct rproc_vdev_data {
36fd28f879SArnaud Pouliquen u32 rsc_offset;
37fd28f879SArnaud Pouliquen unsigned int id;
38fd28f879SArnaud Pouliquen u32 index;
39fd28f879SArnaud Pouliquen struct fw_rsc_vdev *rsc;
40fd28f879SArnaud Pouliquen };
41fd28f879SArnaud Pouliquen
rproc_has_feature(struct rproc * rproc,unsigned int feature)42*49f27f2bSPeng Fan static inline bool rproc_has_feature(struct rproc *rproc, unsigned int feature)
43*49f27f2bSPeng Fan {
44*49f27f2bSPeng Fan return test_bit(feature, rproc->features);
45*49f27f2bSPeng Fan }
46*49f27f2bSPeng Fan
rproc_set_feature(struct rproc * rproc,unsigned int feature)47*49f27f2bSPeng Fan static inline int rproc_set_feature(struct rproc *rproc, unsigned int feature)
48*49f27f2bSPeng Fan {
49*49f27f2bSPeng Fan if (feature >= RPROC_MAX_FEATURES)
50*49f27f2bSPeng Fan return -EINVAL;
51*49f27f2bSPeng Fan
52*49f27f2bSPeng Fan set_bit(feature, rproc->features);
53*49f27f2bSPeng Fan
54*49f27f2bSPeng Fan return 0;
55*49f27f2bSPeng Fan }
56*49f27f2bSPeng Fan
57400e64dfSOhad Ben-Cohen /* from remoteproc_core.c */
58400e64dfSOhad Ben-Cohen void rproc_release(struct kref *kref);
59a8aa5ee1SSuman Anna int rproc_of_parse_firmware(struct device *dev, int index,
60a8aa5ee1SSuman Anna const char **fw_name);
61400e64dfSOhad Ben-Cohen
627a186941SOhad Ben-Cohen /* from remoteproc_virtio.c */
639c31255cSArnaud Pouliquen irqreturn_t rproc_vq_interrupt(struct rproc *rproc, int vq_id);
64400e64dfSOhad Ben-Cohen
65400e64dfSOhad Ben-Cohen /* from remoteproc_debugfs.c */
66400e64dfSOhad Ben-Cohen void rproc_remove_trace_file(struct dentry *tfile);
67400e64dfSOhad Ben-Cohen struct dentry *rproc_create_trace_file(const char *name, struct rproc *rproc,
68a987e6b9SLoic Pallardy struct rproc_debug_trace *trace);
69400e64dfSOhad Ben-Cohen void rproc_delete_debug_dir(struct rproc *rproc);
70400e64dfSOhad Ben-Cohen void rproc_create_debug_dir(struct rproc *rproc);
71400e64dfSOhad Ben-Cohen void rproc_init_debugfs(void);
72400e64dfSOhad Ben-Cohen void rproc_exit_debugfs(void);
73400e64dfSOhad Ben-Cohen
742aefbef0SMatt Redfearn /* from remoteproc_sysfs.c */
752aefbef0SMatt Redfearn extern struct class rproc_class;
762aefbef0SMatt Redfearn int rproc_init_sysfs(void);
772aefbef0SMatt Redfearn void rproc_exit_sysfs(void);
782aefbef0SMatt Redfearn
7944767708SSiddharth Gupta #ifdef CONFIG_REMOTEPROC_CDEV
8044767708SSiddharth Gupta void rproc_init_cdev(void);
8144767708SSiddharth Gupta void rproc_exit_cdev(void);
8244767708SSiddharth Gupta int rproc_char_device_add(struct rproc *rproc);
8344767708SSiddharth Gupta void rproc_char_device_remove(struct rproc *rproc);
8444767708SSiddharth Gupta #else
rproc_init_cdev(void)8544767708SSiddharth Gupta static inline void rproc_init_cdev(void)
8644767708SSiddharth Gupta {
8744767708SSiddharth Gupta }
8844767708SSiddharth Gupta
rproc_exit_cdev(void)8944767708SSiddharth Gupta static inline void rproc_exit_cdev(void)
9044767708SSiddharth Gupta {
9144767708SSiddharth Gupta }
9244767708SSiddharth Gupta
9344767708SSiddharth Gupta /*
9444767708SSiddharth Gupta * The character device interface is an optional feature, if it is not enabled
9544767708SSiddharth Gupta * the function should not return an error.
9644767708SSiddharth Gupta */
rproc_char_device_add(struct rproc * rproc)9744767708SSiddharth Gupta static inline int rproc_char_device_add(struct rproc *rproc)
9844767708SSiddharth Gupta {
9944767708SSiddharth Gupta return 0;
10044767708SSiddharth Gupta }
10144767708SSiddharth Gupta
rproc_char_device_remove(struct rproc * rproc)10244767708SSiddharth Gupta static inline void rproc_char_device_remove(struct rproc *rproc)
10344767708SSiddharth Gupta {
10444767708SSiddharth Gupta }
10544767708SSiddharth Gupta #endif
10644767708SSiddharth Gupta
1076db20ea8SOhad Ben-Cohen void rproc_free_vring(struct rproc_vring *rvring);
1086db20ea8SOhad Ben-Cohen int rproc_alloc_vring(struct rproc_vdev *rvdev, int i);
1099c31255cSArnaud Pouliquen int rproc_parse_vring(struct rproc_vdev *rvdev, struct fw_rsc_vdev *rsc, int i);
11072854fb0SSjur Brændeland
111086d0872SLoic Pallardy phys_addr_t rproc_va_to_pa(void *cpu_addr);
11270b85ef8SFernando Guzman Lugo int rproc_trigger_recovery(struct rproc *rproc);
11372854fb0SSjur Brændeland
114f31e339fSClement Leger int rproc_elf_sanity_check(struct rproc *rproc, const struct firmware *fw);
115e4ae4b7dSClement Leger u64 rproc_elf_get_boot_addr(struct rproc *rproc, const struct firmware *fw);
1160f21f9ccSBjorn Andersson int rproc_elf_load_segments(struct rproc *rproc, const struct firmware *fw);
11758b64090SBjorn Andersson int rproc_elf_load_rsc_table(struct rproc *rproc, const struct firmware *fw);
1180f21f9ccSBjorn Andersson struct resource_table *rproc_elf_find_loaded_rsc_table(struct rproc *rproc,
1190f21f9ccSBjorn Andersson const struct firmware *fw);
120c6aed238SLoic Pallardy struct rproc_mem_entry *
121c6aed238SLoic Pallardy rproc_find_carveout_by_name(struct rproc *rproc, const char *name, ...);
1229c31255cSArnaud Pouliquen void rproc_add_rvdev(struct rproc *rproc, struct rproc_vdev *rvdev);
1239c31255cSArnaud Pouliquen void rproc_remove_rvdev(struct rproc_vdev *rvdev);
1240f21f9ccSBjorn Andersson
rproc_prepare_device(struct rproc * rproc)12533467ac3SLoic Pallardy static inline int rproc_prepare_device(struct rproc *rproc)
12633467ac3SLoic Pallardy {
12733467ac3SLoic Pallardy if (rproc->ops->prepare)
12833467ac3SLoic Pallardy return rproc->ops->prepare(rproc);
12933467ac3SLoic Pallardy
13033467ac3SLoic Pallardy return 0;
13133467ac3SLoic Pallardy }
13233467ac3SLoic Pallardy
rproc_unprepare_device(struct rproc * rproc)13333467ac3SLoic Pallardy static inline int rproc_unprepare_device(struct rproc *rproc)
13433467ac3SLoic Pallardy {
13533467ac3SLoic Pallardy if (rproc->ops->unprepare)
13633467ac3SLoic Pallardy return rproc->ops->unprepare(rproc);
13733467ac3SLoic Pallardy
13833467ac3SLoic Pallardy return 0;
13933467ac3SLoic Pallardy }
14033467ac3SLoic Pallardy
rproc_attach_device(struct rproc * rproc)141a6a4f285SMathieu Poirier static inline int rproc_attach_device(struct rproc *rproc)
142a6a4f285SMathieu Poirier {
143a6a4f285SMathieu Poirier if (rproc->ops->attach)
144a6a4f285SMathieu Poirier return rproc->ops->attach(rproc);
145a6a4f285SMathieu Poirier
146a6a4f285SMathieu Poirier return 0;
147a6a4f285SMathieu Poirier }
148a6a4f285SMathieu Poirier
1494afc89d6SSjur Brændeland static inline
rproc_fw_sanity_check(struct rproc * rproc,const struct firmware * fw)1504afc89d6SSjur Brændeland int rproc_fw_sanity_check(struct rproc *rproc, const struct firmware *fw)
1514afc89d6SSjur Brændeland {
1520f21f9ccSBjorn Andersson if (rproc->ops->sanity_check)
1530f21f9ccSBjorn Andersson return rproc->ops->sanity_check(rproc, fw);
1544afc89d6SSjur Brændeland
1554afc89d6SSjur Brændeland return 0;
1564afc89d6SSjur Brændeland }
1574afc89d6SSjur Brændeland
1584afc89d6SSjur Brændeland static inline
rproc_get_boot_addr(struct rproc * rproc,const struct firmware * fw)159e4ae4b7dSClement Leger u64 rproc_get_boot_addr(struct rproc *rproc, const struct firmware *fw)
1604afc89d6SSjur Brændeland {
1610f21f9ccSBjorn Andersson if (rproc->ops->get_boot_addr)
1620f21f9ccSBjorn Andersson return rproc->ops->get_boot_addr(rproc, fw);
1634afc89d6SSjur Brændeland
1644afc89d6SSjur Brændeland return 0;
1654afc89d6SSjur Brændeland }
1664afc89d6SSjur Brændeland
1674afc89d6SSjur Brændeland static inline
rproc_load_segments(struct rproc * rproc,const struct firmware * fw)1684afc89d6SSjur Brændeland int rproc_load_segments(struct rproc *rproc, const struct firmware *fw)
1694afc89d6SSjur Brændeland {
1700f21f9ccSBjorn Andersson if (rproc->ops->load)
1710f21f9ccSBjorn Andersson return rproc->ops->load(rproc, fw);
1724afc89d6SSjur Brændeland
1734afc89d6SSjur Brændeland return -EINVAL;
1744afc89d6SSjur Brændeland }
1754afc89d6SSjur Brændeland
rproc_parse_fw(struct rproc * rproc,const struct firmware * fw)176c1d35c1aSBjorn Andersson static inline int rproc_parse_fw(struct rproc *rproc, const struct firmware *fw)
1774afc89d6SSjur Brændeland {
178c1d35c1aSBjorn Andersson if (rproc->ops->parse_fw)
179c1d35c1aSBjorn Andersson return rproc->ops->parse_fw(rproc, fw);
1804afc89d6SSjur Brændeland
18158b64090SBjorn Andersson return 0;
1824afc89d6SSjur Brændeland }
1834afc89d6SSjur Brændeland
18495f95781SSjur Brændeland static inline
rproc_handle_rsc(struct rproc * rproc,u32 rsc_type,void * rsc,int offset,int avail)185b1a17513SClement Leger int rproc_handle_rsc(struct rproc *rproc, u32 rsc_type, void *rsc, int offset,
186b1a17513SClement Leger int avail)
187b1a17513SClement Leger {
188b1a17513SClement Leger if (rproc->ops->handle_rsc)
189b1a17513SClement Leger return rproc->ops->handle_rsc(rproc, rsc_type, rsc, offset,
190b1a17513SClement Leger avail);
191b1a17513SClement Leger
192b1a17513SClement Leger return RSC_IGNORED;
193b1a17513SClement Leger }
194b1a17513SClement Leger
195b1a17513SClement Leger static inline
rproc_find_loaded_rsc_table(struct rproc * rproc,const struct firmware * fw)19695f95781SSjur Brændeland struct resource_table *rproc_find_loaded_rsc_table(struct rproc *rproc,
19795f95781SSjur Brændeland const struct firmware *fw)
19895f95781SSjur Brændeland {
1990f21f9ccSBjorn Andersson if (rproc->ops->find_loaded_rsc_table)
2000f21f9ccSBjorn Andersson return rproc->ops->find_loaded_rsc_table(rproc, fw);
20195f95781SSjur Brændeland
20295f95781SSjur Brændeland return NULL;
20395f95781SSjur Brændeland }
20495f95781SSjur Brændeland
205f31e339fSClement Leger static inline
rproc_get_loaded_rsc_table(struct rproc * rproc,size_t * size)2061a631382SMathieu Poirier struct resource_table *rproc_get_loaded_rsc_table(struct rproc *rproc,
2071a631382SMathieu Poirier size_t *size)
2081a631382SMathieu Poirier {
2091a631382SMathieu Poirier if (rproc->ops->get_loaded_rsc_table)
2101a631382SMathieu Poirier return rproc->ops->get_loaded_rsc_table(rproc, size);
2111a631382SMathieu Poirier
2121a631382SMathieu Poirier return NULL;
2131a631382SMathieu Poirier }
2141a631382SMathieu Poirier
2151a631382SMathieu Poirier static inline
rproc_u64_fit_in_size_t(u64 val)216f31e339fSClement Leger bool rproc_u64_fit_in_size_t(u64 val)
217f31e339fSClement Leger {
218f31e339fSClement Leger if (sizeof(size_t) == sizeof(u64))
219f31e339fSClement Leger return true;
220f31e339fSClement Leger
221f31e339fSClement Leger return (val <= (size_t) -1);
222f31e339fSClement Leger }
223f31e339fSClement Leger
224400e64dfSOhad Ben-Cohen #endif /* REMOTEPROC_INTERNAL_H */
225