Blame view

kernel/linux-rt-4.4.41/drivers/media/pci/solo6x10/solo6x10-gpio.c 2.64 KB
5113f6f70   김현기   kernel add
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
  /*
   * Copyright (C) 2010-2013 Bluecherry, LLC <http://www.bluecherrydvr.com>
   *
   * Original author:
   * Ben Collins <bcollins@ubuntu.com>
   *
   * Additional work by:
   * John Brooks <john.brooks@bluecherry.net>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License as published by
   * the Free Software Foundation; either version 2 of the License, or
   * (at your option) any later version.
   *
   * This program is distributed in the hope that it will be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   * GNU General Public License for more details.
   */
  
  #include <linux/kernel.h>
  #include <linux/fs.h>
  #include <linux/delay.h>
  #include <linux/uaccess.h>
  
  #include "solo6x10.h"
  
  static void solo_gpio_mode(struct solo_dev *solo_dev,
  			   unsigned int port_mask, unsigned int mode)
  {
  	int port;
  	unsigned int ret;
  
  	ret = solo_reg_read(solo_dev, SOLO_GPIO_CONFIG_0);
  
  	/* To set gpio */
  	for (port = 0; port < 16; port++) {
  		if (!((1 << port) & port_mask))
  			continue;
  
  		ret &= (~(3 << (port << 1)));
  		ret |= ((mode & 3) << (port << 1));
  	}
  
  	solo_reg_write(solo_dev, SOLO_GPIO_CONFIG_0, ret);
  
  	/* To set extended gpio - sensor */
  	ret = solo_reg_read(solo_dev, SOLO_GPIO_CONFIG_1);
  
  	for (port = 0; port < 16; port++) {
  		if (!((1 << (port + 16)) & port_mask))
  			continue;
  
  		if (!mode)
  			ret &= ~(1 << port);
  		else
  			ret |= 1 << port;
  	}
  
  	solo_reg_write(solo_dev, SOLO_GPIO_CONFIG_1, ret);
  }
  
  static void solo_gpio_set(struct solo_dev *solo_dev, unsigned int value)
  {
  	solo_reg_write(solo_dev, SOLO_GPIO_DATA_OUT,
  		       solo_reg_read(solo_dev, SOLO_GPIO_DATA_OUT) | value);
  }
  
  static void solo_gpio_clear(struct solo_dev *solo_dev, unsigned int value)
  {
  	solo_reg_write(solo_dev, SOLO_GPIO_DATA_OUT,
  		       solo_reg_read(solo_dev, SOLO_GPIO_DATA_OUT) & ~value);
  }
  
  static void solo_gpio_config(struct solo_dev *solo_dev)
  {
  	/* Video reset */
  	solo_gpio_mode(solo_dev, 0x30, 1);
  	solo_gpio_clear(solo_dev, 0x30);
  	udelay(100);
  	solo_gpio_set(solo_dev, 0x30);
  	udelay(100);
  
  	/* Warning: Don't touch the next line unless you're sure of what
  	 * you're doing: first four gpio [0-3] are used for video. */
  	solo_gpio_mode(solo_dev, 0x0f, 2);
  
  	/* We use bit 8-15 of SOLO_GPIO_CONFIG_0 for relay purposes */
  	solo_gpio_mode(solo_dev, 0xff00, 1);
  
  	/* Initially set relay status to 0 */
  	solo_gpio_clear(solo_dev, 0xff00);
  }
  
  int solo_gpio_init(struct solo_dev *solo_dev)
  {
  	solo_gpio_config(solo_dev);
  	return 0;
  }
  
  void solo_gpio_exit(struct solo_dev *solo_dev)
  {
  	solo_gpio_clear(solo_dev, 0x30);
  	solo_gpio_config(solo_dev);
  }