RE: [V5, 2/6] fsl/fman: Add FMan support
From: Liberman Igal <hidden>
Date: 2015-10-27 18:05:10
Also in:
linuxppc-dev, lkml
Possibly related (same subject, not in this thread)
- 2015-10-02 · Re: [v5, 2/6] fsl/fman: Add FMan support · Scott Wood <hidden>
- 2015-09-24 · [v5, 2/6] fsl/fman: Add FMan support · <hidden>
Regards, Igal Liberman
-----Original Message----- From: Wood Scott-B07421 Sent: Saturday, September 26, 2015 2:02 AM To: Liberman Igal-B31950 <redacted> Cc: netdev@vger.kernel.org; linuxppc-dev@lists.ozlabs.org; linux- kernel@vger.kernel.org; Bucur Madalin-Cristian-B32716 [off-list ref] Subject: Re: [V5, 2/6] fsl/fman: Add FMan support On Mon, Sep 21, 2015 at 02:52:34PM +0300, Igal.Liberman wrote:quoted
diff --git a/drivers/net/ethernet/freescale/fman/fman.cb/drivers/net/ethernet/freescale/fman/fman.c new file mode 100644 index 0000000..924685f--- /dev/null +++ b/drivers/net/ethernet/freescale/fman/fman.c@@ -0,0 +1,2738 @@ +/* + * Copyright 2008-2015 Freescale Semiconductor Inc. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions aremet:quoted
+ * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with thedistribution.quoted
+ * * Neither the name of Freescale Semiconductor nor the + * names of its contributors may be used to endorse or promoteproductsquoted
+ * derived from this software without specific prior written permission. + * +// * + * ALTERNATIVELY, this software may be distributed under the terms of +the + * GNU General Public License ("GPL") as published by the Free +Software + * Foundation, either version 2 of that License or (at your option) +any + * later version.What is that // doing there?
Removed.
quoted
+/* Exceptions bit map */ +#define EX_DMA_BUS_ERROR 0x80000000 +#define EX_DMA_READ_ECC 0x40000000 +#define EX_DMA_SYSTEM_WRITE_ECC 0x20000000 +#define EX_DMA_FM_WRITE_ECC 0x10000000 +#define EX_FPM_STALL_ON_TASKS 0x08000000 +#define EX_FPM_SINGLE_ECC 0x04000000 +#define EX_FPM_DOUBLE_ECC 0x02000000 +#define EX_QMI_SINGLE_ECC 0x01000000 +#define EX_QMI_DEQ_FROM_UNKNOWN_PORTID 0x00800000 +#define EX_QMI_DOUBLE_ECC 0x00400000 +#define EX_BMI_LIST_RAM_ECC 0x00200000 +#define EX_BMI_STORAGE_PROFILE_ECC 0x00100000 +#define EX_BMI_STATISTICS_RAM_ECC 0x00080000 +#define EX_IRAM_ECC 0x00040000 +#define EX_MURAM_ECC 0x00020000 +#define EX_BMI_DISPATCH_RAM_ECC 0x00010000 +#define EX_DMA_SINGLE_PORT_ECC 0x00008000 + +#define DFLT_EXCEPTIONS \ + ((EX_DMA_BUS_ERROR) | \ + (EX_DMA_READ_ECC) | \ + (EX_DMA_SYSTEM_WRITE_ECC) | \ + (EX_DMA_FM_WRITE_ECC) | \ + (EX_FPM_STALL_ON_TASKS) | \ + (EX_FPM_SINGLE_ECC) | \ + (EX_FPM_DOUBLE_ECC) | \ + (EX_QMI_DEQ_FROM_UNKNOWN_PORTID) | \ + (EX_BMI_LIST_RAM_ECC) | \ + (EX_BMI_STORAGE_PROFILE_ECC) | \ + (EX_BMI_STATISTICS_RAM_ECC) | \ + (EX_MURAM_ECC) | \ + (EX_BMI_DISPATCH_RAM_ECC) | \ + (EX_QMI_DOUBLE_ECC) | \ + (EX_QMI_SINGLE_ECC))You don't need parentheses around each symbol.
Removed the parentheses (here and in other places)
This is only used in one place -- why put the list here rather than in the place where it's used?
Moved this define.
quoted
+struct fman_state_struct { + u8 fm_id; + u16 fm_clk_freq; + struct fman_rev_info rev_info; + bool enabled_time_stamp; + u8 count1_micro_bit; + u8 total_num_of_tasks; + u8 accumulated_num_of_tasks; + u32 accumulated_fifo_size; + u8 accumulated_num_of_open_dmas; + u8 accumulated_num_of_deq_tnums; + bool low_end_restriction; + u32 exceptions; + u32 extra_fifo_pool_size; + u8 extra_tasks_pool_size; + u8 extra_open_dmas_pool_size; + u16 port_mfl[MAX_NUM_OF_MACS]; + u16 mac_mfl[MAX_NUM_OF_MACS]; + + /* SOC specific */ + u32 fm_iram_size; + /* DMA */ + u32 dma_thresh_max_commq; + u32 dma_thresh_max_buf; + u32 max_num_of_open_dmas; + /* QMI */ + u32 qmi_max_num_of_tnums; + u32 qmi_def_tnums_thresh; + /* BMI */ + u32 bmi_max_num_of_tasks; + u32 bmi_max_fifo_size; + /* General */ + u32 fm_port_num_of_cg; + u32 num_of_rx_ports; + u32 total_fifo_size; + + u32 qman_channel_base; + u32 num_of_qman_channels; + + struct resource *res; +}; + +struct fman_cfg { + u8 disp_limit_tsh; + u8 prs_disp_tsh; + u8 plcr_disp_tsh; + u8 kg_disp_tsh; + u8 bmi_disp_tsh; + u8 qmi_enq_disp_tsh; + u8 qmi_deq_disp_tsh; + u8 fm_ctl1_disp_tsh; + u8 fm_ctl2_disp_tsh; + int dma_cache_override; + enum fman_dma_aid_mode dma_aid_mode; + bool dma_aid_override; + u32 dma_axi_dbg_num_of_beats; + u32 dma_cam_num_of_entries; + u32 dma_watchdog; + u8 dma_comm_qtsh_asrt_emer; + u32 dma_write_buf_tsh_asrt_emer; + u32 dma_read_buf_tsh_asrt_emer; + u8 dma_comm_qtsh_clr_emer; + u32 dma_write_buf_tsh_clr_emer; + u32 dma_read_buf_tsh_clr_emer; + u32 dma_sos_emergency; + int dma_dbg_cnt_mode; + bool dma_stop_on_bus_error; + bool dma_en_emergency; + u32 dma_emergency_bus_select; + int dma_emergency_level; + bool dma_en_emergency_smoother; + u32 dma_emergency_switch_counter; + bool halt_on_external_activ; + bool halt_on_unrecov_ecc_err; + int catastrophic_err; + int dma_err; + bool en_muram_test_mode; + bool en_iram_test_mode; + bool external_ecc_rams_enable; + u16 tnum_aging_period; + u32 exceptions; + u16 clk_freq; + bool pedantic_dma; + u32 cam_base_addr; + u32 fifo_base_addr; + u32 total_fifo_size; + u32 total_num_of_tasks; + bool qmi_deq_option_support; + u32 qmi_def_tnums_thresh; +};Some documentation on this stuff would be nice.quoted
+static inline u8 hw_port_id_to_sw_port_id(u8 major, u8 hw_port_id) { + u8 sw_port_id = 0; + + if (hw_port_id >= BASE_TX_PORTID) { + sw_port_id = hw_port_id - BASE_TX_PORTID; + } else if (hw_port_id >= BASE_RX_PORTID) { + sw_port_id = hw_port_id - BASE_RX_PORTID; + } else { + sw_port_id = 0; + WARN_ON(false);WARN_ON(false) is a no-op.
Removed.
quoted
+ } + + return sw_port_id; +} + +static void set_port_order_restoration(struct fman_fpm_regs __iomem*fpm_rg,quoted
+ u8 port_id) +{ + u32 tmp = 0; + + tmp = (u32)(port_id << FPM_PORT_FM_CTL_PORTID_SHIFT);Unnecessary cast. Likewise elsewhere.
Removed.
quoted
+ + tmp |= (FPM_PRT_FM_CTL2 | FPM_PRT_FM_CTL1); + + /* order restoration */ + if (port_id % 2) + tmp |= (FPM_PRT_FM_CTL1 <<FPM_PRC_ORA_FM_CTL_SEL_SHIFT);quoted
+ else + tmp |= (FPM_PRT_FM_CTL2 <<FPM_PRC_ORA_FM_CTL_SEL_SHIFT); Unnecessary parens.
Removed.
quoted
+static int get_module_event(enum fman_event_modules module, u8mod_id,quoted
+ enum fman_intr_type intr_type) { + int event; + + switch (module) { + case FMAN_MOD_MAC: + event = (intr_type == FMAN_INTR_TYPE_ERR) ? + (FMAN_EV_ERR_MAC0 + mod_id) : + (FMAN_EV_MAC0 + mod_id);Use if/else...
Done.
quoted
+ break; + case FMAN_MOD_FMAN_CTRL: + if (intr_type == FMAN_INTR_TYPE_ERR) + event = FMAN_EV_CNT; + else + event = (FMAN_EV_FMAN_CTRL_0 + mod_id); + break;...just like here.quoted
+ /* Read, modify and write to HW */ + tmp = (u32)((fifo / FMAN_BMI_FIFO_UNITS - 1) | + ((extra_fifo / FMAN_BMI_FIFO_UNITS) << + BMI_EXTRA_FIFO_SIZE_SHIFT));Unnecessary cast.
Removed.
quoted
+ if (extra_tasks) + fman->state->extra_tasks_pool_size = + (u8)max(fman->state->extra_tasks_pool_size, extra_tasks);Unnecessary cast.
Removed.
quoted
+static int fman_init(struct fman *fman) { + struct fman_cfg *cfg = NULL; + struct fman_rg fman_rg; + int err = 0, i; + + if (is_init_done(fman->cfg)) + return -EINVAL; + + fman_rg.bmi_rg = fman->bmi_regs; + fman_rg.qmi_rg = fman->qmi_regs; + fman_rg.fpm_rg = fman->fpm_regs; + fman_rg.dma_rg = fman->dma_regs;Why keep this information in two different places and formats?
Removed struct fman_rg. The pointers to the memory map are saved only in struct fman now.
quoted
+ /* Reset the FM if required. */ + if (fman->reset_on_init) {When is this ever not true?
Removed the check, always preform reset.
quoted
+ if (fman->state->rev_info.major >= 6) { + /* Errata A007273 */ + pr_debug("FManV3 reset is not supported!\n");No plan to implement the workaround involving DEVDISR2?
Not now. This is probably required for loadable module support. I'll consider adding a workaround if needed.
quoted
+ } else { + out_be32(&fman->fpm_regs->fm_rstc,FPM_RSTC_FM_RESET);quoted
+ /* Memory barrier */ + mb(); + usleep_range(100, 300); + }Where does 100us come from? Shouldn't you wait for the FM_RESET bit to be cleared?
Changed the way the driver waits for FMan reset completion.
quoted
+ + if (!!(ioread32be(&fman_rg.qmi_rg->fmqm_gs) & + QMI_GS_HALT_NOT_BUSY)) { + resume(fman->fpm_regs); + usleep_range(100, 300); + }There's no need for !! here.
Removed.
Same question as above regarding the delay.
Changed the way the driver waits for resume completion.
quoted
+static int fman_set_exception(struct fman *fman, + enum fman_exceptions exception, bool enable) { + u32 bit_mask = 0; + struct fman_rg fman_rg; + + if (!is_init_done(fman->cfg)) + return -EINVAL; + + fman_rg.bmi_rg = fman->bmi_regs; + fman_rg.qmi_rg = fman->qmi_regs; + fman_rg.fpm_rg = fman->fpm_regs; + fman_rg.dma_rg = fman->dma_regs; + + bit_mask = get_exception_flag(exception); + if (bit_mask) { + if (enable) + fman->state->exceptions |= bit_mask; + else + fman->state->exceptions &= ~bit_mask; + } else { + pr_err("Undefined exception\n"); + return -EINVAL;This is not a useful error message. Use dev_err, __func__, and print the unexpected value. Likewise elsewhere.
Changed to dev_err/warn/debug where possible.
quoted
+ } + + return set_exckeption(&fman_rg, exception, enable); } + +void fman_register_intr(struct fman *fman, enum fman_event_modulesmodule,quoted
+ u8 mod_id, enum fman_intr_type intr_type, + void (*isr_cb)(void *src_arg), void *src_arg) { + int event = 0; + + event = get_module_event(module, mod_id, intr_type); + WARN_ON(!(event < FMAN_EV_CNT));This isn't floating point. You can safely say WARN_ON(event >= FMAN_EV_CNT). :-P
Done.
quoted
+ spin_lock_irqsave(&fman->spinlock, int_flags);"flags", rather than "int_flags" is idiomatic.
Replaced int_flags with flags.
quoted
+ + err = set_num_of_tasks(fman, port_params->port_id, + &port_params->num_of_tasks, + &port_params->num_of_extra_tasks); + if (err) { + spin_unlock_irqrestore(&fman->spinlock, int_flags); + return err; + }Use the standard goto error handling model.
Done.
quoted
+ /* if deq_th is too small, we enlarge it to the min + * value that is still 0. + * depTh may not be larger than 63 + * (fman->state->qmi_max_num_of_tnums-1). + */ + if ((deq_th <= fman->state- accumulated_num_of_deq_tnums) && + (deq_th < fman->state->qmi_max_num_of_tnums - 1)) { + deq_th = + fman->state- accumulated_num_of_deq_tnums + 1; + reg = ioread32be(&fman_rg.qmi_rg->fmqm_gc);Whitespace
removed.
quoted
+/* Max frame size, across all interfaces. + * Configurable from bootargs, to avoid allocating oversized (socket) + * buffers when not using jumbo frames. + * Must be large enough to accommodate the network MTU, but small +enough + * to avoid wasting skb memory. + * + * Could be overridden once, at boot-time, via the + * fm_set_max_frm() callback. + */ +int fsl_fm_max_frm = FSL_FM_MAX_FRAME_SIZE; +module_param(fsl_fm_max_frm, int, 0); +MODULE_PARM_DESC(fsl_fm_max_frm, "Maximum frame size, across all +interfaces"); + +u16 fman_get_max_frm(void) +{ + static bool fm_check_mfl; + + if (!fm_check_mfl) { + if (fsl_fm_max_frm > FSL_FM_MAX_POSSIBLE_FRAME_SIZE||quoted
+ fsl_fm_max_frm < FSL_FM_MIN_POSSIBLE_FRAME_SIZE){quoted
+ pr_warn("Invalid fsl_fm_max_frm value (%d) inbootargs, valid range is %d-%d. Falling back to the default (%d)\n",quoted
+ fsl_fm_max_frm, + FSL_FM_MIN_POSSIBLE_FRAME_SIZE, + FSL_FM_MAX_POSSIBLE_FRAME_SIZE, + FSL_FM_MAX_FRAME_SIZE); + fsl_fm_max_frm = FSL_FM_MAX_FRAME_SIZE; + } + fm_check_mfl = true; + } + + return fsl_fm_max_frm; +} +EXPORT_SYMBOL(fman_get_max_frm); + +int fman_get_rx_extra_headroom(void) +{ + static bool fm_check_rx_extra_headroom; + + if (!fm_check_rx_extra_headroom) { + if (fsl_fm_rx_extra_headroom >FSL_FM_RX_EXTRA_HEADROOM_MAX ||quoted
+ fsl_fm_rx_extra_headroom <FSL_FM_RX_EXTRA_HEADROOM_MIN) {quoted
+ pr_warn("Invalid fsl_fm_rx_extra_headroom value(%d) in bootargs, valid range is %d-%d. Falling back to the default (%d)\n",quoted
+ fsl_fm_rx_extra_headroom, + FSL_FM_RX_EXTRA_HEADROOM_MIN, + FSL_FM_RX_EXTRA_HEADROOM_MAX, + FSL_FM_RX_EXTRA_HEADROOM); + fsl_fm_rx_extra_headroom =FSL_FM_RX_EXTRA_HEADROOM;quoted
+ } + + fsl_fm_rx_extra_headroom = true; + fsl_fm_rx_extra_headroom =ALIGN(fsl_fm_rx_extra_headroom, 16);quoted
+ } + + return fsl_fm_rx_extra_headroom; +} +EXPORT_SYMBOL(fman_get_rx_extra_headroom); +What calls these functions?
DPAA Ethernet driver.
quoted
+struct fman *fman_bind(struct device *fm_dev) { + return (struct fman *)(dev_get_drvdata(get_device(fm_dev))); +} + +void fman_unbind(struct fman *fman) +{ + put_device(fman->dev); +}Why?
It's was used by the MAC, but it's not mandatory for now, removed.
quoted
+ +struct device *fman_get_device(struct fman *fman) { + return fman->dev; +}Is this really necessary?
Fman port needs fman->dev, fman structure is opaque, so yes, it's needed.
quoted
+static irqreturn_t fman_irq(int irq, void *fman) { + fman_event_isr(fman); + + return IRQ_HANDLED; +}Only return IRQ_HANDLED if something was actually handled.
Done.
quoted
+static const struct of_device_id fman_muram_match[] = { + { + .compatible = "fsl,fman-muram"}, + {}k> +}; Whitespace
Removed.
quoted
+ clk = of_clk_get_by_name(fm_node, NULL); + if (IS_ERR(clk)) { + pr_err("Failed to get FM%d clock structure\n", + fman->dts_params.id); + goto fman_node_put; + }Why is name NULL? If you just want to get the first/only clock without needing a name, why not use of_clk_get(fm_node, 0)?
It's possible, done.
quoted
+ + clk_rate = clk_get_rate(clk); + if (!clk_rate) { + pr_err("Failed to determine FM%d clock rate\n", + fman->dts_params.id); + goto fman_node_put; + } + /* Rounding to MHz */ + fman->dts_params.clk_freq = (u16)((clk_rate + 500000) / 1000000);DIV_ROUND_UP()
OK.
quoted
+ + u32_prop = (const u32 *)of_get_property(fm_node, + "fsl,qman-channel-range", + &lenp); + if (!u32_prop) { + pr_err("of_get_property(%s, fsl,qman-channel-range)failed\n",quoted
+ fm_node->full_name); + goto fman_node_put; + } + if (WARN_ON(lenp != sizeof(u32) * 2)) + goto fman_node_put; + fman->dts_params.qman_channel_base = u32_prop[0]; + fman->dts_params.num_of_qman_channels = u32_prop[1];fdt32_to_cpu()
Done.
quoted
+ + /* Get the MURAM base address and size */ + muram_node = of_find_matching_node(fm_node,fman_muram_match);quoted
+ if (!muram_node) { + pr_err("could not find MURAM node\n"); + goto fman_node_put; + } + + err = of_address_to_resource(muram_node, 0, res); + if (err) { + of_node_put(muram_node); + pr_err("of_address_to_resource() = %d\n", err); + goto fman_node_put; + } + + fman->dts_params.muram_phy_base_addr = res->start; + fman->dts_params.muram_size = res->end + 1 - res->start;Why not just put a struct resource in fman->dts_params?
Changed this code. dts_params holds resource structure for MURAM. In addition, I'm using resource_size instead of calculating the size.
quoted
+ { + /* In B4 rev 2.0 (and above) the MURAM size is 512KB. + * Check the SVR and update MURAM size if required. + */ + u32 svr; + + svr = mfspr(SPRN_SVR); + + if ((SVR_SOC_VER(svr) == SVR_B4860) && (SVR_MAJ(svr) >=2))quoted
+ fman->dts_params.muram_size = 0x80000; + }Why wasn't the MURAM size described in the device tree, as it was with CPM/QE?
MURAM size described by the device-tree. In B4860 rev 2.0 (and above) MURAM size is bigger. This is workaround, in order to have the same device tree for all B4860 revisions.
quoted
+ + of_node_put(muram_node); + of_node_put(fm_node); + + err = devm_request_irq(&of_dev->dev, irq, fman_irq, + IRQF_NO_SUSPEND, "fman", fman); + if (err < 0) { + pr_err("Error: allocating irq %d (error = %d)\n", irq, err); + goto fman_free; + }Why IRQF_NO_SUSPEND?
It shouldn't be IRQF_NO_SUSPEND for now, removed.
Also please use dev_err where possible.
Done.
quoted
+static const struct of_device_id fman_match[] = { + { + .compatible = "fsl,fman"}, + {} +};Whitespace
Removed.
quoted
+ +MODULE_DEVICE_TABLE(of, fm_match); + +static struct platform_driver fman_driver = { + .driver = { + .name = "fsl-fman", + .of_match_table = fman_match, + }, + .probe = fman_probe, +};Whitespace
Removed.
quoted
+/* Parse results memory layout */ +struct fman_prs_result { + u8 lpid; /* Logical port id */ + u8 shimr; /* Shim header result */ + u16 l2r; /* Layer 2 result */ + u16 l3r; /* Layer 3 result */ + u8 l4r; /* Layer 4 result */ + u8 cplan; /* Classification plan id */ + u16 nxthdr; /* Next Header */ + u16 cksum; /* Running-sum */ + /* Flags&fragment-offset field of the last IP-header */ + u16 flags_frag_off; + /* Routing type field of a IPV6 routing extension header */ + u8 route_type; + /* Routing Extension Header Present; last bit is IP valid */ + u8 rhp_ip_valid; + u8 shim_off[2]; /* Shim offset */ + u8 ip_pid_off; /* IP PID (last IP-proto) offset */ + u8 eth_off; /* ETH offset */ + u8 llc_snap_off; /* LLC_SNAP offset */ + u8 vlan_off[2]; /* VLAN offset */ + u8 etype_off; /* ETYPE offset */ + u8 pppoe_off; /* PPP offset */ + u8 mpls_off[2]; /* MPLS offset */ + u8 ip_off[2]; /* IP offset */ + u8 gre_off; /* GRE offset */ + u8 l4_off; /* Layer 4 offset */ + u8 nxthdr_off; /** Parser end point */ +} __attribute__((__packed__));Why is this packed?
Removed __packed__ attribute.
Why does "Parser end point" have a kerneldoc comment?
Removed.
quoted
+ +/** + * fman_get_revision + * @fman - Pointer to the FMan module + * @rev_info - A structure of revision informationparameters.quoted
+ * + * Returns the FM revision + * + * Allowed only following fman_init(). + * + * Return: 0 on success; Error code otherwise. + */ +void fman_get_revision(struct fman *fman, struct fman_rev_info +*rev_info); + +/** + * fman_register_intr + * @fman: A Pointer to FMan device + * @mod: Calling module + * @mod_id: Module id (if more than 1 exists, '0' if not) + * @intr_type: Interrupt type (error/normal) selection. + * @f_isr: The interrupt service routine. + * @h_src_arg: Argument to be passed to f_isr. + * + * Used to register an event handler to be processed by FMan + * + * Return: 0 on success; Error code otherwise. + */ +void fman_register_intr(struct fman *fman, enum fman_event_modulesmod,quoted
+ u8 mod_id, enum fman_intr_type intr_type, + void (*f_isr)(void *h_src_arg), void *h_src_arg); + +/** + * fman_unregister_intr + * @fman: A Pointer to FMan device + * @mod: Calling module + * @mod_id: Module id (if more than 1 exists, '0' if not) + * @intr_type: Interrupt type (error/normal) selection. + * + * Used to unregister an event handler to be processed by FMan + * + * Return: 0 on success; Error code otherwise. + */ +void fman_unregister_intr(struct fman *fman, enumfman_event_modules mod,quoted
+ u8 mod_id, enum fman_intr_type intr_type); + +/** + * fman_set_port_params + * @fman: A Pointer to FMan device + * @port_params: Port parameters + * + * Used by FMan Port to pass parameters to the FMan + * + * Return: 0 on success; Error code otherwise. + */ +int fman_set_port_params(struct fman *fman, + struct fman_port_init_params *port_params); + +/** + * fman_reset_mac + * @fman: A Pointer to FMan device + * @mac_id: MAC id to be reset + * + * Reset a specific MAC + * + * Return: 0 on success; Error code otherwise. + */ +int fman_reset_mac(struct fman *fman, u8 mac_id); + +/** + * fman_get_clock_freq + * @fman: A Pointer to FMan device + * + * Get FMan clock frequency + * + * Return: FMan clock frequency + */ + +u16 fman_get_clock_freq(struct fman *fman); + +/** + * fman_get_bmi_max_fifo_size + * @fman: A Pointer to FMan device + * + * Get FMan maximum FIFO size + * + * Return: FMan Maximum FIFO size + */ +u32 fman_get_bmi_max_fifo_size(struct fman *fman); + +/** + * fman_set_mac_max_frame + * @fman: A Pointer to FMan device + * @mac_id: MAC id + * @mfl: Maximum frame length + * + * Set maximum frame length of specific MAC in FMan driver + * + * Return: 0 on success; Error code otherwise. + */ +int fman_set_mac_max_frame(struct fman *fman, u8 mac_id, u16 mfl); + +/** + * fman_get_qman_channel_id + * @fman: A Pointer to FMan device + * @port_id: Port id + * + * Get QMan channel ID associated to the Port id + * + * Return: QMan channel ID + */ +u32 fman_get_qman_channel_id(struct fman *fman, u32 port_id); + +/** + * fman_get_mem_region + * @fman: A Pointer to FMan device + * + * Get FMan memory region + * + * Return: A structure with FMan memory region information */ struct +resource *fman_get_mem_region(struct fman *fman); + +/** + * fman_get_max_frm + * + * Return: Max frame length configured in the FM driver */ +u16 fman_get_max_frm(void); + +/** + * fman_get_rx_extra_headroom + * + * Return: Extra headroom size configured in the FM driver */ int +fman_get_rx_extra_headroom(void); + +/** + * fman_bind + * @dev: FMan OF device pointer + * + * Bind to a specific FMan device. + * + * Allowed only after the port was created. + * + * Return: A pointer to the FMan device */ struct fman +*fman_bind(struct device *dev); + +/** + * fman_unbind + * @fman: Pointer to the FMan device + * + * Un-bind from a specific FMan device. + * + * Allowed only after the port was created. + */ +void fman_unbind(struct fman *fman); + +/** + * fman_get_device + * @fman: A pointer to the FMan device. + * + * Get the FMan device pointer + * + * Return: Pointer to FMan device. + */ +struct device *fman_get_device(struct fman *fman);Usually kerneldoc comments go on the implementation, not the prototype.
OK, moved kernel doc to the source files, here and elsewhere.
-Scott
_______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev