2020-03-05 22:28:23 -06:00
/* SPDX-License-Identifier: GPL-2.0 */
/* Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
2022-09-30 17:45:49 -05:00
* Copyright ( C ) 2019 - 2022 Linaro Ltd .
2020-03-05 22:28:23 -06:00
*/
# ifndef _GSI_TRANS_H_
# define _GSI_TRANS_H_
# include <linux/types.h>
# include <linux/refcount.h>
# include <linux/completion.h>
# include <linux/dma-direction.h>
# include "ipa_cmd.h"
2021-01-07 17:34:03 -06:00
struct page ;
2020-03-05 22:28:23 -06:00
struct scatterlist ;
struct device ;
struct sk_buff ;
struct gsi ;
struct gsi_trans ;
struct gsi_trans_pool ;
2022-05-21 19:32:22 -05:00
/* Maximum number of TREs in an IPA immediate command transaction */
# define IPA_COMMAND_TRANS_TRE_MAX 8
2020-03-05 22:28:23 -06:00
/**
* struct gsi_trans - a GSI transaction
*
* Most fields in this structure for internal use by the transaction core code :
* @ gsi : GSI pointer
* @ channel_id : Channel number transaction is associated with
* @ cancelled : If set by the core code , transaction was cancelled
2022-06-13 12:17:55 -05:00
* @ rsvd_count : Number of TREs reserved for this transaction
* @ used_count : Number of TREs * used * ( could be less than rsvd_count )
2022-06-13 12:17:56 -05:00
* @ len : Number of bytes sent or received by the transaction
2020-03-05 22:28:23 -06:00
* @ data : Preserved but not touched by the core transaction code
2022-05-21 19:32:22 -05:00
* @ cmd_opcode : Array of command opcodes ( command channel only )
2020-03-05 22:28:23 -06:00
* @ sgl : An array of scatter / gather entries managed by core code
* @ direction : DMA transfer direction ( DMA_NONE for commands )
* @ refcount : Reference count used for destruction
* @ completion : Completed when the transaction completes
* @ byte_count : TX channel byte count recorded when transaction committed
* @ trans_count : Channel transaction count when committed ( for BQL accounting )
*
2022-06-13 12:17:56 -05:00
* The @ len field is set when the transaction is committed . For RX
* transactions it is updated later to reflect the actual number of bytes
* received .
2020-03-05 22:28:23 -06:00
*/
struct gsi_trans {
struct gsi * gsi ;
u8 channel_id ;
bool cancelled ; /* true if transaction was cancelled */
2022-06-13 12:17:55 -05:00
u8 rsvd_count ; /* # TREs requested */
u8 used_count ; /* # entries used in sgl[] */
2020-03-05 22:28:23 -06:00
u32 len ; /* total # bytes across sgl[] */
2022-05-21 19:32:23 -05:00
union {
void * data ;
u8 cmd_opcode [ IPA_COMMAND_TRANS_TRE_MAX ] ;
} ;
2020-03-05 22:28:23 -06:00
struct scatterlist * sgl ;
enum dma_data_direction direction ;
refcount_t refcount ;
struct completion completion ;
u64 byte_count ; /* channel byte_count when committed */
u64 trans_count ; /* channel trans_count when committed */
} ;
/**
* gsi_trans_pool_init ( ) - Initialize a pool of structures for transactions
2022-09-30 17:45:27 -05:00
* @ pool : GSI transaction pool pointer
2020-03-05 22:28:23 -06:00
* @ size : Size of elements in the pool
* @ count : Minimum number of elements in the pool
* @ max_alloc : Maximum number of elements allocated at a time from pool
*
2020-07-13 07:24:18 -05:00
* Return : 0 if successful , or a negative error code
2020-03-05 22:28:23 -06:00
*/
int gsi_trans_pool_init ( struct gsi_trans_pool * pool , size_t size , u32 count ,
u32 max_alloc ) ;
/**
* gsi_trans_pool_alloc ( ) - Allocate one or more elements from a pool
* @ pool : Pool pointer
* @ count : Number of elements to allocate from the pool
*
2020-07-13 07:24:18 -05:00
* Return : Virtual address of element ( s ) allocated from the pool
2020-03-05 22:28:23 -06:00
*/
void * gsi_trans_pool_alloc ( struct gsi_trans_pool * pool , u32 count ) ;
/**
* gsi_trans_pool_exit ( ) - Inverse of gsi_trans_pool_init ( )
* @ pool : Pool pointer
*/
void gsi_trans_pool_exit ( struct gsi_trans_pool * pool ) ;
/**
* gsi_trans_pool_init_dma ( ) - Initialize a pool of DMA - able structures
* @ dev : Device used for DMA
* @ pool : Pool pointer
* @ size : Size of elements in the pool
* @ count : Minimum number of elements in the pool
* @ max_alloc : Maximum number of elements allocated at a time from pool
*
2020-07-13 07:24:18 -05:00
* Return : 0 if successful , or a negative error code
2020-03-05 22:28:23 -06:00
*
* Structures in this pool reside in DMA - coherent memory .
*/
int gsi_trans_pool_init_dma ( struct device * dev , struct gsi_trans_pool * pool ,
size_t size , u32 count , u32 max_alloc ) ;
/**
* gsi_trans_pool_alloc_dma ( ) - Allocate an element from a DMA pool
* @ pool : DMA pool pointer
* @ addr : DMA address " handle " associated with the allocation
*
2020-07-13 07:24:18 -05:00
* Return : Virtual address of element allocated from the pool
2020-03-05 22:28:23 -06:00
*
* Only one element at a time may be allocated from a DMA pool .
*/
void * gsi_trans_pool_alloc_dma ( struct gsi_trans_pool * pool , dma_addr_t * addr ) ;
/**
2021-03-28 12:31:05 -05:00
* gsi_trans_pool_exit_dma ( ) - Inverse of gsi_trans_pool_init_dma ( )
* @ dev : Device used for DMA
2020-03-05 22:28:23 -06:00
* @ pool : Pool pointer
*/
void gsi_trans_pool_exit_dma ( struct device * dev , struct gsi_trans_pool * pool ) ;
2022-02-03 11:09:24 -06:00
/**
* gsi_channel_trans_idle ( ) - Return whether no transactions are allocated
* @ gsi : GSI pointer
* @ channel_id : Channel the transaction is associated with
*
* Return : True if no transactions are allocated , false otherwise
*
*/
bool gsi_channel_trans_idle ( struct gsi * gsi , u32 channel_id ) ;
2020-03-05 22:28:23 -06:00
/**
* gsi_channel_trans_alloc ( ) - Allocate a GSI transaction on a channel
* @ gsi : GSI pointer
* @ channel_id : Channel the transaction is associated with
* @ tre_count : Number of elements in the transaction
* @ direction : DMA direction for entire SGL ( or DMA_NONE )
*
2020-07-13 07:24:18 -05:00
* Return : A GSI transaction structure , or a null pointer if all
2020-03-05 22:28:23 -06:00
* available transactions are in use
*/
struct gsi_trans * gsi_channel_trans_alloc ( struct gsi * gsi , u32 channel_id ,
u32 tre_count ,
enum dma_data_direction direction ) ;
/**
* gsi_trans_free ( ) - Free a previously - allocated GSI transaction
* @ trans : Transaction to be freed
*/
void gsi_trans_free ( struct gsi_trans * trans ) ;
/**
* gsi_trans_cmd_add ( ) - Add an immediate command to a transaction
* @ trans : Transaction
* @ buf : Buffer pointer for command payload
* @ size : Number of bytes in buffer
* @ addr : DMA address for payload
* @ opcode : IPA immediate command opcode
*/
void gsi_trans_cmd_add ( struct gsi_trans * trans , void * buf , u32 size ,
2022-05-21 19:32:21 -05:00
dma_addr_t addr , enum ipa_cmd_opcode opcode ) ;
2020-03-05 22:28:23 -06:00
/**
* gsi_trans_page_add ( ) - Add a page transfer to a transaction
* @ trans : Transaction
* @ page : Page pointer
* @ size : Number of bytes ( starting at offset ) to transfer
* @ offset : Offset within page for start of transfer
*/
int gsi_trans_page_add ( struct gsi_trans * trans , struct page * page , u32 size ,
u32 offset ) ;
/**
* gsi_trans_skb_add ( ) - Add a socket transfer to a transaction
* @ trans : Transaction
* @ skb : Socket buffer for transfer ( outbound )
*
2020-07-13 07:24:18 -05:00
* Return : 0 , or - EMSGSIZE if socket data won ' t fit in transaction .
2020-03-05 22:28:23 -06:00
*/
int gsi_trans_skb_add ( struct gsi_trans * trans , struct sk_buff * skb ) ;
/**
* gsi_trans_commit ( ) - Commit a GSI transaction
* @ trans : Transaction to commit
* @ ring_db : Whether to tell the hardware about these queued transfers
*/
void gsi_trans_commit ( struct gsi_trans * trans , bool ring_db ) ;
/**
* gsi_trans_commit_wait ( ) - Commit a GSI transaction and wait for it
* to complete
* @ trans : Transaction to commit
*/
void gsi_trans_commit_wait ( struct gsi_trans * trans ) ;
/**
* gsi_trans_read_byte ( ) - Issue a single byte read TRE on a channel
* @ gsi : GSI pointer
* @ channel_id : Channel on which to read a byte
* @ addr : DMA address into which to transfer the one byte
*
* This is not a transaction operation at all . It ' s defined here because
* it needs to be done in coordination with other transaction activity .
*/
int gsi_trans_read_byte ( struct gsi * gsi , u32 channel_id , dma_addr_t addr ) ;
/**
* gsi_trans_read_byte_done ( ) - Clean up after a single byte read TRE
* @ gsi : GSI pointer
* @ channel_id : Channel on which byte was read
*
* This function needs to be called to signal that the work related
* to reading a byte initiated by gsi_trans_read_byte ( ) is complete .
*/
void gsi_trans_read_byte_done ( struct gsi * gsi , u32 channel_id ) ;
# endif /* _GSI_TRANS_H_ */