On 27.02.19 08:18, Morris Ku wrote:

Hi,


first of all: the code is pretty unreadable. I doubt that anybody here
seriously likes to review that (nor take it into mainline).

Formatting should be consistent - see other drivers.

./scripts/checkpatch.pl is your friend. You really shut run it and fix
everything it complaints before posting patches. (some maintainers can
get angry if you don't ;-)

> Support SUNIX serial board.
> 
> ---
>  char/snx/snx_serial.c | 4771 +++++++++++++++++++++++++++++++++++++++++
>  1 file changed, 4771 insertions(+)
>  create mode 100644 char/snx/snx_serial.c
> 
> diff --git a/char/snx/snx_serial.c b/char/snx/snx_serial.c
> new file mode 100644
> index 00000000..94caac1a
> --- /dev/null
> +++ b/char/snx/snx_serial.c
> @@ -0,0 +1,4771 @@
> +#include "snx_common.h"
> +#include "driver_extd.h"
> +
> +#define SNX_ioctl_DBG        0
> +#define      EEPROM_ACCESS_DELAY_COUNT                       100000

is this eeprom stuff really specific to the serial ports ?
if not, it probably should go to a separate file, which is called by all
the others, IMHO.

> +
> +#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 37))
> +             static DEFINE_SEMAPHORE(ser_port_sem);
> +#else
> +             static DECLARE_MUTEX(ser_port_sem);
> +#endif

Drop that. We have only one kernel version here, the current one.

> +
> +
> +#define SNX_HIGH_BITS_OFFSET ((sizeof(long)-sizeof(int))*8)
> +#define sunix_ser_users(state)       ((state)->count + ((state)->info ? 
> (state)->info->blocked_open : 0))
> +
> +
> +#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 7, 0))
> +static struct tty_port snx_service_port;
> +#endif

are you really sure this has to be a global field, instead of allocated
per-device ?

> +
> +
> +#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 7, 0))
> +
> +struct serial_uart_config {
> +     char    *name;
> +     int             dfl_xmit_fifo_size;
> +     int             flags;
> +};
> +#endif
> +
> +static const struct serial_uart_config snx_uart_config[PORT_SER_MAX_UART + 
> 1] = {

why +1 ?

maybe PORT_SER_MAX_UART and use ARRAY_SIZE() macro instead.

> +     { "unknown",    1,      0 },
> +     { "8250",       1,      0 },
> +     { "16450",      1,      0 },
> +     { "16550",      1,      0 },
> +     { "16550A",     16,     UART_CLEAR_FIFO | UART_USE_FIFO },
> +     { "Cirrus",     1,      0 },
> +     { "ST16650",    1,      0 },
> +     { "ST16650V2",  32,     UART_CLEAR_FIFO | UART_USE_FIFO },
> +     { "TI16750",    64,     UART_CLEAR_FIFO | UART_USE_FIFO },
> +};
>
> +
> +
> +#if (LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 0))
> +static int           sunix_ser_refcount;
> +static struct tty_struct                     
> *sunix_ser_tty[SNX_SER_TOTAL_MAX + 1];
> +static struct termios                                
> *sunix_ser_termios[SNX_SER_TOTAL_MAX + 1];
> +static struct termios                                
> *sunix_ser_termios_locked[SNX_SER_TOTAL_MAX + 1];
> +#endif

obsolete. drop that.

> +
> +
> +static _INLINE_ void snx_ser_handle_cts_change(struct snx_ser_port *, 
> unsigned int);

what exactly does _INLINE_ suppose to mean ?

<snip>

> +extern int           sunix_ser_interrupt(struct sunix_board *, struct 
> sunix_ser_port *);

why "extern" here ? where is that function coming from ?

<snip>

> +static unsigned char READ_INTERRUPT_VECTOR_BYTE(struct sunix_ser_port *);

I doubt that upper case function names fit in the kernel coding style.

<snip>

> +             case SNX_SER_DUMP_PORT_INFO:
> +             {
> +                     int i;
> +                     struct snx_ser_port_info 
> snx_port_info[SNX_SER_TOTAL_MAX];
> +                     struct snx_ser_port *sdn = NULL;
> +
> +                     memset(snx_port_info, 0, (sizeof(struct 
> snx_ser_port_info) * SNX_SER_TOTAL_MAX));
> +
> +                     if (line == 0) {
> +                             for (i = 0; i < SNX_SER_TOTAL_MAX; i++) {
> +                                     sdn = (struct snx_ser_port *) 
> &sunix_ser_table[i];
> +
> +                                     
> memcpy(&snx_port_info[i].board_name_info[0], &sdn->pb_info.board_name[0], 
> SNX_BOARDNAME_LENGTH);
> +
> +                                     snx_port_info[i].bus_number_info = 
> sdn->bus_number;
> +                                     snx_port_info[i].dev_number_info = 
> sdn->dev_number;
> +                                     snx_port_info[i].port_info       = 
> sdn->line;
> +                                     snx_port_info[i].base_info       = 
> sdn->iobase;
> +                                     snx_port_info[i].irq_info        = 
> sdn->irq;
> +                             }
> +
> +                             if (copy_to_user((void *)arg, snx_port_info, 
> SNX_SER_TOTAL_MAX * sizeof(struct snx_ser_port_info))) {
> +                                     ret = -EFAULT;
> +                             } else {
> +                                     ret = 0;
> +                             }
> +                     } else {
> +                             ret = 0;
> +             }
> +
> +                     ret = 0;
> +                     break;
> +             }
> +
> +             case SNX_SER_DUMP_DRIVER_VER:
> +             {
> +                     char driver_ver[SNX_DRIVERVERSION_LENGTH];
> +                     memset(driver_ver, 0, (sizeof(char) * 
> SNX_DRIVERVERSION_LENGTH));
> +
> +                     if (line == 0) {
> +
> +                             memcpy(&driver_ver[0], SNX_DRIVER_VERSION, 
> sizeof(SNX_DRIVER_VERSION));
> +
> +                             if (copy_to_user((void *)arg, &driver_ver, 
> (sizeof(char) * SNX_DRIVERVERSION_LENGTH))) {
> +                                     ret = -EFAULT;
> +                             } else {
> +                                     ret = 0;
> +                             }
> +                     } else {
> +                             ret = 0;
> +                     }
> +
> +                     break;
> +             }
> +
> +
> +             case SNX_COMM_GET_BOARD_CNT:
> +             {
> +                     SNX_DRVR_BOARD_CNT gb;
> +
> +                     memset(&gb, 0, (sizeof(SNX_DRVR_BOARD_CNT)));
> +
> +                     gb.cnt = snx_board_count;
> +
> +                     if (copy_to_user((void *)arg, &gb, 
> (sizeof(SNX_DRVR_BOARD_CNT)))) {
> +                             ret = -EFAULT;
> +                     } else {
> +                     ret = 0;
> +                     }
> +
> +                     break;
> +             }
> +
> +             case SNX_COMM_GET_BOARD_INFO:
> +             {
> +                     struct sunix_board *sb = NULL;
> +                     SNX_DRVR_BOARD_INFO             board_info;
> +
> +                     memset(&board_info, 0, (sizeof(SNX_DRVR_BOARD_INFO)));
> +
> +                     if (copy_from_user(&board_info, (void *)arg, 
> (sizeof(SNX_DRVR_BOARD_INFO)))) {
> +                             ret = -EFAULT;
> +                     } else {
> +                             ret = 0;
> +                     }
> +
> +                     sb = &sunix_board_table[board_info.board_id - 1];
> +
> +                     board_info.subvender_id = sb->pb_info.sub_vendor_id;
> +                     board_info.subsystem_id = sb->pb_info.sub_device_id;
> +                     board_info.oem_id = sb->oem_id;
> +                     board_info.uart_cnt = sb->uart_cnt;
> +                     board_info.gpio_chl_cnt = sb->gpio_chl_cnt;
> +                     board_info.board_uart_type = sb->board_uart_type;
> +                     board_info.board_gpio_type = sb->board_gpio_type;
> +
> +                     if (copy_to_user((void *)arg, &board_info, 
> (sizeof(SNX_DRVR_BOARD_INFO)))) {
> +                             ret = -EFAULT;
> +                     } else {
> +                             ret = 0;
> +                     }
> +                     break;
> +             }
> +

what exactly is that for ?

> +             case SNX_GPIO_GET:

gpio stuff doesn't belong into a serial / tty.
for that we have the gpio subsystem. (see drivers/gpio/*)

> +             case SNX_UART_GET_TYPE:
> +             {
> +                     struct sunix_board      *sb = NULL;
> +                     SNX_DRVR_UART_GET_TYPE gb;
> +
> +                     int bar3_base_Address;
> +                     int bar1_base_Address;
> +
> +                     int bar3_byte5;
> +                     int uart_type;
> +                     int     RS422state;
> +                     int     AHDCstate;
> +
> +                     memset(&gb, 0, (sizeof(SNX_DRVR_UART_GET_TYPE)));
> +
> +                     if (copy_from_user(&gb, (void *)arg, 
> (sizeof(SNX_DRVR_UART_GET_TYPE)))) {
> +                             ret = -EFAULT;
> +                     } else {
> +                             ret = 0;
> +                     }
> +
> +                     sb = &sunix_board_table[gb.board_id - 1];
> +
> +                     bar3_base_Address = pci_resource_start(sb->pdev, 3);
> +                     bar1_base_Address = pci_resource_start(sb->pdev, 1);
> +
> +                     bar3_byte5 = inb(bar3_base_Address + 5);
> +                     uart_type = (bar3_byte5 & 0xC0) >> 6;
> +
> +                     if (gb.uart_num <= 4) {
> +                             AHDCstate = inb(bar3_base_Address + 2) & 0x0F & 
> (0x01 << ((gb.uart_num - 1) % 4));
> +                             RS422state = inb(bar3_base_Address + 3) & 0xF0 
> & (0x10 << ((gb.uart_num - 1) % 4));
> +                     } else if (gb.uart_num <= 8) {
> +                             AHDCstate = inb(bar1_base_Address + 0x32) & 
> 0x0F & (0x01 << ((gb.uart_num - 1) % 4));
> +                             RS422state = inb(bar1_base_Address + 0x33) & 
> 0xF0 & (0x10 << ((gb.uart_num - 1) % 4));
> +                     } else if (gb.uart_num <= 12) {
> +                             AHDCstate = inb(bar1_base_Address + 0x32 + 
> 0x40) & 0x0F & (0x01 << ((gb.uart_num - 1) % 4));
> +                             RS422state = inb(bar1_base_Address + 0x33 + 
> 0x40) & 0xF0 & (0x10 << ((gb.uart_num - 1) % 4));
> +                     } else if (gb.uart_num <= 16) {
> +                             AHDCstate = inb(bar1_base_Address + 0x32 + 
> 0x80) & 0x0F & (0x01 << ((gb.uart_num - 1) % 4));
> +                             RS422state = inb(bar1_base_Address + 0x33 + 
> 0x80) & 0xF0 & (0x10 << ((gb.uart_num - 1) % 4));
> +                     } else {
> +                             //cmn_err(CE_NOTE, "WARNING : we get an 
> incorrect port number (port = %d)!",gb.uart_num);
> +                             break;
> +                     }
> +
> +                             switch (uart_type) {
> +                             case 0: // RS-232
> +                             {
> +                                     gb.uart_type = 0;
> +                                     break;
> +                             }
> +                             case 1: // RS-422/485
> +                             {
> +                                     if (AHDCstate && RS422state) {
> +                                             gb.uart_type = 3;
> +                                     } else if (AHDCstate && !RS422state) {
> +                                             gb.uart_type = 2;
> +                                     } else if (!AHDCstate && RS422state) {
> +                                             gb.uart_type = 1;
> +                                     } else {
> +                                             gb.uart_type = -1;
> +                                     }
> +                                     break;
> +                             }
> +                             case 2:
> +                             {
> +                                     if (AHDCstate && RS422state) {
> +                                             gb.uart_type = 3;
> +                                     } else if (AHDCstate && !RS422state) {
> +                                             gb.uart_type = 2;
> +                                     } else if (!AHDCstate && RS422state) {
> +                                             gb.uart_type = 1;
> +                                     } else if (!AHDCstate && !RS422state) {
> +                                             gb.uart_type = 0;
> +                                     } else {
> +                                             gb.uart_type = -1;
> +                                     }
> +                                     break;
> +                             }
> +                     }
> +
> +                     if (copy_to_user((void *)arg, &gb, 
> (sizeof(SNX_DRVR_UART_GET_TYPE)))) {
> +                             ret = -EFAULT;
> +                     } else {
> +                     ret = 0;
> +                     }
> +

what is that for ?

> +             case SNX_UART_GET_ACS:
> +             {

what is an "ACS" ?

In general: don't arbitrarily introduce new ioctl()s, especially not
device specific ones - unless there is a damn good reason.


--mtx
-- 
Enrico Weigelt, metux IT consult
Free software and Linux embedded engineering
[email protected] -- +49-151-27565287

Reply via email to