lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Date:   Thu, 7 Mar 2019 04:19:43 +0100
From:   "Enrico Weigelt, metux IT consult" <lkml@...ux.net>
To:     Morris Ku <saumah@...il.com>, gregkh@...uxfoundation.org
Cc:     morris_ku@...ix.com, linux-kernel@...r.kernel.org
Subject: Re: [PATCH 4/5] Add driver for SUNIX Multi-I/O board

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
info@...ux.net -- +49-151-27565287

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ