summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/media/platform/tegra/camera/nvcsi/csi5_fops.c99
1 files changed, 98 insertions, 1 deletions
diff --git a/drivers/media/platform/tegra/camera/nvcsi/csi5_fops.c b/drivers/media/platform/tegra/camera/nvcsi/csi5_fops.c
index 02c3c295e..727ef3006 100644
--- a/drivers/media/platform/tegra/camera/nvcsi/csi5_fops.c
+++ b/drivers/media/platform/tegra/camera/nvcsi/csi5_fops.c
@@ -14,6 +14,20 @@
14#include "nvhost_acm.h" 14#include "nvhost_acm.h"
15#include "nvcsi/nvcsi.h" 15#include "nvcsi/nvcsi.h"
16#include "csi5_fops.h" 16#include "csi5_fops.h"
17#include <linux/tegra-capture-ivc.h>
18#include "soc/tegra/camrtc-capture-messages.h"
19
20/* Referred from capture-scheduler.c defined in rtcpu-fw */
21#define NUM_CAPTURE_CHANNELS 64
22
23/* Temporary ids for the clients whose channel-id is not yet allocated */
24#define NUM_CAPTURE_TRANSACTION_IDS 64
25
26#define TOTAL_CHANNELS (NUM_CAPTURE_CHANNELS + NUM_CAPTURE_TRANSACTION_IDS)
27
28#define TEMP_CHANNEL_ID (NUM_CAPTURE_CHANNELS + 1)
29#define TPG_HBLANK 0
30#define TPG_VBLANK 40800
17 31
18static int csi5_power_on(struct tegra_csi_device *csi) 32static int csi5_power_on(struct tegra_csi_device *csi)
19{ 33{
@@ -41,18 +55,101 @@ static int csi5_start_streaming(struct tegra_csi_channel *chan,
41 enum tegra_csi_port_num port_num) 55 enum tegra_csi_port_num port_num)
42{ 56{
43 struct tegra_csi_device *csi = chan->csi; 57 struct tegra_csi_device *csi = chan->csi;
58 struct tegra_csi_port *port = &chan->ports[port_num];
59 int err = 0;
60
61 /* PG native resolution */
62 const size_t px_max = 0x4000;
63 const size_t py_max = 0x2000;
64 size_t hfreq = 0;
65 size_t vfreq = 0;
66
67 union nvcsi_tpg_config *pgc = NULL;
68 struct CAPTURE_CONTROL_MSG msg;
69 const size_t messageSize = sizeof(msg);
44 70
71 memset(&msg, 0, messageSize);
45 dev_info(csi->dev, "csi5_start_streaming\n"); 72 dev_info(csi->dev, "csi5_start_streaming\n");
73 if (chan->pg_mode) {
74 pgc = &msg.tpg_setup_req.tpg_config;
75
76 hfreq = px_max / port->format.width;
77 vfreq = py_max / port->format.height;
78
79 pgc->t194.virtual_channel = 0;
80 /* hardcode CSI_DT_RAW_16 */
81 pgc->t194.data_type = 46;
82 pgc->t194.lane_count = chan->numlanes;
83 pgc->t194.flags = NVCSI_TPG_FLAG_PATCH_MODE;
84
85 pgc->t194.initial_frame_number = 1;
86 pgc->t194.maximum_frame_number = 32768;
87 pgc->t194.image_width = port->format.width;
88 pgc->t194.image_height = port->format.height;
89
90 pgc->t194.red_horizontal_init_freq = hfreq;
91 pgc->t194.red_vertical_init_freq = vfreq;
92
93 pgc->t194.green_horizontal_init_freq = hfreq;
94 pgc->t194.green_vertical_init_freq = vfreq;
95
96 pgc->t194.blue_horizontal_init_freq = hfreq;
97 pgc->t194.blue_vertical_init_freq = vfreq;
98
99 msg.header.msg_id = CAPTURE_CHANNEL_TPG_SETUP_REQ;
100 /* use a free channel to send control */
101 msg.header.channel_id = TEMP_CHANNEL_ID;
102
103 err = tegra_capture_ivc_control_submit(&msg, messageSize);
104 if (err < 0) {
105 dev_err(csi->dev, "IVC control submit failed\n");
106 return err;
107 }
108
109 /* start streaming */
110 memset(&msg, 0, messageSize);
111 msg.header.msg_id = CAPTURE_CHANNEL_TPG_START_REQ;
112 msg.header.channel_id = TEMP_CHANNEL_ID;
113 msg.tpg_start_req.stream = chan->port[0];
114 msg.tpg_start_req.channel = 0;
115 msg.tpg_start_req.tpg_rate_config.hblank = TPG_HBLANK;
116 msg.tpg_start_req.tpg_rate_config.vblank = TPG_VBLANK;
117 msg.tpg_start_req.tpg_rate_config.pixel_interval = 0;
118 err = tegra_capture_ivc_control_submit(&msg, messageSize);
119 if (err < 0) {
120 dev_err(csi->dev, "IVC control submit failed\n");
121 return err;
122 }
123 }
46 124
47 return 0; 125 return err;
48} 126}
49 127
50static void csi5_stop_streaming(struct tegra_csi_channel *chan, 128static void csi5_stop_streaming(struct tegra_csi_channel *chan,
51 enum tegra_csi_port_num port_num) 129 enum tegra_csi_port_num port_num)
52{ 130{
53 struct tegra_csi_device *csi = chan->csi; 131 struct tegra_csi_device *csi = chan->csi;
132 int err = 0;
133
134 struct CAPTURE_CONTROL_MSG msg;
135 const size_t messageSize = sizeof(msg);
54 136
137 memset(&msg, 0, messageSize);
55 dev_info(csi->dev, "csi5_stop_streaming\n"); 138 dev_info(csi->dev, "csi5_stop_streaming\n");
139
140 if (chan->pg_mode) {
141 msg.header.msg_id = CAPTURE_CHANNEL_TPG_STOP_REQ;
142 msg.header.channel_id = TEMP_CHANNEL_ID;
143
144 msg.tpg_stop_req.stream = chan->port[0];
145 msg.tpg_stop_req.channel = 0;
146
147 err = tegra_capture_ivc_control_submit(&msg, messageSize);
148 if (err < 0) {
149 dev_err(csi->dev, "IVC control submit failed\n");
150 return;
151 }
152 }
56} 153}
57 154
58static int csi5_mipi_cal(struct tegra_csi_channel *chan) 155static int csi5_mipi_cal(struct tegra_csi_channel *chan)