Commit abb9c9b8 authored by Srinivas Kandagatla's avatar Srinivas Kandagatla Committed by Greg Kroah-Hartman

slimbus: stream: add stream support

This patch adds support to SLIMbus stream apis for slimbus device.
SLIMbus streaming involves adding support to Data Channel Management and
channel Reconfiguration Messages to slim core plus few stream apis.
>From slim device side the apis are very simple mostly inline with other
stream apis.

Currently it only supports Isochronous and Push/Pull transport protocols,
which are sufficient for audio use cases.
Signed-off-by: default avatarSrinivas Kandagatla <srinivas.kandagatla@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 917809e2
...@@ -125,3 +125,8 @@ Messaging APIs: ...@@ -125,3 +125,8 @@ Messaging APIs:
~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~
.. kernel-doc:: drivers/slimbus/messaging.c .. kernel-doc:: drivers/slimbus/messaging.c
:export: :export:
Streaming APIs:
~~~~~~~~~~~~~~~
.. kernel-doc:: drivers/slimbus/stream.c
:export:
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
# Makefile for kernel SLIMbus framework. # Makefile for kernel SLIMbus framework.
# #
obj-$(CONFIG_SLIMBUS) += slimbus.o obj-$(CONFIG_SLIMBUS) += slimbus.o
slimbus-y := core.o messaging.o sched.o slimbus-y := core.o messaging.o sched.o stream.o
#Controllers #Controllers
obj-$(CONFIG_SLIM_QCOM_CTRL) += slim-qcom-ctrl.o obj-$(CONFIG_SLIM_QCOM_CTRL) += slim-qcom-ctrl.o
......
...@@ -114,6 +114,8 @@ static int slim_add_device(struct slim_controller *ctrl, ...@@ -114,6 +114,8 @@ static int slim_add_device(struct slim_controller *ctrl,
sbdev->dev.release = slim_dev_release; sbdev->dev.release = slim_dev_release;
sbdev->dev.driver = NULL; sbdev->dev.driver = NULL;
sbdev->ctrl = ctrl; sbdev->ctrl = ctrl;
INIT_LIST_HEAD(&sbdev->stream_list);
spin_lock_init(&sbdev->stream_list_lock);
if (node) if (node)
sbdev->dev.of_node = of_node_get(node); sbdev->dev.of_node = of_node_get(node);
......
...@@ -45,9 +45,20 @@ ...@@ -45,9 +45,20 @@
#define SLIM_MSG_MC_ASSIGN_LOGICAL_ADDRESS 0x2 #define SLIM_MSG_MC_ASSIGN_LOGICAL_ADDRESS 0x2
#define SLIM_MSG_MC_REPORT_ABSENT 0xF #define SLIM_MSG_MC_REPORT_ABSENT 0xF
/* Data channel management messages */
#define SLIM_MSG_MC_CONNECT_SOURCE 0x10
#define SLIM_MSG_MC_CONNECT_SINK 0x11
#define SLIM_MSG_MC_DISCONNECT_PORT 0x14
#define SLIM_MSG_MC_CHANGE_CONTENT 0x18
/* Clock pause Reconfiguration messages */ /* Clock pause Reconfiguration messages */
#define SLIM_MSG_MC_BEGIN_RECONFIGURATION 0x40 #define SLIM_MSG_MC_BEGIN_RECONFIGURATION 0x40
#define SLIM_MSG_MC_NEXT_PAUSE_CLOCK 0x4A #define SLIM_MSG_MC_NEXT_PAUSE_CLOCK 0x4A
#define SLIM_MSG_MC_NEXT_DEFINE_CHANNEL 0x50
#define SLIM_MSG_MC_NEXT_DEFINE_CONTENT 0x51
#define SLIM_MSG_MC_NEXT_ACTIVATE_CHANNEL 0x54
#define SLIM_MSG_MC_NEXT_DEACTIVATE_CHANNEL 0x55
#define SLIM_MSG_MC_NEXT_REMOVE_CHANNEL 0x58
#define SLIM_MSG_MC_RECONFIGURE_NOW 0x5F #define SLIM_MSG_MC_RECONFIGURE_NOW 0x5F
/* /*
...@@ -69,7 +80,15 @@ ...@@ -69,7 +80,15 @@
/* Standard values per SLIMbus spec needed by controllers and devices */ /* Standard values per SLIMbus spec needed by controllers and devices */
#define SLIM_MAX_CLK_GEAR 10 #define SLIM_MAX_CLK_GEAR 10
#define SLIM_MIN_CLK_GEAR 1 #define SLIM_MIN_CLK_GEAR 1
#define SLIM_SLOT_LEN_BITS 4
/* Indicate that the frequency of the flow and the bus frequency are locked */
#define SLIM_CHANNEL_CONTENT_FL BIT(7)
/* Standard values per SLIMbus spec needed by controllers and devices */
#define SLIM_CL_PER_SUPERFRAME 6144
#define SLIM_SLOTS_PER_SUPERFRAME (SLIM_CL_PER_SUPERFRAME >> 2)
#define SLIM_SL_PER_SUPERFRAME (SLIM_CL_PER_SUPERFRAME >> 2)
/* Manager's logical address is set to 0xFF per spec */ /* Manager's logical address is set to 0xFF per spec */
#define SLIM_LA_MANAGER 0xFF #define SLIM_LA_MANAGER 0xFF
...@@ -167,6 +186,170 @@ struct slim_sched { ...@@ -167,6 +186,170 @@ struct slim_sched {
struct mutex m_reconf; struct mutex m_reconf;
}; };
/**
* enum slim_port_direction: SLIMbus port direction
*
* @SLIM_PORT_SINK: SLIMbus port is a sink
* @SLIM_PORT_SOURCE: SLIMbus port is a source
*/
enum slim_port_direction {
SLIM_PORT_SINK = 0,
SLIM_PORT_SOURCE,
};
/**
* enum slim_port_state: SLIMbus Port/Endpoint state machine
* according to SLIMbus Spec 2.0
* @SLIM_PORT_DISCONNECTED: SLIMbus port is disconnected
* entered from Unconfigure/configured state after
* DISCONNECT_PORT or REMOVE_CHANNEL core command
* @SLIM_PORT_UNCONFIGURED: SLIMbus port is in unconfigured state.
* entered from disconnect state after CONNECT_SOURCE/SINK core command
* @SLIM_PORT_CONFIGURED: SLIMbus port is in configured state.
* entered from unconfigured state after DEFINE_CHANNEL, DEFINE_CONTENT
* and ACTIVATE_CHANNEL core commands. Ready for data transmission.
*/
enum slim_port_state {
SLIM_PORT_DISCONNECTED = 0,
SLIM_PORT_UNCONFIGURED,
SLIM_PORT_CONFIGURED,
};
/**
* enum slim_channel_state: SLIMbus channel state machine used by core.
* @SLIM_CH_STATE_DISCONNECTED: SLIMbus channel is disconnected
* @SLIM_CH_STATE_ALLOCATED: SLIMbus channel is allocated
* @SLIM_CH_STATE_ASSOCIATED: SLIMbus channel is associated with port
* @SLIM_CH_STATE_DEFINED: SLIMbus channel parameters are defined
* @SLIM_CH_STATE_CONTENT_DEFINED: SLIMbus channel content is defined
* @SLIM_CH_STATE_ACTIVE: SLIMbus channel is active and ready for data
* @SLIM_CH_STATE_REMOVED: SLIMbus channel is inactive and removed
*/
enum slim_channel_state {
SLIM_CH_STATE_DISCONNECTED = 0,
SLIM_CH_STATE_ALLOCATED,
SLIM_CH_STATE_ASSOCIATED,
SLIM_CH_STATE_DEFINED,
SLIM_CH_STATE_CONTENT_DEFINED,
SLIM_CH_STATE_ACTIVE,
SLIM_CH_STATE_REMOVED,
};
/**
* enum slim_ch_data_fmt: SLIMbus channel data Type identifiers according to
* Table 60 of SLIMbus Spec 1.01.01
* @SLIM_CH_DATA_FMT_NOT_DEFINED: Undefined
* @SLIM_CH_DATA_FMT_LPCM_AUDIO: LPCM audio
* @SLIM_CH_DATA_FMT_IEC61937_COMP_AUDIO: IEC61937 Compressed audio
* @SLIM_CH_DATA_FMT_PACKED_PDM_AUDIO: Packed PDM audio
*/
enum slim_ch_data_fmt {
SLIM_CH_DATA_FMT_NOT_DEFINED = 0,
SLIM_CH_DATA_FMT_LPCM_AUDIO = 1,
SLIM_CH_DATA_FMT_IEC61937_COMP_AUDIO = 2,
SLIM_CH_DATA_FMT_PACKED_PDM_AUDIO = 3,
};
/**
* enum slim_ch_aux_fmt: SLIMbus channel Aux Field format IDs according to
* Table 63 of SLIMbus Spec 2.0
* @SLIM_CH_AUX_FMT_NOT_APPLICABLE: Undefined
* @SLIM_CH_AUX_FMT_ZCUV_TUNNEL_IEC60958: ZCUV for tunneling IEC60958
* @SLIM_CH_AUX_FMT_USER_DEFINED: User defined
*/
enum slim_ch_aux_bit_fmt {
SLIM_CH_AUX_FMT_NOT_APPLICABLE = 0,
SLIM_CH_AUX_FMT_ZCUV_TUNNEL_IEC60958 = 1,
SLIM_CH_AUX_FMT_USER_DEFINED = 0xF,
};
/**
* struct slim_channel - SLIMbus channel, used for state machine
*
* @id: ID of channel
* @prrate: Presense rate of channel from Table 66 of SLIMbus 2.0 Specs
* @seg_dist: segment distribution code from Table 20 of SLIMbus 2.0 Specs
* @data_fmt: Data format of channel.
* @aux_fmt: Aux format for this channel.
* @state: channel state machine
*/
struct slim_channel {
int id;
int prrate;
int seg_dist;
enum slim_ch_data_fmt data_fmt;
enum slim_ch_aux_bit_fmt aux_fmt;
enum slim_channel_state state;
};
/**
* struct slim_port - SLIMbus port
*
* @id: Port id
* @direction: Port direction, Source or Sink.
* @state: state machine of port.
* @ch: channel associated with this port.
*/
struct slim_port {
int id;
enum slim_port_direction direction;
enum slim_port_state state;
struct slim_channel ch;
};
/**
* enum slim_transport_protocol: SLIMbus Transport protocol list from
* Table 47 of SLIMbus 2.0 specs.
* @SLIM_PROTO_ISO: Isochronous Protocol, no flow control as data rate match
* channel rate flow control embedded in the data.
* @SLIM_PROTO_PUSH: Pushed Protocol, includes flow control, Used to carry
* data whose rate is equal to, or lower than the channel rate.
* @SLIM_PROTO_PULL: Pulled Protocol, similar usage as pushed protocol
* but pull is a unicast.
* @SLIM_PROTO_LOCKED: Locked Protocol
* @SLIM_PROTO_ASYNC_SMPLX: Asynchronous Protocol-Simplex
* @SLIM_PROTO_ASYNC_HALF_DUP: Asynchronous Protocol-Half-duplex
* @SLIM_PROTO_EXT_SMPLX: Extended Asynchronous Protocol-Simplex
* @SLIM_PROTO_EXT_HALF_DUP: Extended Asynchronous Protocol-Half-duplex
*/
enum slim_transport_protocol {
SLIM_PROTO_ISO = 0,
SLIM_PROTO_PUSH,
SLIM_PROTO_PULL,
SLIM_PROTO_LOCKED,
SLIM_PROTO_ASYNC_SMPLX,
SLIM_PROTO_ASYNC_HALF_DUP,
SLIM_PROTO_EXT_SMPLX,
SLIM_PROTO_EXT_HALF_DUP,
};
/**
* struct slim_stream_runtime - SLIMbus stream runtime instance
*
* @dev: Name of the stream
* @dev: SLIM Device instance associated with this stream
* @state: state of stream
* @direction: direction of stream
* @prot: Transport protocol used in this stream
* @rate: Data rate of samples *
* @bps: bits per sample
* @ratem: rate multipler which is super frame rate/data rate
* @num_ports: number of ports
* @ports: pointer to instance of ports
* @node: list head for stream associated with slim device.
*/
struct slim_stream_runtime {
const char *name;
struct slim_device *dev;
int direction;
enum slim_transport_protocol prot;
unsigned int rate;
unsigned int bps;
unsigned int ratem;
int num_ports;
struct slim_port *ports;
struct list_head node;
};
/** /**
* struct slim_controller - Controls every instance of SLIMbus * struct slim_controller - Controls every instance of SLIMbus
* (similar to 'master' on SPI) * (similar to 'master' on SPI)
...@@ -196,6 +379,10 @@ struct slim_sched { ...@@ -196,6 +379,10 @@ struct slim_sched {
* @wakeup: This function pointer implements controller-specific procedure * @wakeup: This function pointer implements controller-specific procedure
* to wake it up from clock-pause. Framework will call this to bring * to wake it up from clock-pause. Framework will call this to bring
* the controller out of clock pause. * the controller out of clock pause.
* @enable_stream: This function pointer implements controller-specific procedure
* to enable a stream.
* @disable_stream: This function pointer implements controller-specific procedure
* to disable stream.
* *
* 'Manager device' is responsible for device management, bandwidth * 'Manager device' is responsible for device management, bandwidth
* allocation, channel setup, and port associations per channel. * allocation, channel setup, and port associations per channel.
...@@ -237,6 +424,8 @@ struct slim_controller { ...@@ -237,6 +424,8 @@ struct slim_controller {
struct slim_eaddr *ea, u8 laddr); struct slim_eaddr *ea, u8 laddr);
int (*get_laddr)(struct slim_controller *ctrl, int (*get_laddr)(struct slim_controller *ctrl,
struct slim_eaddr *ea, u8 *laddr); struct slim_eaddr *ea, u8 *laddr);
int (*enable_stream)(struct slim_stream_runtime *rt);
int (*disable_stream)(struct slim_stream_runtime *rt);
int (*wakeup)(struct slim_controller *ctrl); int (*wakeup)(struct slim_controller *ctrl);
}; };
......
This diff is collapsed.
...@@ -48,6 +48,8 @@ struct slim_controller; ...@@ -48,6 +48,8 @@ struct slim_controller;
* @ctrl: slim controller instance. * @ctrl: slim controller instance.
* @laddr: 1-byte Logical address of this device. * @laddr: 1-byte Logical address of this device.
* @is_laddr_valid: indicates if the laddr is valid or not * @is_laddr_valid: indicates if the laddr is valid or not
* @stream_list: List of streams on this device
* @stream_list_lock: lock to protect the stream list
* *
* This is the client/device handle returned when a SLIMbus * This is the client/device handle returned when a SLIMbus
* device is registered with a controller. * device is registered with a controller.
...@@ -60,6 +62,8 @@ struct slim_device { ...@@ -60,6 +62,8 @@ struct slim_device {
enum slim_device_status status; enum slim_device_status status;
u8 laddr; u8 laddr;
bool is_laddr_valid; bool is_laddr_valid;
struct list_head stream_list;
spinlock_t stream_list_lock;
}; };
#define to_slim_device(d) container_of(d, struct slim_device, dev) #define to_slim_device(d) container_of(d, struct slim_device, dev)
...@@ -108,6 +112,36 @@ struct slim_val_inf { ...@@ -108,6 +112,36 @@ struct slim_val_inf {
struct completion *comp; struct completion *comp;
}; };
#define SLIM_DEVICE_MAX_CHANNELS 256
/* A SLIMBus Device may have frmo 0 to 31 Ports (inclusive) */
#define SLIM_DEVICE_MAX_PORTS 32
/**
* struct slim_stream_config - SLIMbus stream configuration
* Configuring a stream is done at hw_params or prepare call
* from audio drivers where they have all the required information
* regarding rate, number of channels and so on.
* There is a 1:1 mapping of channel and ports.
*
* @rate: data rate
* @bps: bits per data sample
* @ch_count: number of channels
* @chs: pointer to list of channel numbers
* @port_mask: port mask of ports to use for this stream
* @direction: direction of the stream, SNDRV_PCM_STREAM_PLAYBACK
* or SNDRV_PCM_STREAM_CAPTURE.
*/
struct slim_stream_config {
unsigned int rate;
unsigned int bps;
/* MAX 256 channels */
unsigned int ch_count;
unsigned int *chs;
/* Max 32 ports per device */
unsigned long port_mask;
int direction;
};
/* /*
* use a macro to avoid include chaining to get THIS_MODULE * use a macro to avoid include chaining to get THIS_MODULE
*/ */
...@@ -163,4 +197,16 @@ int slim_readb(struct slim_device *sdev, u32 addr); ...@@ -163,4 +197,16 @@ int slim_readb(struct slim_device *sdev, u32 addr);
int slim_writeb(struct slim_device *sdev, u32 addr, u8 value); int slim_writeb(struct slim_device *sdev, u32 addr, u8 value);
int slim_read(struct slim_device *sdev, u32 addr, size_t count, u8 *val); int slim_read(struct slim_device *sdev, u32 addr, size_t count, u8 *val);
int slim_write(struct slim_device *sdev, u32 addr, size_t count, u8 *val); int slim_write(struct slim_device *sdev, u32 addr, size_t count, u8 *val);
/* SLIMbus Stream apis */
struct slim_stream_runtime;
struct slim_stream_runtime *slim_stream_allocate(struct slim_device *dev,
const char *sname);
int slim_stream_prepare(struct slim_stream_runtime *stream,
struct slim_stream_config *c);
int slim_stream_enable(struct slim_stream_runtime *stream);
int slim_stream_disable(struct slim_stream_runtime *stream);
int slim_stream_unprepare(struct slim_stream_runtime *stream);
int slim_stream_free(struct slim_stream_runtime *stream);
#endif /* _LINUX_SLIMBUS_H */ #endif /* _LINUX_SLIMBUS_H */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment