| // SPDX-License-Identifier: GPL-1.0+ |
| /* |
| * Renesas USB driver |
| * |
| * Copyright (C) 2011 Renesas Solutions Corp. |
| * Copyright (C) 2019 Renesas Electronics Corporation |
| * Kuninori Morimoto <kuninori.morimoto.gx@renesas.com> |
| */ |
| #ifndef RENESAS_USB_MOD_H |
| #define RENESAS_USB_MOD_H |
| |
| #include <linux/spinlock.h> |
| #include <linux/usb/renesas_usbhs.h> |
| #include "common.h" |
| |
| /* |
| * struct |
| */ |
| struct usbhs_irq_state { |
| u16 intsts0; |
| u16 intsts1; |
| u16 brdysts; |
| u16 nrdysts; |
| u16 bempsts; |
| }; |
| |
| struct usbhs_mod { |
| char *name; |
| |
| /* |
| * entry point from common.c |
| */ |
| int (*start)(struct usbhs_priv *priv); |
| int (*stop)(struct usbhs_priv *priv); |
| |
| /* |
| * INTSTS0 |
| */ |
| |
| /* DVST (DVSQ) */ |
| int (*irq_dev_state)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| /* CTRT (CTSQ) */ |
| int (*irq_ctrl_stage)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| /* BEMP / BEMPSTS */ |
| int (*irq_empty)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| u16 irq_bempsts; |
| |
| /* BRDY / BRDYSTS */ |
| int (*irq_ready)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| u16 irq_brdysts; |
| |
| /* |
| * INTSTS1 |
| */ |
| |
| /* ATTCHE */ |
| int (*irq_attch)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| /* DTCHE */ |
| int (*irq_dtch)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| /* SIGN */ |
| int (*irq_sign)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| /* SACK */ |
| int (*irq_sack)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| struct usbhs_priv *priv; |
| }; |
| |
| struct usbhs_mod_info { |
| struct usbhs_mod *mod[USBHS_MAX]; |
| struct usbhs_mod *curt; /* current mod */ |
| |
| /* |
| * INTSTS0 :: VBINT |
| * |
| * This function will be used as autonomy mode (runtime_pwctrl == 0) |
| * when the platform doesn't have own get_vbus function. |
| * |
| * This callback cannot be member of "struct usbhs_mod" because it |
| * will be used even though host/gadget has not been selected. |
| */ |
| int (*irq_vbus)(struct usbhs_priv *priv, |
| struct usbhs_irq_state *irq_state); |
| |
| /* |
| * This function will be used on any gadget mode. To simplify the code, |
| * this member is in here. |
| */ |
| int (*get_vbus)(struct platform_device *pdev); |
| }; |
| |
| /* |
| * for host/gadget module |
| */ |
| struct usbhs_mod *usbhs_mod_get(struct usbhs_priv *priv, int id); |
| struct usbhs_mod *usbhs_mod_get_current(struct usbhs_priv *priv); |
| void usbhs_mod_register(struct usbhs_priv *priv, struct usbhs_mod *usb, int id); |
| int usbhs_mod_is_host(struct usbhs_priv *priv); |
| int usbhs_mod_change(struct usbhs_priv *priv, int id); |
| int usbhs_mod_probe(struct usbhs_priv *priv); |
| void usbhs_mod_remove(struct usbhs_priv *priv); |
| |
| void usbhs_mod_autonomy_mode(struct usbhs_priv *priv); |
| void usbhs_mod_non_autonomy_mode(struct usbhs_priv *priv); |
| |
| /* |
| * status functions |
| */ |
| int usbhs_status_get_device_state(struct usbhs_irq_state *irq_state); |
| int usbhs_status_get_ctrl_stage(struct usbhs_irq_state *irq_state); |
| |
| /* |
| * callback functions |
| */ |
| void usbhs_irq_callback_update(struct usbhs_priv *priv, struct usbhs_mod *mod); |
| |
| |
| #define usbhs_mod_call(priv, func, param...) \ |
| ({ \ |
| struct usbhs_mod *mod; \ |
| mod = usbhs_mod_get_current(priv); \ |
| !mod ? -ENODEV : \ |
| !mod->func ? 0 : \ |
| mod->func(param); \ |
| }) |
| |
| #define usbhs_priv_to_modinfo(priv) (&priv->mod_info) |
| #define usbhs_mod_info_call(priv, func, param...) \ |
| ({ \ |
| struct usbhs_mod_info *info; \ |
| info = usbhs_priv_to_modinfo(priv); \ |
| !info->func ? 0 : \ |
| info->func(param); \ |
| }) |
| |
| /* |
| * host / gadget control |
| */ |
| #if defined(CONFIG_USB_RENESAS_USBHS_HCD) || \ |
| defined(CONFIG_USB_RENESAS_USBHS_HCD_MODULE) |
| extern int usbhs_mod_host_probe(struct usbhs_priv *priv); |
| extern int usbhs_mod_host_remove(struct usbhs_priv *priv); |
| #else |
| static inline int usbhs_mod_host_probe(struct usbhs_priv *priv) |
| { |
| return 0; |
| } |
| static inline void usbhs_mod_host_remove(struct usbhs_priv *priv) |
| { |
| } |
| #endif |
| |
| #if defined(CONFIG_USB_RENESAS_USBHS_UDC) || \ |
| defined(CONFIG_USB_RENESAS_USBHS_UDC_MODULE) |
| extern int usbhs_mod_gadget_probe(struct usbhs_priv *priv); |
| extern void usbhs_mod_gadget_remove(struct usbhs_priv *priv); |
| #else |
| static inline int usbhs_mod_gadget_probe(struct usbhs_priv *priv) |
| { |
| return 0; |
| } |
| static inline void usbhs_mod_gadget_remove(struct usbhs_priv *priv) |
| { |
| } |
| #endif |
| |
| #endif /* RENESAS_USB_MOD_H */ |