Linux Audio

Check our new training course

Loading...
Note: File does not exist in v3.1.
  1/* SPDX-License-Identifier: GPL-2.0 */
  2#ifndef __FIRMWARE_LOADER_H
  3#define __FIRMWARE_LOADER_H
  4
  5#include <linux/firmware.h>
  6#include <linux/types.h>
  7#include <linux/kref.h>
  8#include <linux/list.h>
  9#include <linux/completion.h>
 10
 11#include <generated/utsrelease.h>
 12
 13/* firmware behavior options */
 14#define FW_OPT_UEVENT			(1U << 0)
 15#define FW_OPT_NOWAIT			(1U << 1)
 16#define FW_OPT_USERHELPER		(1U << 2)
 17#define FW_OPT_NO_WARN			(1U << 3)
 18#define FW_OPT_NOCACHE			(1U << 4)
 19#define FW_OPT_NOFALLBACK		(1U << 5)
 20
 21enum fw_status {
 22	FW_STATUS_UNKNOWN,
 23	FW_STATUS_LOADING,
 24	FW_STATUS_DONE,
 25	FW_STATUS_ABORTED,
 26};
 27
 28/*
 29 * Concurrent request_firmware() for the same firmware need to be
 30 * serialized.  struct fw_state is simple state machine which hold the
 31 * state of the firmware loading.
 32 */
 33struct fw_state {
 34	struct completion completion;
 35	enum fw_status status;
 36};
 37
 38struct fw_priv {
 39	struct kref ref;
 40	struct list_head list;
 41	struct firmware_cache *fwc;
 42	struct fw_state fw_st;
 43	void *data;
 44	size_t size;
 45	size_t allocated_size;
 46#ifdef CONFIG_FW_LOADER_USER_HELPER
 47	bool is_paged_buf;
 48	bool need_uevent;
 49	struct page **pages;
 50	int nr_pages;
 51	int page_array_size;
 52	struct list_head pending_list;
 53#endif
 54	const char *fw_name;
 55};
 56
 57extern struct mutex fw_lock;
 58
 59static inline bool __fw_state_check(struct fw_priv *fw_priv,
 60				    enum fw_status status)
 61{
 62	struct fw_state *fw_st = &fw_priv->fw_st;
 63
 64	return fw_st->status == status;
 65}
 66
 67static inline int __fw_state_wait_common(struct fw_priv *fw_priv, long timeout)
 68{
 69	struct fw_state *fw_st = &fw_priv->fw_st;
 70	long ret;
 71
 72	ret = wait_for_completion_killable_timeout(&fw_st->completion, timeout);
 73	if (ret != 0 && fw_st->status == FW_STATUS_ABORTED)
 74		return -ENOENT;
 75	if (!ret)
 76		return -ETIMEDOUT;
 77
 78	return ret < 0 ? ret : 0;
 79}
 80
 81static inline void __fw_state_set(struct fw_priv *fw_priv,
 82				  enum fw_status status)
 83{
 84	struct fw_state *fw_st = &fw_priv->fw_st;
 85
 86	WRITE_ONCE(fw_st->status, status);
 87
 88	if (status == FW_STATUS_DONE || status == FW_STATUS_ABORTED)
 89		complete_all(&fw_st->completion);
 90}
 91
 92static inline void fw_state_aborted(struct fw_priv *fw_priv)
 93{
 94	__fw_state_set(fw_priv, FW_STATUS_ABORTED);
 95}
 96
 97static inline bool fw_state_is_aborted(struct fw_priv *fw_priv)
 98{
 99	return __fw_state_check(fw_priv, FW_STATUS_ABORTED);
100}
101
102static inline void fw_state_start(struct fw_priv *fw_priv)
103{
104	__fw_state_set(fw_priv, FW_STATUS_LOADING);
105}
106
107static inline void fw_state_done(struct fw_priv *fw_priv)
108{
109	__fw_state_set(fw_priv, FW_STATUS_DONE);
110}
111
112int assign_fw(struct firmware *fw, struct device *device,
113	      unsigned int opt_flags);
114
115#endif /* __FIRMWARE_LOADER_H */