// SPDX-License-Identifier: GPL-2.0-only /* * Faraday Technology FTIDE010 driver * Copyright (C) 2017 Linus Walleij <[email protected]> * * Includes portions of the SL2312/SL3516/Gemini PATA driver * Copyright (C) 2003 StorLine, Inc <[email protected]> * Copyright (C) 2009 Janos Laube <[email protected]> * Copyright (C) 2010 Frederic Pecourt <[email protected]> * Copyright (C) 2011 Tobias Waldvogel <[email protected]> */ #include <linux/platform_device.h> #include <linux/module.h> #include <linux/libata.h> #include <linux/bitops.h> #include <linux/of.h> #include <linux/clk.h> #include "sata_gemini.h" #define DRV_NAME … /** * struct ftide010 - state container for the Faraday FTIDE010 * @dev: pointer back to the device representing this controller * @base: remapped I/O space address * @pclk: peripheral clock for the IDE block * @host: pointer to the ATA host for this device * @master_cbl: master cable type * @slave_cbl: slave cable type * @sg: Gemini SATA bridge pointer, if running on the Gemini * @master_to_sata0: Gemini SATA bridge: the ATA master is connected * to the SATA0 bridge * @slave_to_sata0: Gemini SATA bridge: the ATA slave is connected * to the SATA0 bridge * @master_to_sata1: Gemini SATA bridge: the ATA master is connected * to the SATA1 bridge * @slave_to_sata1: Gemini SATA bridge: the ATA slave is connected * to the SATA1 bridge */ struct ftide010 { … }; #define FTIDE010_DMA_REG … #define FTIDE010_DMA_STATUS … #define FTIDE010_IDE_BMDTPR … #define FTIDE010_IDE_DEVICE_ID … #define FTIDE010_PIO_TIMING … #define FTIDE010_MWDMA_TIMING … #define FTIDE010_UDMA_TIMING0 … #define FTIDE010_UDMA_TIMING1 … #define FTIDE010_CLK_MOD … /* These registers are mapped directly to the IDE registers */ #define FTIDE010_CMD_DATA … #define FTIDE010_ERROR_FEATURES … #define FTIDE010_NSECT … #define FTIDE010_LBAL … #define FTIDE010_LBAM … #define FTIDE010_LBAH … #define FTIDE010_DEVICE … #define FTIDE010_STATUS_COMMAND … #define FTIDE010_ALTSTAT_CTRL … /* Set this bit for UDMA mode 5 and 6 */ #define FTIDE010_UDMA_TIMING_MODE_56 … /* 0 = 50 MHz, 1 = 66 MHz */ #define FTIDE010_CLK_MOD_DEV0_CLK_SEL … #define FTIDE010_CLK_MOD_DEV1_CLK_SEL … /* Enable UDMA on a device */ #define FTIDE010_CLK_MOD_DEV0_UDMA_EN … #define FTIDE010_CLK_MOD_DEV1_UDMA_EN … static const struct scsi_host_template pata_ftide010_sht = …; /* * Bus timings * * The unit of the below required timings is two clock periods of the ATA * reference clock which is 30 nanoseconds per unit at 66MHz and 20 * nanoseconds per unit at 50 MHz. The PIO timings assume 33MHz speed for * PIO. * * pio_active_time: array of 5 elements for T2 timing for Mode 0, * 1, 2, 3 and 4. Range 0..15. * pio_recovery_time: array of 5 elements for T2l timing for Mode 0, * 1, 2, 3 and 4. Range 0..15. * mdma_50_active_time: array of 4 elements for Td timing for multi * word DMA, Mode 0, 1, and 2 at 50 MHz. Range 0..15. * mdma_50_recovery_time: array of 4 elements for Tk timing for * multi word DMA, Mode 0, 1 and 2 at 50 MHz. Range 0..15. * mdma_66_active_time: array of 4 elements for Td timing for multi * word DMA, Mode 0, 1 and 2 at 66 MHz. Range 0..15. * mdma_66_recovery_time: array of 4 elements for Tk timing for * multi word DMA, Mode 0, 1 and 2 at 66 MHz. Range 0..15. * udma_50_setup_time: array of 4 elements for Tvds timing for ultra * DMA, Mode 0, 1, 2, 3, 4 and 5 at 50 MHz. Range 0..7. * udma_50_hold_time: array of 4 elements for Tdvh timing for * multi word DMA, Mode 0, 1, 2, 3, 4 and 5 at 50 MHz, Range 0..7. * udma_66_setup_time: array of 4 elements for Tvds timing for multi * word DMA, Mode 0, 1, 2, 3, 4, 5 and 6 at 66 MHz. Range 0..7. * udma_66_hold_time: array of 4 elements for Tdvh timing for * multi word DMA, Mode 0, 1, 2, 3, 4, 5 and 6 at 66 MHz. Range 0..7. */ static const u8 pio_active_time[5] = …; static const u8 pio_recovery_time[5] = …; static const u8 mwdma_50_active_time[3] = …; static const u8 mwdma_50_recovery_time[3] = …; static const u8 mwdma_66_active_time[3] = …; static const u8 mwdma_66_recovery_time[3] = …; static const u8 udma_50_setup_time[6] = …; static const u8 udma_50_hold_time[6] = …; static const u8 udma_66_setup_time[7] = …; static const u8 udma_66_hold_time[7] = …; /* * We set 66 MHz for all MWDMA modes */ static const bool set_mdma_66_mhz[] = …; /* * We set 66 MHz for UDMA modes 3, 4 and 6 and no others */ static const bool set_udma_66_mhz[] = …; static void ftide010_set_dmamode(struct ata_port *ap, struct ata_device *adev) { … } static void ftide010_set_piomode(struct ata_port *ap, struct ata_device *adev) { … } /* * We implement our own qc_issue() callback since we may need to set up * the timings differently for master and slave transfers: the CLK_MOD_REG * and MWDMA_TIMING_REG is shared between master and slave, so reprogramming * this may be necessary. */ static unsigned int ftide010_qc_issue(struct ata_queued_cmd *qc) { … } static struct ata_port_operations pata_ftide010_port_ops = …; static struct ata_port_info ftide010_port_info = …; #if IS_ENABLED(CONFIG_SATA_GEMINI) static int pata_ftide010_gemini_port_start(struct ata_port *ap) { … } static void pata_ftide010_gemini_port_stop(struct ata_port *ap) { … } static int pata_ftide010_gemini_cable_detect(struct ata_port *ap) { … } static int pata_ftide010_gemini_init(struct ftide010 *ftide, struct ata_port_info *pi, bool is_ata1) { … } #else static int pata_ftide010_gemini_init(struct ftide010 *ftide, struct ata_port_info *pi, bool is_ata1) { return -ENOTSUPP; } #endif static int pata_ftide010_probe(struct platform_device *pdev) { … } static void pata_ftide010_remove(struct platform_device *pdev) { … } static const struct of_device_id pata_ftide010_of_match[] = …; MODULE_DEVICE_TABLE(of, pata_ftide010_of_match); static struct platform_driver pata_ftide010_driver = …; module_platform_driver(…) …; MODULE_DESCRIPTION(…) …; MODULE_AUTHOR(…) …; MODULE_LICENSE(…) …; MODULE_ALIAS(…) …;