summaryrefslogtreecommitdiff
path: root/drivers/video/wmt
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/video/wmt')
-rw-r--r--drivers/video/wmt/Makefile25
-rw-r--r--drivers/video/wmt/bootanimation/LzmaDec.c1023
-rw-r--r--drivers/video/wmt/bootanimation/LzmaDec.h239
-rw-r--r--drivers/video/wmt/bootanimation/Makefile19
-rw-r--r--drivers/video/wmt/bootanimation/Types.h253
-rw-r--r--drivers/video/wmt/bootanimation/anim_data.h930
-rw-r--r--drivers/video/wmt/bootanimation/animation.c459
-rw-r--r--drivers/video/wmt/bootanimation/animation.h34
-rw-r--r--drivers/video/wmt/bootanimation/buffer.c102
-rw-r--r--drivers/video/wmt/bootanimation/buffer.h53
-rw-r--r--drivers/video/wmt/cec.c323
-rw-r--r--drivers/video/wmt/cec.h65
-rw-r--r--drivers/video/wmt/com-cec.h63
-rw-r--r--drivers/video/wmt/com-mb.h78
-rw-r--r--drivers/video/wmt/com-vpp.h680
-rw-r--r--drivers/video/wmt/dev-cec.c488
-rw-r--r--drivers/video/wmt/devices/Makefile20
-rw-r--r--drivers/video/wmt/devices/cs8556.c617
-rw-r--r--drivers/video/wmt/devices/lcd-AUO-A080SN01.c93
-rw-r--r--drivers/video/wmt/devices/lcd-CHILIN-LW700at9003.c102
-rw-r--r--drivers/video/wmt/devices/lcd-EKING-EK08009-70135.c101
-rw-r--r--drivers/video/wmt/devices/lcd-HANNSTAR-HSD101PFW2.c101
-rw-r--r--drivers/video/wmt/devices/lcd-INNOLUX-AT070TN83.c122
-rw-r--r--drivers/video/wmt/devices/lcd-b079xan01.c465
-rwxr-xr-xdrivers/video/wmt/devices/lcd-gl5001w.c1412
-rw-r--r--drivers/video/wmt/devices/lcd-lvds-1024x600.c92
-rw-r--r--drivers/video/wmt/devices/lcd-oem.c323
-rwxr-xr-xdrivers/video/wmt/devices/lcd-setup.c311
-rwxr-xr-xdrivers/video/wmt/devices/lcd-spi.c786
-rw-r--r--drivers/video/wmt/devices/lcd.c422
-rwxr-xr-xdrivers/video/wmt/devices/ts8224b.h446
-rw-r--r--drivers/video/wmt/devices/vt1625.c454
-rw-r--r--drivers/video/wmt/devices/vt1632.c157
-rw-r--r--drivers/video/wmt/edid.h155
-rw-r--r--drivers/video/wmt/ge/Makefile5
-rw-r--r--drivers/video/wmt/ge/ge_accel.c551
-rw-r--r--drivers/video/wmt/ge/ge_accel.h79
-rw-r--r--drivers/video/wmt/ge/ge_main.c1003
-rw-r--r--drivers/video/wmt/govrh.c1926
-rw-r--r--drivers/video/wmt/govrh.h168
-rw-r--r--drivers/video/wmt/hdmi.c1614
-rw-r--r--drivers/video/wmt/hdmi.h345
-rw-r--r--drivers/video/wmt/hdmi_cp.c45
-rw-r--r--drivers/video/wmt/hdmi_cp.h63
-rw-r--r--drivers/video/wmt/hw/wmt-cec-reg.h169
-rw-r--r--drivers/video/wmt/hw/wmt-govrh-reg.h362
-rw-r--r--drivers/video/wmt/hw/wmt-hdmi-reg.h266
-rw-r--r--drivers/video/wmt/hw/wmt-lvds-reg.h92
-rw-r--r--drivers/video/wmt/hw/wmt-scl-reg.h374
-rw-r--r--drivers/video/wmt/hw/wmt-vpp-hw.h135
-rw-r--r--drivers/video/wmt/hw/wmt-vpp-reg.h98
-rw-r--r--drivers/video/wmt/lcd.h98
-rw-r--r--drivers/video/wmt/lvds.c204
-rw-r--r--drivers/video/wmt/lvds.h73
-rw-r--r--drivers/video/wmt/mali.c759
-rw-r--r--drivers/video/wmt/mali.h49
-rw-r--r--drivers/video/wmt/parse-edid.c1027
-rw-r--r--drivers/video/wmt/scl.c1604
-rw-r--r--drivers/video/wmt/scl.h126
-rw-r--r--drivers/video/wmt/sw_i2c.c426
-rw-r--r--drivers/video/wmt/sw_i2c.h61
-rw-r--r--drivers/video/wmt/vout-wmt.c1523
-rw-r--r--drivers/video/wmt/vout.c936
-rw-r--r--drivers/video/wmt/vout.h283
-rw-r--r--drivers/video/wmt/vpp-osif.c891
-rw-r--r--drivers/video/wmt/vpp-osif.h247
-rw-r--r--drivers/video/wmt/vpp.c1674
-rw-r--r--drivers/video/wmt/vpp.h687
-rw-r--r--drivers/video/wmt/vppm.c283
-rw-r--r--drivers/video/wmt/vppm.h62
-rw-r--r--drivers/video/wmt/wmt-mb.c3357
-rw-r--r--drivers/video/wmt/wmt-sync.c346
-rwxr-xr-xdrivers/video/wmt/wmt-vpp.c1538
-rw-r--r--drivers/video/wmt/wmtfb.c2764
74 files changed, 37326 insertions, 0 deletions
diff --git a/drivers/video/wmt/Makefile b/drivers/video/wmt/Makefile
new file mode 100644
index 00000000..9603160c
--- /dev/null
+++ b/drivers/video/wmt/Makefile
@@ -0,0 +1,25 @@
+#
+# Makefile for the Wonder Media framebuffer driver
+#
+
+# wmt external video device
+obj-$(CONFIG_FB_WMT) += devices/
+obj-$(CONFIG_WMT_EDID) += parse-edid.o sw_i2c.o
+
+# wmt vout for platform
+obj-$(CONFIG_FB_WMT) += vout-wmt.o
+
+# wmt vpp
+obj-$(CONFIG_FB_WMT) += vout.o vpp.o govrh.o scl.o vppm.o lvds.o
+obj-$(CONFIG_WMT_HDMI) += hdmi.o hdmi_cp.o
+obj-$(CONFIG_FB_WMT) += vpp-osif.o wmt-sync.o
+obj-$(CONFIG_WMT_CEC) += cec.o dev-cec.o
+
+obj-$(CONFIG_FB_WMT) += mali.o
+obj-$(CONFIG_FB_WMT_GE) += ge/
+
+# wmt fb
+obj-$(CONFIG_FB_WMT) += wmtfb.o
+
+# wmt mb
+obj-$(CONFIG_WMT_MB) += wmt-mb.o
diff --git a/drivers/video/wmt/bootanimation/LzmaDec.c b/drivers/video/wmt/bootanimation/LzmaDec.c
new file mode 100644
index 00000000..b8be701d
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/LzmaDec.c
@@ -0,0 +1,1023 @@
+/*++
+ linux/drivers/video/wmt/LzmaDec.c
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#include "LzmaDec.h"
+
+#include <linux/string.h>
+
+#define kNumTopBits 24
+#define kTopValue ((UInt32)1 << kNumTopBits)
+
+#define kNumBitModelTotalBits 11
+#define kBitModelTotal (1 << kNumBitModelTotalBits)
+#define kNumMoveBits 5
+
+#define RC_INIT_SIZE 5
+
+#define NORMALIZE if (range < kTopValue) { range <<= 8; code = (code << 8) | (*buf++); }
+
+#define IF_BIT_0(p) ttt = *(p); NORMALIZE; bound = (range >> kNumBitModelTotalBits) * ttt; if (code < bound)
+#define UPDATE_0(p) range = bound; *(p) = (CLzmaProb)(ttt + ((kBitModelTotal - ttt) >> kNumMoveBits));
+#define UPDATE_1(p) range -= bound; code -= bound; *(p) = (CLzmaProb)(ttt - (ttt >> kNumMoveBits));
+#define GET_BIT2(p, i, A0, A1) IF_BIT_0(p) \
+ { UPDATE_0(p); i = (i + i); A0; } else \
+ { UPDATE_1(p); i = (i + i) + 1; A1; }
+#define GET_BIT(p, i) GET_BIT2(p, i, ; , ;)
+
+#define TREE_GET_BIT(probs, i) { GET_BIT((probs + i), i); }
+#define TREE_DECODE(probs, limit, i) \
+ { i = 1; do { TREE_GET_BIT(probs, i); } while (i < limit); i -= limit; }
+
+/* #define _LZMA_SIZE_OPT */
+
+#ifdef _LZMA_SIZE_OPT
+#define TREE_6_DECODE(probs, i) TREE_DECODE(probs, (1 << 6), i)
+#else
+#define TREE_6_DECODE(probs, i) \
+ { i = 1; \
+ TREE_GET_BIT(probs, i); \
+ TREE_GET_BIT(probs, i); \
+ TREE_GET_BIT(probs, i); \
+ TREE_GET_BIT(probs, i); \
+ TREE_GET_BIT(probs, i); \
+ TREE_GET_BIT(probs, i); \
+ i -= 0x40; }
+#endif
+
+#define NORMALIZE_CHECK if (range < kTopValue) { if (buf >= bufLimit) return DUMMY_ERROR; range <<= 8; code = (code << 8) | (*buf++); }
+
+#define IF_BIT_0_CHECK(p) ttt = *(p); NORMALIZE_CHECK; bound = (range >> kNumBitModelTotalBits) * ttt; if (code < bound)
+#define UPDATE_0_CHECK range = bound;
+#define UPDATE_1_CHECK range -= bound; code -= bound;
+#define GET_BIT2_CHECK(p, i, A0, A1) IF_BIT_0_CHECK(p) \
+ { UPDATE_0_CHECK; i = (i + i); A0; } else \
+ { UPDATE_1_CHECK; i = (i + i) + 1; A1; }
+#define GET_BIT_CHECK(p, i) GET_BIT2_CHECK(p, i, ; , ;)
+#define TREE_DECODE_CHECK(probs, limit, i) \
+ { i = 1; do { GET_BIT_CHECK(probs + i, i) } while (i < limit); i -= limit; }
+
+
+#define kNumPosBitsMax 4
+#define kNumPosStatesMax (1 << kNumPosBitsMax)
+
+#define kLenNumLowBits 3
+#define kLenNumLowSymbols (1 << kLenNumLowBits)
+#define kLenNumMidBits 3
+#define kLenNumMidSymbols (1 << kLenNumMidBits)
+#define kLenNumHighBits 8
+#define kLenNumHighSymbols (1 << kLenNumHighBits)
+
+#define LenChoice 0
+#define LenChoice2 (LenChoice + 1)
+#define LenLow (LenChoice2 + 1)
+#define LenMid (LenLow + (kNumPosStatesMax << kLenNumLowBits))
+#define LenHigh (LenMid + (kNumPosStatesMax << kLenNumMidBits))
+#define kNumLenProbs (LenHigh + kLenNumHighSymbols)
+
+
+#define kNumStates 12
+#define kNumLitStates 7
+
+#define kStartPosModelIndex 4
+#define kEndPosModelIndex 14
+#define kNumFullDistances (1 << (kEndPosModelIndex >> 1))
+
+#define kNumPosSlotBits 6
+#define kNumLenToPosStates 4
+
+#define kNumAlignBits 4
+#define kAlignTableSize (1 << kNumAlignBits)
+
+#define kMatchMinLen 2
+#define kMatchSpecLenStart (kMatchMinLen + kLenNumLowSymbols + kLenNumMidSymbols + kLenNumHighSymbols)
+
+#define IsMatch 0
+#define IsRep (IsMatch + (kNumStates << kNumPosBitsMax))
+#define IsRepG0 (IsRep + kNumStates)
+#define IsRepG1 (IsRepG0 + kNumStates)
+#define IsRepG2 (IsRepG1 + kNumStates)
+#define IsRep0Long (IsRepG2 + kNumStates)
+#define PosSlot (IsRep0Long + (kNumStates << kNumPosBitsMax))
+#define SpecPos (PosSlot + (kNumLenToPosStates << kNumPosSlotBits))
+#define Align (SpecPos + kNumFullDistances - kEndPosModelIndex)
+#define LenCoder (Align + kAlignTableSize)
+#define RepLenCoder (LenCoder + kNumLenProbs)
+#define Literal (RepLenCoder + kNumLenProbs)
+
+#define LZMA_BASE_SIZE 1846
+#define LZMA_LIT_SIZE 768
+
+#define LzmaProps_GetNumProbs(p) ((UInt32)LZMA_BASE_SIZE + (LZMA_LIT_SIZE << ((p)->lc + (p)->lp)))
+
+#if Literal != LZMA_BASE_SIZE
+StopCompilingDueBUG
+#endif
+
+static const Byte kLiteralNextStates[kNumStates * 2] =
+{
+ 0, 0, 0, 0, 1, 2, 3, 4, 5, 6, 4, 5,
+ 7, 7, 7, 7, 7, 7, 7, 10, 10, 10, 10, 10
+};
+
+#define LZMA_DIC_MIN (1 << 12)
+
+/* First LZMA-symbol is always decoded.
+And it decodes new LZMA-symbols while (buf < bufLimit), but "buf" is without last normalization
+Out:
+ Result:
+ SZ_OK - OK
+ SZ_ERROR_DATA - Error
+ p->remainLen:
+ < kMatchSpecLenStart : normal remain
+ = kMatchSpecLenStart : finished
+ = kMatchSpecLenStart + 1 : Flush marker
+ = kMatchSpecLenStart + 2 : State Init Marker
+*/
+
+static int MY_FAST_CALL LzmaDec_DecodeReal(CLzmaDec *p, SizeT limit, const Byte *bufLimit)
+{
+ CLzmaProb *probs = p->probs;
+
+ unsigned state = p->state;
+ UInt32 rep0 = p->reps[0], rep1 = p->reps[1], rep2 = p->reps[2], rep3 = p->reps[3];
+ unsigned pbMask = ((unsigned)1 << (p->prop.pb)) - 1;
+ unsigned lpMask = ((unsigned)1 << (p->prop.lp)) - 1;
+ unsigned lc = p->prop.lc;
+
+ Byte *dic = p->dic;
+ SizeT dicBufSize = p->dicBufSize;
+ SizeT dicPos = p->dicPos;
+
+ UInt32 processedPos = p->processedPos;
+ UInt32 checkDicSize = p->checkDicSize;
+ unsigned len = 0;
+
+ const Byte *buf = p->buf;
+ UInt32 range = p->range;
+ UInt32 code = p->code;
+
+ do
+ {
+ CLzmaProb *prob;
+ UInt32 bound;
+ unsigned ttt;
+ unsigned posState = processedPos & pbMask;
+
+ prob = probs + IsMatch + (state << kNumPosBitsMax) + posState;
+ IF_BIT_0(prob)
+ {
+ unsigned symbol;
+ UPDATE_0(prob);
+ prob = probs + Literal;
+ if (checkDicSize != 0 || processedPos != 0)
+ prob += (LZMA_LIT_SIZE * (((processedPos & lpMask) << lc) +
+ (dic[(dicPos == 0 ? dicBufSize : dicPos) - 1] >> (8 - lc))));
+
+ if (state < kNumLitStates)
+ {
+ symbol = 1;
+ do { GET_BIT(prob + symbol, symbol) } while (symbol < 0x100);
+ }
+ else
+ {
+ unsigned matchByte = p->dic[(dicPos - rep0) + ((dicPos < rep0) ? dicBufSize : 0)];
+ unsigned offs = 0x100;
+ symbol = 1;
+ do
+ {
+ unsigned bit;
+ CLzmaProb *probLit;
+ matchByte <<= 1;
+ bit = (matchByte & offs);
+ probLit = prob + offs + bit + symbol;
+ GET_BIT2(probLit, symbol, offs &= ~bit, offs &= bit)
+ }
+ while (symbol < 0x100);
+ }
+ dic[dicPos++] = (Byte)symbol;
+ processedPos++;
+
+ state = kLiteralNextStates[state];
+ /* if (state < 4) state = 0; else if (state < 10) state -= 3; else state -= 6; */
+ continue;
+ }
+ else
+ {
+ UPDATE_1(prob);
+ prob = probs + IsRep + state;
+ IF_BIT_0(prob)
+ {
+ UPDATE_0(prob);
+ state += kNumStates;
+ prob = probs + LenCoder;
+ }
+ else
+ {
+ UPDATE_1(prob);
+ if (checkDicSize == 0 && processedPos == 0)
+ return SZ_ERROR_DATA;
+ prob = probs + IsRepG0 + state;
+ IF_BIT_0(prob)
+ {
+ UPDATE_0(prob);
+ prob = probs + IsRep0Long + (state << kNumPosBitsMax) + posState;
+ IF_BIT_0(prob)
+ {
+ UPDATE_0(prob);
+ dic[dicPos] = dic[(dicPos - rep0) + ((dicPos < rep0) ? dicBufSize : 0)];
+ dicPos++;
+ processedPos++;
+ state = state < kNumLitStates ? 9 : 11;
+ continue;
+ }
+ UPDATE_1(prob);
+ }
+ else
+ {
+ UInt32 distance;
+ UPDATE_1(prob);
+ prob = probs + IsRepG1 + state;
+ IF_BIT_0(prob)
+ {
+ UPDATE_0(prob);
+ distance = rep1;
+ }
+ else
+ {
+ UPDATE_1(prob);
+ prob = probs + IsRepG2 + state;
+ IF_BIT_0(prob)
+ {
+ UPDATE_0(prob);
+ distance = rep2;
+ }
+ else
+ {
+ UPDATE_1(prob);
+ distance = rep3;
+ rep3 = rep2;
+ }
+ rep2 = rep1;
+ }
+ rep1 = rep0;
+ rep0 = distance;
+ }
+ state = state < kNumLitStates ? 8 : 11;
+ prob = probs + RepLenCoder;
+ }
+ {
+ unsigned limit, offset;
+ CLzmaProb *probLen = prob + LenChoice;
+ IF_BIT_0(probLen)
+ {
+ UPDATE_0(probLen);
+ probLen = prob + LenLow + (posState << kLenNumLowBits);
+ offset = 0;
+ limit = (1 << kLenNumLowBits);
+ }
+ else
+ {
+ UPDATE_1(probLen);
+ probLen = prob + LenChoice2;
+ IF_BIT_0(probLen)
+ {
+ UPDATE_0(probLen);
+ probLen = prob + LenMid + (posState << kLenNumMidBits);
+ offset = kLenNumLowSymbols;
+ limit = (1 << kLenNumMidBits);
+ }
+ else
+ {
+ UPDATE_1(probLen);
+ probLen = prob + LenHigh;
+ offset = kLenNumLowSymbols + kLenNumMidSymbols;
+ limit = (1 << kLenNumHighBits);
+ }
+ }
+ TREE_DECODE(probLen, limit, len);
+ len += offset;
+ }
+
+ if (state >= kNumStates)
+ {
+ UInt32 distance;
+ prob = probs + PosSlot +
+ ((len < kNumLenToPosStates ? len : kNumLenToPosStates - 1) << kNumPosSlotBits);
+ TREE_6_DECODE(prob, distance);
+ if (distance >= kStartPosModelIndex)
+ {
+ unsigned posSlot = (unsigned)distance;
+ int numDirectBits = (int)(((distance >> 1) - 1));
+ distance = (2 | (distance & 1));
+ if (posSlot < kEndPosModelIndex)
+ {
+ distance <<= numDirectBits;
+ prob = probs + SpecPos + distance - posSlot - 1;
+ {
+ UInt32 mask = 1;
+ unsigned i = 1;
+ do
+ {
+ GET_BIT2(prob + i, i, ; , distance |= mask);
+ mask <<= 1;
+ }
+ while (--numDirectBits != 0);
+ }
+ }
+ else
+ {
+ numDirectBits -= kNumAlignBits;
+ do
+ {
+ NORMALIZE
+ range >>= 1;
+
+ {
+ UInt32 t;
+ code -= range;
+ t = (0 - ((UInt32)code >> 31)); /* (UInt32)((Int32)code >> 31) */
+ distance = (distance << 1) + (t + 1);
+ code += range & t;
+ }
+ /*
+ distance <<= 1;
+ if (code >= range)
+ {
+ code -= range;
+ distance |= 1;
+ }
+ */
+ }
+ while (--numDirectBits != 0);
+ prob = probs + Align;
+ distance <<= kNumAlignBits;
+ {
+ unsigned i = 1;
+ GET_BIT2(prob + i, i, ; , distance |= 1);
+ GET_BIT2(prob + i, i, ; , distance |= 2);
+ GET_BIT2(prob + i, i, ; , distance |= 4);
+ GET_BIT2(prob + i, i, ; , distance |= 8);
+ }
+ if (distance == (UInt32)0xFFFFFFFF)
+ {
+ len += kMatchSpecLenStart;
+ state -= kNumStates;
+ break;
+ }
+ }
+ }
+ rep3 = rep2;
+ rep2 = rep1;
+ rep1 = rep0;
+ rep0 = distance + 1;
+ if (checkDicSize == 0)
+ {
+ if (distance >= processedPos)
+ return SZ_ERROR_DATA;
+ }
+ else if (distance >= checkDicSize)
+ return SZ_ERROR_DATA;
+ state = (state < kNumStates + kNumLitStates) ? kNumLitStates : kNumLitStates + 3;
+ /* state = kLiteralNextStates[state]; */
+ }
+
+ len += kMatchMinLen;
+
+ if (limit == dicPos)
+ return SZ_ERROR_DATA;
+ {
+ SizeT rem = limit - dicPos;
+ unsigned curLen = ((rem < len) ? (unsigned)rem : len);
+ SizeT pos = (dicPos - rep0) + ((dicPos < rep0) ? dicBufSize : 0);
+
+ processedPos += curLen;
+
+ len -= curLen;
+ if (pos + curLen <= dicBufSize)
+ {
+ Byte *dest = dic + dicPos;
+ ptrdiff_t src = (ptrdiff_t)pos - (ptrdiff_t)dicPos;
+ const Byte *lim = dest + curLen;
+ dicPos += curLen;
+ do
+ *(dest) = (Byte)*(dest + src);
+ while (++dest != lim);
+ }
+ else
+ {
+ do
+ {
+ dic[dicPos++] = dic[pos];
+ if (++pos == dicBufSize)
+ pos = 0;
+ }
+ while (--curLen != 0);
+ }
+ }
+ }
+ }
+ while (dicPos < limit && buf < bufLimit);
+ NORMALIZE;
+ p->buf = buf;
+ p->range = range;
+ p->code = code;
+ p->remainLen = len;
+ p->dicPos = dicPos;
+ p->processedPos = processedPos;
+ p->reps[0] = rep0;
+ p->reps[1] = rep1;
+ p->reps[2] = rep2;
+ p->reps[3] = rep3;
+ p->state = state;
+
+ return SZ_OK;
+}
+
+static void MY_FAST_CALL LzmaDec_WriteRem(CLzmaDec *p, SizeT limit)
+{
+ if (p->remainLen != 0 && p->remainLen < kMatchSpecLenStart)
+ {
+ Byte *dic = p->dic;
+ SizeT dicPos = p->dicPos;
+ SizeT dicBufSize = p->dicBufSize;
+ unsigned len = p->remainLen;
+ UInt32 rep0 = p->reps[0];
+ if (limit - dicPos < len)
+ len = (unsigned)(limit - dicPos);
+
+ if (p->checkDicSize == 0 && p->prop.dicSize - p->processedPos <= len)
+ p->checkDicSize = p->prop.dicSize;
+
+ p->processedPos += len;
+ p->remainLen -= len;
+ while (len-- != 0)
+ {
+ dic[dicPos] = dic[(dicPos - rep0) + ((dicPos < rep0) ? dicBufSize : 0)];
+ dicPos++;
+ }
+ p->dicPos = dicPos;
+ }
+}
+
+static int MY_FAST_CALL LzmaDec_DecodeReal2(CLzmaDec *p, SizeT limit, const Byte *bufLimit)
+{
+ do
+ {
+ SizeT limit2 = limit;
+ if (p->checkDicSize == 0)
+ {
+ UInt32 rem = p->prop.dicSize - p->processedPos;
+ if (limit - p->dicPos > rem)
+ limit2 = p->dicPos + rem;
+ }
+ RINOK(LzmaDec_DecodeReal(p, limit2, bufLimit));
+ if (p->processedPos >= p->prop.dicSize)
+ p->checkDicSize = p->prop.dicSize;
+ LzmaDec_WriteRem(p, limit);
+ }
+ while (p->dicPos < limit && p->buf < bufLimit && p->remainLen < kMatchSpecLenStart);
+
+ if (p->remainLen > kMatchSpecLenStart)
+ {
+ p->remainLen = kMatchSpecLenStart;
+ }
+ return 0;
+}
+
+typedef enum
+{
+ DUMMY_ERROR, /* unexpected end of input stream */
+ DUMMY_LIT,
+ DUMMY_MATCH,
+ DUMMY_REP
+} ELzmaDummy;
+
+static ELzmaDummy LzmaDec_TryDummy(const CLzmaDec *p, const Byte *buf, SizeT inSize)
+{
+ UInt32 range = p->range;
+ UInt32 code = p->code;
+ const Byte *bufLimit = buf + inSize;
+ CLzmaProb *probs = p->probs;
+ unsigned state = p->state;
+ ELzmaDummy res;
+
+ {
+ CLzmaProb *prob;
+ UInt32 bound;
+ unsigned ttt;
+ unsigned posState = (p->processedPos) & ((1 << p->prop.pb) - 1);
+
+ prob = probs + IsMatch + (state << kNumPosBitsMax) + posState;
+ IF_BIT_0_CHECK(prob)
+ {
+ UPDATE_0_CHECK
+
+ /* if (bufLimit - buf >= 7) return DUMMY_LIT; */
+
+ prob = probs + Literal;
+ if (p->checkDicSize != 0 || p->processedPos != 0)
+ prob += (LZMA_LIT_SIZE *
+ ((((p->processedPos) & ((1 << (p->prop.lp)) - 1)) << p->prop.lc) +
+ (p->dic[(p->dicPos == 0 ? p->dicBufSize : p->dicPos) - 1] >> (8 - p->prop.lc))));
+
+ if (state < kNumLitStates)
+ {
+ unsigned symbol = 1;
+ do { GET_BIT_CHECK(prob + symbol, symbol) } while (symbol < 0x100);
+ }
+ else
+ {
+ unsigned matchByte = p->dic[p->dicPos - p->reps[0] +
+ ((p->dicPos < p->reps[0]) ? p->dicBufSize : 0)];
+ unsigned offs = 0x100;
+ unsigned symbol = 1;
+ do
+ {
+ unsigned bit;
+ CLzmaProb *probLit;
+ matchByte <<= 1;
+ bit = (matchByte & offs);
+ probLit = prob + offs + bit + symbol;
+ GET_BIT2_CHECK(probLit, symbol, offs &= ~bit, offs &= bit)
+ }
+ while (symbol < 0x100);
+ }
+ res = DUMMY_LIT;
+ }
+ else
+ {
+ unsigned len;
+ UPDATE_1_CHECK;
+
+ prob = probs + IsRep + state;
+ IF_BIT_0_CHECK(prob)
+ {
+ UPDATE_0_CHECK;
+ state = 0;
+ prob = probs + LenCoder;
+ res = DUMMY_MATCH;
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ res = DUMMY_REP;
+ prob = probs + IsRepG0 + state;
+ IF_BIT_0_CHECK(prob)
+ {
+ UPDATE_0_CHECK;
+ prob = probs + IsRep0Long + (state << kNumPosBitsMax) + posState;
+ IF_BIT_0_CHECK(prob)
+ {
+ UPDATE_0_CHECK;
+ NORMALIZE_CHECK;
+ return DUMMY_REP;
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ }
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ prob = probs + IsRepG1 + state;
+ IF_BIT_0_CHECK(prob)
+ {
+ UPDATE_0_CHECK;
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ prob = probs + IsRepG2 + state;
+ IF_BIT_0_CHECK(prob)
+ {
+ UPDATE_0_CHECK;
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ }
+ }
+ }
+ state = kNumStates;
+ prob = probs + RepLenCoder;
+ }
+ {
+ unsigned limit, offset;
+ CLzmaProb *probLen = prob + LenChoice;
+ IF_BIT_0_CHECK(probLen)
+ {
+ UPDATE_0_CHECK;
+ probLen = prob + LenLow + (posState << kLenNumLowBits);
+ offset = 0;
+ limit = 1 << kLenNumLowBits;
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ probLen = prob + LenChoice2;
+ IF_BIT_0_CHECK(probLen)
+ {
+ UPDATE_0_CHECK;
+ probLen = prob + LenMid + (posState << kLenNumMidBits);
+ offset = kLenNumLowSymbols;
+ limit = 1 << kLenNumMidBits;
+ }
+ else
+ {
+ UPDATE_1_CHECK;
+ probLen = prob + LenHigh;
+ offset = kLenNumLowSymbols + kLenNumMidSymbols;
+ limit = 1 << kLenNumHighBits;
+ }
+ }
+ TREE_DECODE_CHECK(probLen, limit, len);
+ len += offset;
+ }
+
+ if (state < 4)
+ {
+ unsigned posSlot;
+ prob = probs + PosSlot +
+ ((len < kNumLenToPosStates ? len : kNumLenToPosStates - 1) <<
+ kNumPosSlotBits);
+ TREE_DECODE_CHECK(prob, 1 << kNumPosSlotBits, posSlot);
+ if (posSlot >= kStartPosModelIndex)
+ {
+ int numDirectBits = ((posSlot >> 1) - 1);
+
+ /* if (bufLimit - buf >= 8) return DUMMY_MATCH; */
+
+ if (posSlot < kEndPosModelIndex)
+ {
+ prob = probs + SpecPos + ((2 | (posSlot & 1)) << numDirectBits) - posSlot - 1;
+ }
+ else
+ {
+ numDirectBits -= kNumAlignBits;
+ do
+ {
+ NORMALIZE_CHECK
+ range >>= 1;
+ code -= range & (((code - range) >> 31) - 1);
+ /* if (code >= range) code -= range; */
+ }
+ while (--numDirectBits != 0);
+ prob = probs + Align;
+ numDirectBits = kNumAlignBits;
+ }
+ {
+ unsigned i = 1;
+ do
+ {
+ GET_BIT_CHECK(prob + i, i);
+ }
+ while (--numDirectBits != 0);
+ }
+ }
+ }
+ }
+ }
+ NORMALIZE_CHECK;
+ return res;
+}
+
+
+static void LzmaDec_InitRc(CLzmaDec *p, const Byte *data)
+{
+ p->code = ((UInt32)data[1] << 24) | ((UInt32)data[2] << 16) | ((UInt32)data[3] << 8) | ((UInt32)data[4]);
+ p->range = 0xFFFFFFFF;
+ p->needFlush = 0;
+}
+
+void LzmaDec_InitDicAndState(CLzmaDec *p, Bool initDic, Bool initState)
+{
+ p->needFlush = 1;
+ p->remainLen = 0;
+ p->tempBufSize = 0;
+
+ if (initDic)
+ {
+ p->processedPos = 0;
+ p->checkDicSize = 0;
+ p->needInitState = 1;
+ }
+ if (initState)
+ p->needInitState = 1;
+}
+
+void LzmaDec_Init(CLzmaDec *p)
+{
+ p->dicPos = 0;
+ LzmaDec_InitDicAndState(p, True, True);
+}
+
+static void LzmaDec_InitStateReal(CLzmaDec *p)
+{
+ UInt32 numProbs = Literal + ((UInt32)LZMA_LIT_SIZE << (p->prop.lc + p->prop.lp));
+ UInt32 i;
+ CLzmaProb *probs = p->probs;
+ for (i = 0; i < numProbs; i++)
+ probs[i] = kBitModelTotal >> 1;
+ p->reps[0] = p->reps[1] = p->reps[2] = p->reps[3] = 1;
+ p->state = 0;
+ p->needInitState = 0;
+}
+
+SRes LzmaDec_DecodeToDic(CLzmaDec *p, SizeT dicLimit, const Byte *src, SizeT *srcLen,
+ ELzmaFinishMode finishMode, ELzmaStatus *status)
+{
+ SizeT inSize = *srcLen;
+ (*srcLen) = 0;
+ LzmaDec_WriteRem(p, dicLimit);
+
+ *status = LZMA_STATUS_NOT_SPECIFIED;
+
+ while (p->remainLen != kMatchSpecLenStart)
+ {
+ int checkEndMarkNow;
+
+ if (p->needFlush != 0)
+ {
+ for (; inSize > 0 && p->tempBufSize < RC_INIT_SIZE; (*srcLen)++, inSize--)
+ p->tempBuf[p->tempBufSize++] = *src++;
+ if (p->tempBufSize < RC_INIT_SIZE)
+ {
+ *status = LZMA_STATUS_NEEDS_MORE_INPUT;
+ return SZ_OK;
+ }
+ if (p->tempBuf[0] != 0)
+ return SZ_ERROR_DATA;
+
+ LzmaDec_InitRc(p, p->tempBuf);
+ p->tempBufSize = 0;
+ }
+
+ checkEndMarkNow = 0;
+ if (p->dicPos >= dicLimit)
+ {
+ if (p->remainLen == 0 && p->code == 0)
+ {
+ *status = LZMA_STATUS_MAYBE_FINISHED_WITHOUT_MARK;
+ return SZ_OK;
+ }
+ if (finishMode == LZMA_FINISH_ANY)
+ {
+ *status = LZMA_STATUS_NOT_FINISHED;
+ return SZ_OK;
+ }
+ if (p->remainLen != 0)
+ {
+ *status = LZMA_STATUS_NOT_FINISHED;
+ return SZ_ERROR_DATA;
+ }
+ checkEndMarkNow = 1;
+ }
+
+ if (p->needInitState)
+ LzmaDec_InitStateReal(p);
+
+ if (p->tempBufSize == 0)
+ {
+ SizeT processed;
+ const Byte *bufLimit;
+ if (inSize < LZMA_REQUIRED_INPUT_MAX || checkEndMarkNow)
+ {
+ int dummyRes = LzmaDec_TryDummy(p, src, inSize);
+ if (dummyRes == DUMMY_ERROR)
+ {
+ memcpy(p->tempBuf, src, inSize);
+ p->tempBufSize = (unsigned)inSize;
+ (*srcLen) += inSize;
+ *status = LZMA_STATUS_NEEDS_MORE_INPUT;
+ return SZ_OK;
+ }
+ if (checkEndMarkNow && dummyRes != DUMMY_MATCH)
+ {
+ *status = LZMA_STATUS_NOT_FINISHED;
+ return SZ_ERROR_DATA;
+ }
+ bufLimit = src;
+ }
+ else
+ bufLimit = src + inSize - LZMA_REQUIRED_INPUT_MAX;
+ p->buf = src;
+ if (LzmaDec_DecodeReal2(p, dicLimit, bufLimit) != 0)
+ return SZ_ERROR_DATA;
+ processed = (SizeT)(p->buf - src);
+ (*srcLen) += processed;
+ src += processed;
+ inSize -= processed;
+ }
+ else
+ {
+ unsigned rem = p->tempBufSize, lookAhead = 0;
+ while (rem < LZMA_REQUIRED_INPUT_MAX && lookAhead < inSize)
+ p->tempBuf[rem++] = src[lookAhead++];
+ p->tempBufSize = rem;
+ if (rem < LZMA_REQUIRED_INPUT_MAX || checkEndMarkNow)
+ {
+ int dummyRes = LzmaDec_TryDummy(p, p->tempBuf, rem);
+ if (dummyRes == DUMMY_ERROR)
+ {
+ (*srcLen) += lookAhead;
+ *status = LZMA_STATUS_NEEDS_MORE_INPUT;
+ return SZ_OK;
+ }
+ if (checkEndMarkNow && dummyRes != DUMMY_MATCH)
+ {
+ *status = LZMA_STATUS_NOT_FINISHED;
+ return SZ_ERROR_DATA;
+ }
+ }
+ p->buf = p->tempBuf;
+ if (LzmaDec_DecodeReal2(p, dicLimit, p->buf) != 0)
+ return SZ_ERROR_DATA;
+ lookAhead -= (rem - (unsigned)(p->buf - p->tempBuf));
+ (*srcLen) += lookAhead;
+ src += lookAhead;
+ inSize -= lookAhead;
+ p->tempBufSize = 0;
+ }
+ }
+ if (p->code == 0)
+ *status = LZMA_STATUS_FINISHED_WITH_MARK;
+ return (p->code == 0) ? SZ_OK : SZ_ERROR_DATA;
+}
+
+SRes LzmaDec_DecodeToBuf(CLzmaDec *p, Byte *dest, SizeT *destLen, const Byte *src, SizeT *srcLen, ELzmaFinishMode finishMode, ELzmaStatus *status)
+{
+ SizeT outSize = *destLen;
+ SizeT inSize = *srcLen;
+ *srcLen = *destLen = 0;
+ for (;;)
+ {
+ SizeT inSizeCur = inSize, outSizeCur, dicPos;
+ ELzmaFinishMode curFinishMode;
+ SRes res;
+ if (p->dicPos == p->dicBufSize)
+ p->dicPos = 0;
+ dicPos = p->dicPos;
+ if (outSize > p->dicBufSize - dicPos)
+ {
+ outSizeCur = p->dicBufSize;
+ curFinishMode = LZMA_FINISH_ANY;
+ }
+ else
+ {
+ outSizeCur = dicPos + outSize;
+ curFinishMode = finishMode;
+ }
+
+ res = LzmaDec_DecodeToDic(p, outSizeCur, src, &inSizeCur, curFinishMode, status);
+ src += inSizeCur;
+ inSize -= inSizeCur;
+ *srcLen += inSizeCur;
+ outSizeCur = p->dicPos - dicPos;
+ memcpy(dest, p->dic + dicPos, outSizeCur);
+ dest += outSizeCur;
+ outSize -= outSizeCur;
+ *destLen += outSizeCur;
+ if (res != 0)
+ return res;
+ if (outSizeCur == 0 || outSize == 0)
+ return SZ_OK;
+ }
+}
+
+void LzmaDec_FreeProbs(CLzmaDec *p, ISzAlloc *alloc)
+{
+ alloc->Free(alloc, p->probs);
+ p->probs = 0;
+}
+
+static void LzmaDec_FreeDict(CLzmaDec *p, ISzAlloc *alloc)
+{
+ alloc->Free(alloc, p->dic);
+ p->dic = 0;
+}
+
+void LzmaDec_Free(CLzmaDec *p, ISzAlloc *alloc)
+{
+ LzmaDec_FreeProbs(p, alloc);
+ LzmaDec_FreeDict(p, alloc);
+}
+
+SRes LzmaProps_Decode(CLzmaProps *p, const Byte *data, unsigned size)
+{
+ UInt32 dicSize;
+ Byte d;
+
+ if (size < LZMA_PROPS_SIZE)
+ return SZ_ERROR_UNSUPPORTED;
+ else
+ dicSize = data[1] | ((UInt32)data[2] << 8) | ((UInt32)data[3] << 16) | ((UInt32)data[4] << 24);
+
+ if (dicSize < LZMA_DIC_MIN)
+ dicSize = LZMA_DIC_MIN;
+ p->dicSize = dicSize;
+
+ d = data[0];
+ if (d >= (9 * 5 * 5))
+ return SZ_ERROR_UNSUPPORTED;
+
+ p->lc = d % 9;
+ d /= 9;
+ p->pb = d / 5;
+ p->lp = d % 5;
+
+ return SZ_OK;
+}
+
+static SRes LzmaDec_AllocateProbs2(CLzmaDec *p, const CLzmaProps *propNew, ISzAlloc *alloc)
+{
+ UInt32 numProbs = LzmaProps_GetNumProbs(propNew);
+ if (p->probs == 0 || numProbs != p->numProbs)
+ {
+ LzmaDec_FreeProbs(p, alloc);
+ p->probs = (CLzmaProb *)alloc->Alloc(alloc, numProbs * sizeof(CLzmaProb));
+ p->numProbs = numProbs;
+ if (p->probs == 0)
+ return SZ_ERROR_MEM;
+ }
+ return SZ_OK;
+}
+
+SRes LzmaDec_AllocateProbs(CLzmaDec *p, const Byte *props, unsigned propsSize, ISzAlloc *alloc)
+{
+ CLzmaProps propNew;
+ RINOK(LzmaProps_Decode(&propNew, props, propsSize));
+ RINOK(LzmaDec_AllocateProbs2(p, &propNew, alloc));
+ p->prop = propNew;
+ return SZ_OK;
+}
+
+SRes LzmaDec_Allocate(CLzmaDec *p, const Byte *props, unsigned propsSize, ISzAlloc *alloc)
+{
+ CLzmaProps propNew;
+ SizeT dicBufSize;
+ RINOK(LzmaProps_Decode(&propNew, props, propsSize));
+ RINOK(LzmaDec_AllocateProbs2(p, &propNew, alloc));
+ dicBufSize = propNew.dicSize;
+ if (p->dic == 0 || dicBufSize != p->dicBufSize)
+ {
+ LzmaDec_FreeDict(p, alloc);
+ p->dic = (Byte *)alloc->Alloc(alloc, dicBufSize);
+ if (p->dic == 0)
+ {
+ LzmaDec_FreeProbs(p, alloc);
+ return SZ_ERROR_MEM;
+ }
+ }
+ p->dicBufSize = dicBufSize;
+ p->prop = propNew;
+ return SZ_OK;
+}
+
+SRes LzmaDecode(Byte *dest, SizeT *destLen, const Byte *src, SizeT *srcLen,
+ const Byte *propData, unsigned propSize, ELzmaFinishMode finishMode,
+ ELzmaStatus *status, ISzAlloc *alloc)
+{
+ CLzmaDec p;
+ SRes res;
+ SizeT inSize = *srcLen;
+ SizeT outSize = *destLen;
+ *srcLen = *destLen = 0;
+ if (inSize < RC_INIT_SIZE)
+ return SZ_ERROR_INPUT_EOF;
+
+ LzmaDec_Construct(&p);
+ res = LzmaDec_AllocateProbs(&p, propData, propSize, alloc);
+ if (res != 0)
+ return res;
+ p.dic = dest;
+ p.dicBufSize = outSize;
+
+ LzmaDec_Init(&p);
+
+ *srcLen = inSize;
+ res = LzmaDec_DecodeToDic(&p, outSize, src, srcLen, finishMode, status);
+
+ if (res == SZ_OK && *status == LZMA_STATUS_NEEDS_MORE_INPUT)
+ res = SZ_ERROR_INPUT_EOF;
+
+ (*destLen) = p.dicPos;
+ LzmaDec_FreeProbs(&p, alloc);
+ return res;
+}
diff --git a/drivers/video/wmt/bootanimation/LzmaDec.h b/drivers/video/wmt/bootanimation/LzmaDec.h
new file mode 100644
index 00000000..232ec65e
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/LzmaDec.h
@@ -0,0 +1,239 @@
+/*++
+ linux/drivers/video/wmt/LzmaDec.h
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#ifndef __LZMADEC_H
+#define __LZMADEC_H
+
+#include "Types.h"
+
+/* #define _LZMA_PROB32 */
+/* _LZMA_PROB32 can increase the speed on some CPUs,
+ but memory usage for CLzmaDec::probs will be doubled in that case */
+
+#ifdef _LZMA_PROB32
+#define CLzmaProb UInt32
+#else
+#define CLzmaProb UInt16
+#endif
+
+
+/* ---------- LZMA Properties ---------- */
+
+#define LZMA_PROPS_SIZE 5
+
+typedef struct _CLzmaProps
+{
+ unsigned lc, lp, pb;
+ UInt32 dicSize;
+} CLzmaProps;
+
+/* LzmaProps_Decode - decodes properties
+Returns:
+ SZ_OK
+ SZ_ERROR_UNSUPPORTED - Unsupported properties
+*/
+
+SRes LzmaProps_Decode(CLzmaProps *p, const Byte *data, unsigned size);
+
+
+/* ---------- LZMA Decoder state ---------- */
+
+/* LZMA_REQUIRED_INPUT_MAX = number of required input bytes for worst case.
+ Num bits = log2((2^11 / 31) ^ 22) + 26 < 134 + 26 = 160; */
+
+#define LZMA_REQUIRED_INPUT_MAX 20
+
+typedef struct
+{
+ CLzmaProps prop;
+ CLzmaProb *probs;
+ Byte *dic;
+ const Byte *buf;
+ UInt32 range, code;
+ SizeT dicPos;
+ SizeT dicBufSize;
+ UInt32 processedPos;
+ UInt32 checkDicSize;
+ unsigned state;
+ UInt32 reps[4];
+ unsigned remainLen;
+ int needFlush;
+ int needInitState;
+ UInt32 numProbs;
+ unsigned tempBufSize;
+ Byte tempBuf[LZMA_REQUIRED_INPUT_MAX];
+} CLzmaDec;
+
+#define LzmaDec_Construct(p) { (p)->dic = 0; (p)->probs = 0; }
+
+void LzmaDec_Init(CLzmaDec *p);
+
+/* There are two types of LZMA streams:
+ 0) Stream with end mark. That end mark adds about 6 bytes to compressed size.
+ 1) Stream without end mark. You must know exact uncompressed size to decompress such stream. */
+
+typedef enum
+{
+ LZMA_FINISH_ANY, /* finish at any point */
+ LZMA_FINISH_END /* block must be finished at the end */
+} ELzmaFinishMode;
+
+/* ELzmaFinishMode has meaning only if the decoding reaches output limit !!!
+
+ You must use LZMA_FINISH_END, when you know that current output buffer
+ covers last bytes of block. In other cases you must use LZMA_FINISH_ANY.
+
+ If LZMA decoder sees end marker before reaching output limit, it returns SZ_OK,
+ and output value of destLen will be less than output buffer size limit.
+ You can check status result also.
+
+ You can use multiple checks to test data integrity after full decompression:
+ 1) Check Result and "status" variable.
+ 2) Check that output(destLen) = uncompressedSize, if you know real uncompressedSize.
+ 3) Check that output(srcLen) = compressedSize, if you know real compressedSize.
+ You must use correct finish mode in that case. */
+
+typedef enum
+{
+ LZMA_STATUS_NOT_SPECIFIED, /* use main error code instead */
+ LZMA_STATUS_FINISHED_WITH_MARK, /* stream was finished with end mark. */
+ LZMA_STATUS_NOT_FINISHED, /* stream was not finished */
+ LZMA_STATUS_NEEDS_MORE_INPUT, /* you must provide more input bytes */
+ LZMA_STATUS_MAYBE_FINISHED_WITHOUT_MARK /* there is probability that stream was finished without end mark */
+} ELzmaStatus;
+
+/* ELzmaStatus is used only as output value for function call */
+
+
+/* ---------- Interfaces ---------- */
+
+/* There are 3 levels of interfaces:
+ 1) Dictionary Interface
+ 2) Buffer Interface
+ 3) One Call Interface
+ You can select any of these interfaces, but don't mix functions from different
+ groups for same object. */
+
+
+/* There are two variants to allocate state for Dictionary Interface:
+ 1) LzmaDec_Allocate / LzmaDec_Free
+ 2) LzmaDec_AllocateProbs / LzmaDec_FreeProbs
+ You can use variant 2, if you set dictionary buffer manually.
+ For Buffer Interface you must always use variant 1.
+
+LzmaDec_Allocate* can return:
+ SZ_OK
+ SZ_ERROR_MEM - Memory allocation error
+ SZ_ERROR_UNSUPPORTED - Unsupported properties
+*/
+
+SRes LzmaDec_AllocateProbs(CLzmaDec *p, const Byte *props, unsigned propsSize, ISzAlloc *alloc);
+void LzmaDec_FreeProbs(CLzmaDec *p, ISzAlloc *alloc);
+
+SRes LzmaDec_Allocate(CLzmaDec *state, const Byte *prop, unsigned propsSize, ISzAlloc *alloc);
+void LzmaDec_Free(CLzmaDec *state, ISzAlloc *alloc);
+
+/* ---------- Dictionary Interface ---------- */
+
+/* You can use it, if you want to eliminate the overhead for data copying from
+ dictionary to some other external buffer.
+ You must work with CLzmaDec variables directly in this interface.
+
+ STEPS:
+ LzmaDec_Constr()
+ LzmaDec_Allocate()
+ for (each new stream)
+ {
+ LzmaDec_Init()
+ while (it needs more decompression)
+ {
+ LzmaDec_DecodeToDic()
+ use data from CLzmaDec::dic and update CLzmaDec::dicPos
+ }
+ }
+ LzmaDec_Free()
+*/
+
+/* LzmaDec_DecodeToDic
+
+ The decoding to internal dictionary buffer (CLzmaDec::dic).
+ You must manually update CLzmaDec::dicPos, if it reaches CLzmaDec::dicBufSize !!!
+
+finishMode:
+ It has meaning only if the decoding reaches output limit (dicLimit).
+ LZMA_FINISH_ANY - Decode just dicLimit bytes.
+ LZMA_FINISH_END - Stream must be finished after dicLimit.
+
+Returns:
+ SZ_OK
+ status:
+ LZMA_STATUS_FINISHED_WITH_MARK
+ LZMA_STATUS_NOT_FINISHED
+ LZMA_STATUS_NEEDS_MORE_INPUT
+ LZMA_STATUS_MAYBE_FINISHED_WITHOUT_MARK
+ SZ_ERROR_DATA - Data error
+*/
+
+SRes LzmaDec_DecodeToDic(CLzmaDec *p, SizeT dicLimit,
+ const Byte *src, SizeT *srcLen, ELzmaFinishMode finishMode, ELzmaStatus *status);
+
+
+/* ---------- Buffer Interface ---------- */
+
+/* It's zlib-like interface.
+ See LzmaDec_DecodeToDic description for information about STEPS and return results,
+ but you must use LzmaDec_DecodeToBuf instead of LzmaDec_DecodeToDic and you don't need
+ to work with CLzmaDec variables manually.
+
+finishMode:
+ It has meaning only if the decoding reaches output limit (*destLen).
+ LZMA_FINISH_ANY - Decode just destLen bytes.
+ LZMA_FINISH_END - Stream must be finished after (*destLen).
+*/
+
+SRes LzmaDec_DecodeToBuf(CLzmaDec *p, Byte *dest, SizeT *destLen,
+ const Byte *src, SizeT *srcLen, ELzmaFinishMode finishMode, ELzmaStatus *status);
+
+
+/* ---------- One Call Interface ---------- */
+
+/* LzmaDecode
+
+finishMode:
+ It has meaning only if the decoding reaches output limit (*destLen).
+ LZMA_FINISH_ANY - Decode just destLen bytes.
+ LZMA_FINISH_END - Stream must be finished after (*destLen).
+
+Returns:
+ SZ_OK
+ status:
+ LZMA_STATUS_FINISHED_WITH_MARK
+ LZMA_STATUS_NOT_FINISHED
+ LZMA_STATUS_MAYBE_FINISHED_WITHOUT_MARK
+ SZ_ERROR_DATA - Data error
+ SZ_ERROR_MEM - Memory allocation error
+ SZ_ERROR_UNSUPPORTED - Unsupported properties
+ SZ_ERROR_INPUT_EOF - It needs more bytes in input buffer (src).
+*/
+
+SRes LzmaDecode(Byte *dest, SizeT *destLen, const Byte *src, SizeT *srcLen,
+ const Byte *propData, unsigned propSize, ELzmaFinishMode finishMode,
+ ELzmaStatus *status, ISzAlloc *alloc);
+
+#endif
diff --git a/drivers/video/wmt/bootanimation/Makefile b/drivers/video/wmt/bootanimation/Makefile
new file mode 100644
index 00000000..f0d021a8
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/Makefile
@@ -0,0 +1,19 @@
+#
+# This is a modified version of zlib, which does all memory
+# allocation ahead of time.
+#
+# This is only the decompression, see zlib_deflate for the
+# the compression
+#
+# Decompression needs to be serialized for each memory
+# allocation.
+#
+# (The upsides of the simplification is that you can't get in
+# any nasty situations wrt memory management, and that the
+# uncompression can be done without blocking on allocation).
+#
+lzma_dec-objs := LzmaDec.o
+
+obj-$(CONFIG_LOGO_WMT_ANIMATION) += lzma_dec.o buffer.o animation.o
+
+
diff --git a/drivers/video/wmt/bootanimation/Types.h b/drivers/video/wmt/bootanimation/Types.h
new file mode 100644
index 00000000..1b74ece9
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/Types.h
@@ -0,0 +1,253 @@
+/*++
+ linux/drivers/video/wmt/Types.h
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#ifndef __7Z_TYPES_H
+#define __7Z_TYPES_H
+
+#include <stddef.h>
+
+#ifdef _WIN32
+#include <windows.h>
+#endif
+
+#ifndef EXTERN_C_BEGIN
+#ifdef __cplusplus
+#define EXTERN_C_BEGIN extern "C" {
+#define EXTERN_C_END }
+#else
+#define EXTERN_C_BEGIN
+#define EXTERN_C_END
+#endif
+#endif
+
+EXTERN_C_BEGIN
+
+#define SZ_OK 0
+
+#define SZ_ERROR_DATA 1
+#define SZ_ERROR_MEM 2
+#define SZ_ERROR_CRC 3
+#define SZ_ERROR_UNSUPPORTED 4
+#define SZ_ERROR_PARAM 5
+#define SZ_ERROR_INPUT_EOF 6
+#define SZ_ERROR_OUTPUT_EOF 7
+#define SZ_ERROR_READ 8
+#define SZ_ERROR_WRITE 9
+#define SZ_ERROR_PROGRESS 10
+#define SZ_ERROR_FAIL 11
+#define SZ_ERROR_THREAD 12
+
+#define SZ_ERROR_ARCHIVE 16
+#define SZ_ERROR_NO_ARCHIVE 17
+
+typedef int SRes;
+
+#ifdef _WIN32
+typedef DWORD WRes;
+#else
+typedef int WRes;
+#endif
+
+#ifndef RINOK
+#define RINOK(x) { int __result__ = (x); if (__result__ != 0) return __result__; }
+#endif
+
+typedef unsigned char Byte;
+typedef short Int16;
+typedef unsigned short UInt16;
+
+#ifdef _LZMA_UINT32_IS_ULONG
+typedef long Int32;
+typedef unsigned long UInt32;
+#else
+typedef int Int32;
+typedef unsigned int UInt32;
+#endif
+
+#ifdef _SZ_NO_INT_64
+
+/* define _SZ_NO_INT_64, if your compiler doesn't support 64-bit integers.
+ NOTES: Some code will work incorrectly in that case! */
+
+typedef long Int64;
+typedef unsigned long UInt64;
+
+#else
+
+#if defined(_MSC_VER) || defined(__BORLANDC__)
+typedef __int64 Int64;
+typedef unsigned __int64 UInt64;
+#else
+typedef long long int Int64;
+typedef unsigned long long int UInt64;
+#endif
+
+#endif
+
+#ifdef _LZMA_NO_SYSTEM_SIZE_T
+typedef UInt32 SizeT;
+#else
+typedef size_t SizeT;
+#endif
+
+typedef int Bool;
+#define True 1
+#define False 0
+
+
+#ifdef _WIN32
+#define MY_STD_CALL __stdcall
+#else
+#define MY_STD_CALL
+#endif
+
+#ifdef _MSC_VER
+
+#if _MSC_VER >= 1300
+#define MY_NO_INLINE __declspec(noinline)
+#else
+#define MY_NO_INLINE
+#endif
+
+#define MY_CDECL __cdecl
+#define MY_FAST_CALL __fastcall
+
+#else
+
+#define MY_CDECL
+#define MY_FAST_CALL
+
+#endif
+
+
+/* The following interfaces use first parameter as pointer to structure */
+
+typedef struct
+{
+ Byte (*Read)(void *p); /* reads one byte, returns 0 in case of EOF or error */
+} IByteIn;
+
+typedef struct
+{
+ void (*Write)(void *p, Byte b);
+} IByteOut;
+
+typedef struct
+{
+ SRes (*Read)(void *p, void *buf, size_t *size);
+ /* if (input(*size) != 0 && output(*size) == 0) means end_of_stream.
+ (output(*size) < input(*size)) is allowed */
+} ISeqInStream;
+
+/* it can return SZ_ERROR_INPUT_EOF */
+SRes SeqInStream_Read(ISeqInStream *stream, void *buf, size_t size);
+SRes SeqInStream_Read2(ISeqInStream *stream, void *buf, size_t size, SRes errorType);
+SRes SeqInStream_ReadByte(ISeqInStream *stream, Byte *buf);
+
+typedef struct
+{
+ size_t (*Write)(void *p, const void *buf, size_t size);
+ /* Returns: result - the number of actually written bytes.
+ (result < size) means error */
+} ISeqOutStream;
+
+typedef enum
+{
+ SZ_SEEK_SET = 0,
+ SZ_SEEK_CUR = 1,
+ SZ_SEEK_END = 2
+} ESzSeek;
+
+typedef struct
+{
+ SRes (*Read)(void *p, void *buf, size_t *size); /* same as ISeqInStream::Read */
+ SRes (*Seek)(void *p, Int64 *pos, ESzSeek origin);
+} ISeekInStream;
+
+typedef struct
+{
+ SRes (*Look)(void *p, const void **buf, size_t *size);
+ /* if (input(*size) != 0 && output(*size) == 0) means end_of_stream.
+ (output(*size) > input(*size)) is not allowed
+ (output(*size) < input(*size)) is allowed */
+ SRes (*Skip)(void *p, size_t offset);
+ /* offset must be <= output(*size) of Look */
+
+ SRes (*Read)(void *p, void *buf, size_t *size);
+ /* reads directly (without buffer). It's same as ISeqInStream::Read */
+ SRes (*Seek)(void *p, Int64 *pos, ESzSeek origin);
+} ILookInStream;
+
+SRes LookInStream_LookRead(ILookInStream *stream, void *buf, size_t *size);
+SRes LookInStream_SeekTo(ILookInStream *stream, UInt64 offset);
+
+/* reads via ILookInStream::Read */
+SRes LookInStream_Read2(ILookInStream *stream, void *buf, size_t size, SRes errorType);
+SRes LookInStream_Read(ILookInStream *stream, void *buf, size_t size);
+
+#define LookToRead_BUF_SIZE (1 << 14)
+
+typedef struct
+{
+ ILookInStream s;
+ ISeekInStream *realStream;
+ size_t pos;
+ size_t size;
+ Byte buf[LookToRead_BUF_SIZE];
+} CLookToRead;
+
+void LookToRead_CreateVTable(CLookToRead *p, int lookahead);
+void LookToRead_Init(CLookToRead *p);
+
+typedef struct
+{
+ ISeqInStream s;
+ ILookInStream *realStream;
+} CSecToLook;
+
+void SecToLook_CreateVTable(CSecToLook *p);
+
+typedef struct
+{
+ ISeqInStream s;
+ ILookInStream *realStream;
+} CSecToRead;
+
+void SecToRead_CreateVTable(CSecToRead *p);
+
+typedef struct
+{
+ SRes (*Progress)(void *p, UInt64 inSize, UInt64 outSize);
+ /* Returns: result. (result != SZ_OK) means break.
+ Value (UInt64)(Int64)-1 for size means unknown value. */
+} ICompressProgress;
+
+typedef struct
+{
+ void *(*Alloc)(void *p, size_t size);
+ void (*Free)(void *p, void *address); /* address can be 0 */
+ int private;
+} ISzAlloc;
+
+#define IAlloc_Alloc(p, size) (p)->Alloc((p), size)
+#define IAlloc_Free(p, a) (p)->Free((p), a)
+
+EXTERN_C_END
+
+#endif
diff --git a/drivers/video/wmt/bootanimation/anim_data.h b/drivers/video/wmt/bootanimation/anim_data.h
new file mode 100644
index 00000000..079528f6
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/anim_data.h
@@ -0,0 +1,930 @@
+/*++
+ linux/drivers/video/wmt/anim_data.h
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+static const unsigned char __initdata anim_data[] =
+{
+ 0x21,0x43,0x34,0x12,0x01,0x01,0x01,0x00,0xAA,0x38,0x00,0x00,0x05,0x01,0x00,0x00,
+ 0x20,0x00,0x00,0x00,0x20,0x02,0x00,0x00,0x01,0x02,0x00,0x00,0xD8,0xFF,0x01,0x00,
+ 0x64,0x00,0x00,0x00,0x21,0x00,0x00,0x00,0x7E,0x38,0x00,0x00,0x5D,0x00,0x00,0x40,
+ 0x00,0x00,0xC4,0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x6F,0xFD,0xFF,0xFF,0xA3,
+ 0xB7,0xFE,0xD9,0xBF,0x11,0xE8,0x0D,0x8B,0xC0,0xF6,0x05,0x19,0xBF,0x01,0x86,0x8C,
+ 0xCB,0xE9,0x9E,0xEA,0x0E,0x25,0xB9,0x55,0xEB,0xCF,0x2E,0x7F,0x12,0xC1,0x7D,0xF3,
+ 0x20,0x9F,0xCE,0x4B,0xD3,0xA1,0xC5,0x29,0x2E,0x78,0x11,0x91,0x41,0x01,0x5D,0xC2,
+ 0x66,0xAC,0x26,0xA8,0x42,0xCD,0x15,0xCC,0x13,0xE0,0xEA,0x77,0xC5,0x42,0x67,0xF8,
+ 0xFF,0x3B,0x44,0xE6,0x50,0xF4,0xDB,0xB4,0x8A,0x44,0x71,0x36,0x27,0xCB,0xE8,0xA3,
+ 0x23,0x09,0x64,0x82,0x3C,0x54,0x6C,0x86,0x3A,0xE4,0x8C,0x3D,0x85,0x9C,0xEE,0x88,
+ 0x9A,0x23,0x2F,0xE7,0x15,0x39,0x0E,0x48,0x6B,0x5B,0xA3,0x69,0x8E,0xA9,0xF6,0x30,
+ 0xC2,0x2A,0x04,0x6C,0x1F,0x9B,0x02,0xC6,0x88,0xFC,0x40,0x8F,0x59,0xE7,0x6F,0x09,
+ 0x95,0x00,0x16,0xE1,0x6D,0xC4,0x52,0x5F,0x28,0x64,0x9E,0x6B,0xFA,0x9C,0x3C,0x35,
+ 0x33,0xC1,0x06,0xFB,0xB7,0x79,0x2B,0xAA,0xCE,0xD9,0xA1,0x3E,0x38,0xC7,0xF2,0x53,
+ 0xFB,0x23,0x82,0x76,0xDD,0x2B,0x3E,0x0C,0x8D,0xA6,0xAC,0xD6,0x27,0x4F,0x24,0xC1,
+ 0x6B,0x2D,0xF8,0xD1,0xE1,0xE2,0xC2,0x54,0x8D,0x7A,0xC7,0x4C,0x35,0x4B,0x09,0x7A,
+ 0x73,0xDE,0xB5,0xF1,0xB7,0x11,0xCE,0xAD,0x34,0x7F,0x23,0xC7,0x48,0xC5,0x41,0xB2,
+ 0x25,0xD6,0x68,0x29,0xD7,0xDA,0x92,0xAF,0x11,0x60,0x07,0xDB,0xCA,0xA1,0x59,0xD4,
+ 0xF1,0xC2,0x70,0x2A,0xC8,0xDF,0x2B,0xA4,0xD4,0x0B,0x5E,0xFA,0x30,0x27,0x1E,0xDB,
+ 0x68,0xB0,0xD6,0x25,0x11,0xEF,0xDC,0x48,0xD4,0x1C,0xF7,0xB8,0x7E,0xF0,0xBF,0x2C,
+ 0x4D,0x61,0xF0,0x94,0x20,0xA2,0x4C,0xF1,0x6F,0x52,0xEC,0x30,0xF0,0x92,0x7D,0xCB,
+ 0x4A,0x8E,0xA3,0x06,0x6F,0xBB,0x7A,0x6B,0xCB,0x5F,0xE7,0xE4,0x5D,0x3D,0x0B,0x5A,
+ 0xB5,0xC7,0x81,0x59,0x35,0x72,0x6C,0x21,0xED,0x5C,0xF4,0xBF,0xD7,0xEA,0x20,0xCC,
+ 0x3F,0x72,0x85,0x51,0xF9,0xE2,0x44,0xB6,0x73,0xD1,0xE0,0x07,0x90,0x41,0x94,0xC5,
+ 0x40,0xCD,0xBC,0x21,0xD4,0x9D,0x86,0xE5,0x1F,0xDE,0xDE,0x38,0xB3,0x30,0x23,0x15,
+ 0x6F,0x99,0x4F,0x12,0x12,0x5B,0x39,0x4F,0x98,0x34,0x2B,0x69,0x11,0xBF,0xF8,0x8E,
+ 0xDD,0x77,0x8E,0x63,0x9F,0xEE,0xD7,0x60,0xE9,0x2B,0x84,0x9B,0x4F,0xDE,0xD5,0x5D,
+ 0xF6,0x76,0x6D,0x3A,0x0C,0xD3,0x21,0xBA,0x98,0xD9,0x62,0xA4,0xA3,0x88,0xF4,0xB4,
+ 0x8C,0x2B,0x67,0xE6,0x49,0x0A,0x89,0x72,0x09,0x90,0xAD,0x14,0x31,0xA1,0x18,0x79,
+ 0xC9,0x6F,0xCF,0x41,0xDD,0x74,0x01,0x43,0x8B,0x38,0x71,0x0B,0xD4,0x23,0x8C,0x4B,
+ 0x83,0x17,0xBC,0x56,0xEB,0x1E,0x2C,0x7E,0x6F,0x86,0x9B,0x49,0xFF,0x38,0x01,0xC1,
+ 0xE7,0x2A,0x94,0xA4,0x7D,0x37,0xB8,0xBA,0xE1,0x8F,0xCA,0x0B,0x78,0xC5,0xD2,0x0A,
+ 0x36,0xA6,0xE6,0xC9,0x92,0x08,0xE5,0x4E,0xCD,0xF2,0xB7,0xB8,0xEF,0xEC,0x9F,0x37,
+ 0xDC,0x7B,0x24,0x0C,0x6B,0xB4,0xB4,0xAB,0x04,0x5A,0x2C,0x84,0x95,0xC6,0xCE,0x5D,
+ 0x02,0x2A,0x74,0x78,0xB0,0xDB,0x30,0xFA,0xDA,0x67,0x84,0xF3,0x57,0x8E,0x6D,0x67,
+ 0x7D,0xAE,0x28,0x03,0x31,0x05,0x0E,0xD4,0x5F,0x64,0x4D,0xC7,0xDE,0x89,0x8D,0xB8,
+ 0x07,0xC1,0x38,0xE9,0x01,0x53,0x3D,0xEE,0xFB,0x0A,0x0A,0x20,0x6F,0xF3,0x2B,0x83,
+ 0xD9,0xD4,0x31,0xB4,0x33,0xAD,0x5D,0x74,0xBC,0xF8,0x67,0x3A,0xCC,0x7A,0xA3,0x10,
+ 0x8F,0x45,0x31,0x34,0x58,0xC3,0x44,0xD8,0x03,0x18,0xF9,0xD2,0xFA,0xC5,0xDB,0x0B,
+ 0x0C,0xB9,0x15,0x09,0xEE,0xB0,0x07,0x80,0xF6,0x04,0xDF,0x0B,0x41,0x08,0xE0,0x3C,
+ 0x5E,0xF5,0x82,0xDC,0x20,0x40,0x22,0x96,0x1C,0x31,0xD6,0x4E,0x7C,0x13,0xD8,0x17,
+ 0xC5,0x1A,0xFB,0xB0,0x7B,0x9B,0xE8,0x45,0xA9,0x53,0x98,0x21,0xB3,0xE8,0xD2,0x3F,
+ 0xDE,0x4A,0xF2,0xF9,0x0F,0xBF,0xBE,0x62,0x56,0xB7,0x90,0xBA,0xA4,0x43,0x39,0x60,
+ 0x0A,0x4B,0xCE,0xD2,0x46,0xBC,0x90,0x04,0xE1,0xAD,0x12,0xCB,0xBE,0x17,0xC3,0x9A,
+ 0x9F,0xE4,0xAD,0x0B,0xE0,0x21,0x6B,0x42,0xD6,0x52,0xBB,0x51,0xFE,0xFB,0x50,0x09,
+ 0xA3,0x77,0xC7,0x06,0x49,0xDC,0xCC,0x5C,0xBB,0xA8,0x5A,0x87,0x11,0xFA,0x67,0x17,
+ 0x1A,0x41,0x12,0x71,0x88,0xE1,0x99,0xEC,0xEB,0xE5,0xBC,0x4E,0x8F,0x8F,0x51,0x8E,
+ 0xFC,0x0C,0xC4,0x5A,0xCB,0x09,0x9A,0xDA,0x24,0xCB,0x13,0xAC,0x82,0x77,0xC6,0x8F,
+ 0x4C,0x08,0xF4,0xEE,0x80,0x01,0xEF,0xEC,0xE1,0x47,0x22,0xEC,0xEE,0xF9,0x7D,0x56,
+ 0xAF,0x8E,0xF2,0xF2,0x85,0xA3,0x9C,0xB5,0x1E,0xD7,0xFF,0xB1,0xEA,0x12,0xB5,0xE2,
+ 0xDF,0x01,0x13,0x3A,0x8F,0xAA,0x67,0x7A,0x06,0xE0,0x7F,0xC2,0xF1,0xF5,0x35,0x3F,
+ 0x5E,0x68,0x72,0x1C,0x6B,0x49,0x8D,0xD6,0xC0,0xCA,0xCC,0xFF,0xC8,0xD9,0xE2,0x75,
+ 0x1C,0x50,0x1A,0xEC,0xB0,0xDB,0x7B,0x88,0xD1,0x81,0xE2,0xEA,0xA8,0x1C,0xB2,0x64,
+ 0xFC,0x3C,0x66,0x40,0xDC,0xB0,0x39,0x57,0x2E,0x80,0x65,0x34,0x0F,0xA0,0xBE,0xFF,
+ 0x6D,0x5D,0x1A,0xEE,0xF3,0x18,0x77,0x06,0x00,0x49,0x9D,0xDC,0x27,0xD8,0xB9,0x2E,
+ 0xA5,0xC8,0x52,0x9B,0xBE,0xA1,0x32,0x83,0xA2,0x35,0x1A,0x42,0xA4,0xA8,0x95,0x98,
+ 0x31,0xE7,0x14,0xFC,0xBA,0xD5,0x70,0x0C,0x33,0x33,0x21,0x30,0xB9,0xD2,0xDA,0x4C,
+ 0x21,0x05,0xA4,0x0F,0xD2,0x73,0x7D,0xC0,0xD3,0xD3,0xED,0x67,0x11,0x0B,0xD3,0xCF,
+ 0x9E,0xBE,0xDF,0x91,0x2C,0x3F,0x3D,0x50,0xBF,0x02,0xE2,0xC6,0xC5,0xE0,0xA3,0xB4,
+ 0x60,0x6A,0x6C,0xC0,0x57,0x7E,0x4B,0x84,0x0B,0x6B,0x4E,0xDE,0x70,0x04,0xED,0xE0,
+ 0x2B,0xEB,0x95,0xDB,0xDF,0xA6,0x65,0x0C,0xF0,0x90,0x39,0x06,0xE0,0x6D,0xB8,0xDB,
+ 0x21,0x42,0xA9,0x0F,0x35,0x8C,0x32,0xD3,0xCC,0x1E,0x49,0x3D,0xC9,0xBB,0x04,0x08,
+ 0x14,0x94,0x9F,0x2A,0x10,0xF9,0x60,0xAE,0x91,0xEF,0x88,0xDA,0x4B,0xA1,0x57,0xDA,
+ 0xD8,0xD4,0x81,0x67,0x8B,0x4D,0x19,0xCD,0x04,0x0D,0xFE,0x18,0xDD,0x40,0x09,0xCE,
+ 0x31,0x77,0xEB,0xDC,0xD0,0xB8,0xB1,0x8B,0x31,0x36,0x1A,0xF0,0xD7,0xF3,0xC3,0xD2,
+ 0x7B,0x37,0xDC,0x34,0xBA,0x31,0xAF,0xC6,0x87,0x4E,0x0B,0x2D,0x30,0x33,0xA9,0xEA,
+ 0xC2,0x68,0x8C,0xCF,0x41,0xD5,0x93,0x22,0x2A,0x94,0x46,0x6D,0xF7,0x2A,0x89,0x83,
+ 0x2A,0xD0,0x13,0xE8,0xAD,0xE9,0x82,0x61,0xA4,0xD8,0x94,0xFC,0xFF,0x9D,0x9B,0xD9,
+ 0x68,0x36,0x8E,0x05,0xA5,0x61,0x9B,0x7C,0x6F,0x01,0x7E,0x90,0xC9,0x57,0x2B,0x6C,
+ 0x9C,0xC0,0xD3,0xEC,0xB7,0x89,0x43,0x97,0xF5,0xA0,0x64,0x89,0xB1,0x1D,0xF9,0x5A,
+ 0x69,0x92,0x16,0x3E,0x5E,0x52,0x53,0x78,0x61,0x03,0xFB,0xA3,0x3F,0x97,0x46,0xFE,
+ 0x98,0xE2,0xB6,0x7B,0x2A,0x48,0x73,0x91,0x11,0xC5,0xF0,0xAF,0xF8,0xE3,0x2E,0xF6,
+ 0xD8,0x81,0xE7,0xA0,0x44,0x15,0x56,0x19,0x43,0xF0,0xAC,0x40,0x51,0xBD,0x9F,0xFD,
+ 0x14,0x58,0x63,0x68,0x3A,0x91,0x94,0xF0,0x0F,0x94,0xE8,0xA0,0x8E,0xFB,0xC4,0x46,
+ 0xF4,0x0F,0xD3,0x92,0xD9,0xCC,0x19,0x6C,0xE9,0x52,0x4A,0x56,0xD5,0x82,0xA7,0x00,
+ 0x0B,0xB2,0xC9,0x99,0xD8,0xA4,0xA7,0x5B,0xC6,0x99,0xC4,0x87,0xCE,0xE5,0x45,0xE6,
+ 0xC4,0xBA,0xE1,0x26,0x84,0x03,0x0C,0x34,0x97,0xD9,0xC2,0xE3,0x1C,0xE6,0xF4,0x78,
+ 0x9F,0x1A,0x29,0xEB,0x5B,0x88,0x85,0x4A,0xE9,0x06,0xEE,0x5A,0x61,0xCF,0x47,0xE5,
+ 0xA5,0x68,0x99,0xFD,0x58,0x13,0x49,0x96,0xA4,0xA1,0xEC,0xF1,0xB7,0x0C,0x96,0x92,
+ 0x5D,0xC9,0x46,0x38,0xA3,0x97,0x71,0x92,0x5F,0x5E,0x2D,0xB6,0x8C,0x96,0x0B,0xE6,
+ 0xF8,0x8D,0x46,0x61,0x78,0xAD,0xD4,0x62,0x2F,0xD2,0x3B,0x0F,0x46,0x65,0x1F,0x3B,
+ 0x0F,0x2B,0xD3,0x98,0x4D,0x90,0x34,0xD5,0x5E,0x11,0x88,0x93,0x46,0xE8,0x57,0x1A,
+ 0x47,0xA7,0xDB,0xA9,0xC5,0x5B,0x49,0xAB,0x1A,0x52,0xF4,0xDB,0xE0,0x4E,0x0C,0xA2,
+ 0x28,0x1C,0xCB,0x04,0xF1,0xCF,0x2B,0x41,0xBF,0x46,0xC5,0x92,0x4E,0x49,0x82,0x5B,
+ 0xD4,0x36,0x3F,0xFC,0x7D,0xCD,0x8A,0xCB,0xEC,0x75,0xE7,0x2E,0xA9,0x1A,0x5A,0xFD,
+ 0x4C,0x15,0x0E,0x82,0x2A,0x8C,0x15,0x14,0x5D,0x70,0x70,0xE6,0x05,0xAE,0x5C,0x67,
+ 0xF7,0xF0,0x09,0x23,0x0A,0x27,0xA0,0x1D,0x7F,0x7D,0x01,0x50,0x7E,0xEB,0x35,0x79,
+ 0x39,0xA9,0x83,0xCB,0x89,0xDC,0x76,0xB1,0x77,0x96,0x1E,0xF8,0x71,0xD5,0x58,0xD9,
+ 0x93,0x6E,0x99,0x6A,0xBB,0x89,0x39,0xDF,0x48,0x18,0xDB,0xE7,0xD3,0x78,0x55,0x94,
+ 0x35,0xAA,0x84,0x1C,0xBD,0xD5,0xF4,0xED,0x94,0x9C,0x23,0x6B,0x48,0xE6,0xB7,0x66,
+ 0xFE,0x14,0x38,0xCE,0xB1,0x05,0x3D,0xBF,0xC9,0x3F,0x97,0x60,0x3C,0xB0,0xD7,0xD7,
+ 0x2B,0xAA,0x1C,0x6C,0x03,0x99,0xD9,0x41,0x42,0x4A,0x5F,0xFF,0x6B,0x2C,0x22,0x31,
+ 0xAE,0x65,0xB8,0xB2,0x70,0x19,0x8A,0xA6,0x2C,0xDC,0xA3,0xF5,0x84,0x8C,0x3B,0x4D,
+ 0xF6,0xA3,0x65,0x60,0x8E,0x30,0x79,0x51,0x77,0xC0,0x1E,0xCC,0x5D,0xC4,0x31,0x07,
+ 0xEE,0x28,0xC3,0x1D,0x9C,0xC8,0xD4,0x70,0x51,0xA9,0xE6,0x64,0x78,0x7D,0x6A,0x3A,
+ 0xE3,0xDE,0xD3,0x08,0x3A,0x01,0x9A,0x01,0x4D,0x81,0x49,0x5B,0xF4,0xEE,0xCE,0x6A,
+ 0x4C,0x31,0xB9,0x42,0xCC,0x29,0x71,0x08,0x1F,0x15,0xF5,0xCC,0x34,0x24,0xD5,0x93,
+ 0x18,0x8F,0x04,0x0F,0x7D,0xEB,0xC7,0x09,0x90,0x1A,0xE9,0xC6,0x01,0xDB,0x1A,0x84,
+ 0x33,0x5D,0x4D,0x04,0x02,0xB9,0xC4,0x8D,0xCA,0xCF,0xC6,0xCE,0x7C,0x61,0x0D,0xFF,
+ 0x18,0x87,0xEE,0x58,0x81,0xCE,0x13,0x43,0x18,0xA6,0xF2,0x80,0xAB,0x14,0xAF,0x49,
+ 0x60,0x34,0x4A,0xE4,0x6B,0x31,0xCD,0x6E,0x17,0x0D,0x6D,0x52,0xC6,0x47,0x66,0x1B,
+ 0x44,0x59,0xD7,0x49,0x2E,0x94,0x65,0x88,0x09,0x38,0x52,0xC5,0x94,0x98,0x84,0x1F,
+ 0xDD,0xFF,0xE9,0x47,0x2C,0xB7,0xE7,0x94,0x6A,0x9D,0x56,0x9F,0xAE,0xBC,0x2E,0x3E,
+ 0x97,0x15,0xA1,0x29,0xA2,0xB4,0xD5,0x8A,0x60,0xFA,0xB0,0x08,0xBD,0xF7,0xD4,0xDC,
+ 0x92,0x40,0x5B,0x85,0x75,0x50,0x0F,0x42,0x45,0x3D,0xEF,0xE9,0x9B,0xF9,0x09,0x5C,
+ 0x63,0xD3,0xF5,0x93,0xFE,0xC5,0xA0,0x1E,0x39,0xCC,0x98,0x40,0xC1,0xEE,0x21,0x7E,
+ 0x20,0xFD,0xBD,0xF7,0xEE,0xC6,0x9D,0x3B,0xD1,0xD7,0x73,0xCB,0x07,0x97,0xF9,0x0E,
+ 0xBB,0xBA,0x79,0xD0,0x23,0xE7,0x36,0x0F,0xE0,0x9A,0x15,0x79,0x6A,0x6A,0x37,0xFB,
+ 0xC9,0x20,0x7D,0xFD,0x39,0x6F,0x79,0x2F,0xD0,0x24,0x94,0x41,0xE3,0xA3,0x68,0x0A,
+ 0xDD,0x15,0x22,0x9C,0xC9,0x87,0xD6,0x4F,0x08,0x4C,0xDA,0x3F,0x9B,0xBC,0x40,0x9A,
+ 0x08,0xAE,0xBB,0x07,0x41,0xEC,0x75,0x2B,0x90,0xA9,0x0D,0x9A,0xBF,0x98,0xCC,0xB3,
+ 0x15,0x4B,0x3A,0x6A,0x00,0x59,0xB4,0x85,0x17,0xB8,0xAE,0xA5,0xDB,0x79,0xB7,0x6D,
+ 0x92,0x30,0x7B,0xF4,0x37,0xA3,0x4E,0xDE,0x7D,0xF7,0xFC,0x7B,0x72,0x30,0xD0,0x11,
+ 0x1F,0x05,0xD8,0xF4,0xD8,0xAE,0x6E,0x52,0x5A,0xFF,0xF4,0xE3,0xCF,0x92,0xA0,0x90,
+ 0x76,0xCC,0xFB,0xFF,0x60,0xEF,0xAE,0xF6,0x86,0x50,0x36,0x5C,0x56,0x86,0x00,0x37,
+ 0x2F,0xB1,0xAD,0xA7,0x0C,0xFA,0xE1,0x74,0x93,0x5C,0x63,0xB2,0x52,0xE3,0xBA,0x1D,
+ 0x2C,0x83,0x5C,0x00,0x5A,0x7C,0xB2,0x12,0x4C,0x7D,0x81,0x45,0x30,0xD0,0x53,0xE1,
+ 0xD7,0x9F,0x02,0xC3,0x2A,0xB9,0x9A,0x9E,0x61,0xBD,0x6C,0x3B,0x0A,0xA7,0xB3,0xF9,
+ 0xFB,0x51,0xB7,0x33,0x26,0x77,0x4F,0x78,0x3E,0xA0,0xF9,0xA2,0x15,0xEA,0x3E,0xFF,
+ 0x48,0x5A,0x4B,0x05,0x3A,0xFF,0x28,0x24,0x18,0x7F,0xF9,0xF9,0x69,0x5C,0x01,0xC7,
+ 0x5F,0xB7,0x78,0xD2,0x3D,0x7F,0xF0,0x07,0xD0,0xCC,0x4C,0x42,0x9C,0x87,0xC9,0x12,
+ 0xDB,0x45,0x8B,0xC1,0x26,0x00,0xE4,0x47,0x01,0x81,0x5B,0x3F,0x38,0x2B,0x70,0x18,
+ 0x15,0x86,0xC5,0x9C,0x22,0x0B,0xE6,0x93,0x99,0x7B,0x32,0xB9,0x1A,0xA6,0x4B,0x6F,
+ 0xAA,0xEC,0x06,0xFA,0x8B,0xCD,0xEE,0x80,0xB2,0xE7,0xE0,0x5F,0x5F,0x09,0x33,0xBD,
+ 0x18,0x5B,0x90,0xEA,0xD1,0x4A,0x27,0xC3,0xD0,0x64,0x8A,0x03,0x8E,0x53,0x13,0xDD,
+ 0x96,0x61,0x4E,0xC0,0xCE,0x3A,0xA2,0x54,0xBB,0x56,0x6C,0x91,0xF9,0xF8,0xD5,0x35,
+ 0x86,0xE1,0xA0,0x8C,0xBA,0xDB,0x79,0xF4,0x22,0xC5,0x4A,0x1E,0x99,0x74,0x49,0x94,
+ 0xDE,0x64,0x1F,0x0F,0x6E,0x4A,0x5F,0x07,0xB9,0xBE,0xE0,0xCE,0x07,0xF9,0xA1,0x7A,
+ 0xA3,0x91,0x4B,0x0E,0x64,0xED,0x06,0xAE,0x98,0xC8,0xBA,0xB5,0x00,0x38,0xA7,0xE3,
+ 0x68,0xAD,0x2F,0x5A,0xAF,0xD8,0xDE,0x21,0xF1,0x0D,0x6A,0x00,0xEF,0x7F,0x9B,0x94,
+ 0x6A,0xE5,0xEC,0x2B,0xA3,0x4F,0x2D,0x7F,0xD4,0x26,0x87,0x98,0x66,0x76,0x2B,0x2C,
+ 0xFC,0x11,0x62,0x0D,0xEB,0x1C,0xE9,0xA7,0x97,0xAC,0x4D,0x24,0xB7,0x5A,0x87,0x1D,
+ 0x30,0xF8,0x38,0xE3,0x59,0x5A,0x61,0xA0,0x67,0x6A,0x15,0x56,0x25,0xA6,0xFC,0x0E,
+ 0x51,0x55,0xDB,0x5C,0xE0,0xFC,0x59,0x78,0x76,0x27,0x07,0x68,0xEF,0xF4,0xDA,0xBA,
+ 0x52,0x7E,0x5F,0xA9,0x2E,0x21,0x9C,0xBD,0x6B,0xBB,0x64,0x27,0x66,0x0B,0x63,0xC2,
+ 0x3C,0x77,0xA9,0x9B,0x2C,0x48,0x93,0xF5,0xC9,0x48,0x2C,0x70,0x05,0xA0,0x75,0x81,
+ 0x0E,0xBD,0x21,0x54,0x68,0xC4,0x93,0x78,0x77,0x1D,0xCD,0x90,0x92,0x99,0x53,0x94,
+ 0x82,0xFE,0x09,0x0C,0x77,0x2D,0x3F,0x30,0xFF,0xF1,0xC0,0x0D,0xEC,0xC4,0xBA,0xFD,
+ 0x94,0x50,0xB7,0x15,0x51,0x09,0xB5,0xD0,0x7C,0x1E,0xF8,0x1B,0xD1,0x89,0x45,0xD6,
+ 0x59,0x56,0xDA,0x69,0x39,0x1A,0x75,0x6C,0x79,0xDE,0xD0,0x42,0x45,0xC0,0xCC,0xDC,
+ 0x20,0x7E,0x69,0x6A,0xAC,0xCA,0x46,0x7C,0xB9,0xB6,0x4A,0xC8,0xD8,0x22,0x99,0x63,
+ 0xF1,0x76,0x06,0x39,0x8F,0xCA,0xC6,0xA5,0x9E,0x59,0x43,0xDE,0xDF,0xCA,0xE8,0x07,
+ 0xAC,0x83,0xC5,0xD1,0xB7,0xF4,0xF2,0x1E,0xAD,0xFB,0x6C,0x30,0x4C,0x26,0xE5,0xEB,
+ 0x6E,0x02,0x0A,0xE0,0xC9,0x22,0x93,0xED,0x64,0x98,0x09,0x2A,0x1E,0x01,0x95,0x2B,
+ 0x00,0xE3,0xDC,0x62,0x25,0x75,0xC7,0x6C,0xB7,0xC7,0x60,0xE1,0x63,0x2A,0x8D,0xB1,
+ 0xBE,0x6C,0xE7,0x80,0xC0,0x03,0x4F,0x45,0x8A,0xE3,0x13,0x4C,0xB9,0x10,0xD1,0x98,
+ 0x4B,0xCE,0x94,0x6D,0x03,0x50,0x53,0x16,0xEC,0xEA,0x35,0x6F,0x90,0xE0,0x00,0x83,
+ 0x55,0xC3,0xAF,0x72,0xFD,0x78,0xC3,0xA5,0xE2,0x77,0xB4,0x00,0xED,0x21,0x1D,0x8B,
+ 0xB3,0xD2,0x0E,0x95,0x88,0x54,0x05,0xD9,0x4E,0x9D,0x4D,0x9B,0xFB,0x77,0x9C,0x6B,
+ 0xCF,0x0A,0x0F,0x86,0x9C,0xFE,0x5B,0x4A,0xD3,0x94,0xF9,0x24,0xE1,0x26,0x4E,0x4E,
+ 0xC7,0x0A,0x59,0x2E,0xF8,0x03,0x2E,0xA2,0x4E,0x7C,0x43,0x54,0x8B,0xAA,0x4A,0x55,
+ 0xF2,0x35,0xFD,0xAC,0xC0,0xCB,0x4A,0x8C,0x52,0x80,0xB8,0x70,0x4F,0x70,0x0D,0xE2,
+ 0x48,0x42,0x50,0xA5,0x9A,0x86,0x5D,0x3F,0xD9,0x33,0xE5,0xCE,0x85,0x35,0x72,0xD3,
+ 0xF5,0x40,0x3D,0x2C,0x42,0x3E,0x02,0x0B,0x98,0x81,0x59,0x03,0x4E,0x48,0xA0,0xC7,
+ 0xA2,0xC0,0xA5,0x3B,0x3C,0xB5,0x88,0x36,0xDC,0xDA,0x48,0x51,0x1A,0x5E,0xF7,0xB6,
+ 0xD9,0x7E,0xA2,0xEB,0x94,0xD9,0xA6,0x32,0xC6,0x55,0x74,0x63,0xE2,0xCB,0xB0,0x90,
+ 0xEF,0xBA,0xA4,0x1C,0x85,0xB6,0x9C,0x2B,0x79,0x20,0xCD,0x71,0xD9,0x01,0x05,0x42,
+ 0x4F,0x6F,0x4E,0x7C,0x09,0x62,0x0C,0x69,0xC7,0x4B,0xC2,0x9D,0xE2,0x36,0x59,0xBE,
+ 0xC1,0xE0,0xA8,0x10,0xAD,0x49,0x9D,0xEB,0x1A,0xB0,0x58,0x74,0x66,0xCF,0x32,0x5F,
+ 0xD0,0x0A,0x24,0x27,0x73,0x1C,0x36,0x61,0x3A,0xEE,0x7C,0xD7,0x68,0xAB,0xBF,0xEF,
+ 0x6A,0x65,0xA6,0xCC,0x1C,0x83,0x50,0x11,0x69,0x6E,0x8A,0x2C,0xBC,0xE8,0xDC,0x5A,
+ 0x62,0x41,0x2B,0x93,0xA0,0x97,0x2E,0xFA,0xE2,0x1B,0x39,0x37,0x61,0x38,0xD9,0x3D,
+ 0x37,0x8D,0x30,0x3A,0xF6,0x19,0x3D,0x0A,0xCC,0xED,0xD6,0x92,0xA2,0x4A,0xF4,0xD7,
+ 0x94,0xEC,0x5F,0xC8,0x65,0x78,0x9F,0xC3,0x1F,0x84,0x7D,0xB1,0xDC,0x9B,0x32,0x79,
+ 0x66,0xF2,0x55,0x42,0xA7,0x73,0xFA,0x7A,0xBF,0xFA,0x8B,0x24,0xBE,0x15,0x7F,0xA5,
+ 0x1D,0x05,0x0C,0xA8,0x35,0xB2,0x37,0xAC,0x58,0x28,0x24,0x97,0xD2,0x49,0x32,0x3D,
+ 0x6E,0xA4,0xCB,0x0B,0x79,0xF7,0x1B,0x10,0x7D,0xA4,0xAD,0x3E,0xDB,0x99,0xE1,0x37,
+ 0x0C,0x71,0x90,0x95,0x4A,0x4A,0xE2,0xA6,0x31,0x68,0x94,0xC5,0x13,0xA7,0xF4,0xAE,
+ 0x4D,0xF9,0xEC,0x3C,0x80,0x30,0x0A,0x29,0x9E,0x79,0x0A,0x9C,0x42,0xFE,0x1E,0x2F,
+ 0x02,0x6D,0xBA,0xF0,0xA3,0x9B,0x89,0xE8,0x56,0xA2,0xB9,0xBE,0xFC,0xC6,0xFE,0x16,
+ 0xDF,0x82,0xCD,0x36,0x18,0xA1,0x23,0x44,0x31,0x64,0xBB,0x8C,0x6D,0x3D,0xCD,0xBF,
+ 0xC9,0xB3,0xA4,0x15,0xF4,0xA9,0x83,0xD1,0x8B,0x30,0xFB,0xBD,0x32,0xA0,0xC2,0x8E,
+ 0xD0,0x0A,0xFC,0xB2,0x99,0x61,0x4C,0x55,0xEF,0x40,0x95,0xD3,0x8F,0x4B,0x0A,0xC3,
+ 0x25,0x21,0xAA,0x4F,0xFD,0xF5,0x57,0x22,0x6E,0x5C,0xD2,0x01,0xA6,0x4D,0xDD,0x92,
+ 0x7A,0x9D,0xC7,0x38,0x14,0xDF,0x05,0xCF,0xBD,0x95,0x3F,0xC9,0x17,0x44,0x6B,0x6F,
+ 0x30,0x02,0xB2,0xCE,0xA2,0x21,0xB8,0x4C,0x07,0x18,0x3A,0x45,0xE9,0xC8,0xFE,0x0F,
+ 0xDB,0x29,0x25,0x1F,0x58,0x56,0x08,0x4F,0x1A,0xF6,0x74,0xEF,0x92,0xD1,0x29,0x24,
+ 0xC7,0xCD,0x5C,0x22,0x19,0xD3,0x50,0x77,0xC0,0xE8,0x2A,0x5E,0x94,0x3F,0xC8,0x61,
+ 0x4E,0x89,0x33,0xB0,0xA3,0xFA,0xB9,0x60,0xCB,0x83,0x81,0x5C,0x52,0x4E,0xC0,0xE5,
+ 0x92,0x80,0x9F,0x37,0x93,0xD4,0xDA,0x9B,0x5C,0x3A,0x69,0x5D,0x19,0x81,0x9F,0x9F,
+ 0xB7,0xD1,0x8D,0xCF,0xA2,0x9A,0xF7,0xCD,0xAB,0x66,0x6E,0x68,0x93,0x38,0x0A,0x1E,
+ 0x0B,0x3D,0xCC,0x0C,0xA9,0xDF,0xDE,0x38,0xF2,0x87,0x3E,0xCE,0x50,0x4A,0x36,0x02,
+ 0xD2,0x85,0xD8,0x2A,0x7E,0x1F,0x72,0xAD,0x5F,0x9E,0x28,0xC0,0xE0,0x74,0xC3,0x53,
+ 0x7A,0x94,0x49,0xD8,0x6C,0x38,0x06,0xCC,0x15,0x44,0xED,0x76,0x87,0x9D,0x1D,0x82,
+ 0x33,0x55,0xD3,0x3A,0x6F,0x25,0x28,0x2E,0x2D,0xE6,0xAE,0x1F,0xA5,0x4C,0x53,0x19,
+ 0xE4,0xFD,0x6B,0x62,0x88,0xAA,0xD0,0xC3,0x68,0x58,0x75,0x78,0xB9,0x9F,0x07,0x61,
+ 0xFE,0x90,0xE4,0x5F,0x5C,0x1E,0xE4,0x35,0xA6,0xED,0x4E,0xFD,0x1C,0x28,0xEE,0x2E,
+ 0x34,0x35,0xFE,0x0B,0x64,0xB4,0xB8,0x93,0xBF,0x44,0x92,0x67,0x0E,0x8E,0x85,0x67,
+ 0xD7,0x5F,0xF0,0x1A,0x65,0x7C,0x6E,0xB0,0x01,0xA8,0xCA,0x86,0x10,0x62,0xDB,0x02,
+ 0x2F,0x93,0x37,0x07,0xE1,0xBA,0xA9,0x09,0x80,0x6A,0x01,0xE5,0x76,0x8B,0x52,0x8F,
+ 0x00,0xC7,0x60,0xEF,0x04,0x4E,0xE6,0x40,0x9D,0xA8,0xDA,0x5B,0xD8,0xD7,0x98,0x39,
+ 0xE8,0xB9,0x26,0xCE,0xBE,0xCA,0x30,0xB9,0x11,0x8B,0xEF,0x82,0x37,0x70,0xDC,0xDF,
+ 0xC9,0x40,0x10,0xA1,0x28,0x69,0x82,0x64,0x39,0xAF,0x2D,0xD3,0x03,0x91,0x5A,0x86,
+ 0x55,0xFC,0xF0,0x55,0x2E,0xEC,0xD8,0x74,0x67,0xF7,0x9C,0x4C,0xB6,0x57,0xAF,0xEC,
+ 0x6A,0xE4,0x4B,0x73,0x89,0x3C,0xFE,0x63,0x99,0x3A,0xFF,0xC8,0x21,0x58,0x96,0x24,
+ 0xCA,0x11,0xCB,0x71,0x50,0xC9,0x7A,0xBB,0xBB,0x0F,0xFA,0x24,0x3C,0x34,0xBB,0x4E,
+ 0xAB,0x37,0x92,0xB9,0x63,0x88,0x5C,0x06,0xB0,0xF0,0x7B,0xD7,0x0C,0xFD,0xB6,0x0B,
+ 0x5F,0x78,0xE7,0x9D,0x06,0x7D,0x46,0x59,0x44,0xDD,0x31,0x91,0xC7,0x12,0x80,0xA4,
+ 0x33,0x23,0xFC,0xAE,0x62,0xE2,0xE8,0x85,0x3E,0x9C,0x4A,0xC9,0x61,0x9D,0x0D,0x65,
+ 0x61,0x72,0x45,0xF1,0xCF,0xA2,0x7F,0x9D,0xC3,0x04,0xB2,0xA5,0x68,0x3E,0xE1,0x56,
+ 0xE9,0x49,0x00,0xBF,0x19,0xCB,0x75,0xFB,0xCB,0x05,0x73,0x50,0x7E,0xB2,0x24,0x95,
+ 0x0E,0x9D,0x3D,0xA6,0xF3,0x09,0x5C,0x43,0xE8,0x6C,0xFD,0x68,0x8C,0x2F,0x64,0xCD,
+ 0xD3,0x4B,0x7E,0x66,0x70,0xB7,0xB2,0x76,0xF3,0xE2,0x63,0x92,0x0F,0x45,0xFE,0x9C,
+ 0xFB,0xD4,0x30,0xC7,0x93,0x77,0xEF,0x98,0xBE,0x66,0x56,0x51,0xB6,0xA3,0x7E,0xC7,
+ 0xBB,0x4D,0x7A,0xFD,0xCB,0x13,0x01,0x05,0x8C,0x96,0x06,0xF6,0x37,0x1E,0x78,0x70,
+ 0x13,0xCC,0x00,0xC3,0x7F,0xB8,0xA5,0xF3,0xB5,0xB3,0x4E,0xD9,0xAB,0x2E,0x1E,0xFE,
+ 0xA9,0x2D,0x4B,0xA5,0x9F,0xEB,0xD5,0x07,0x13,0x3D,0xAB,0xA4,0x1B,0xC2,0xD3,0xB3,
+ 0x57,0x41,0x96,0xBC,0x5D,0x4F,0xC3,0x5F,0x07,0x06,0x25,0x65,0x19,0xF6,0xD4,0x0D,
+ 0x40,0x66,0xA8,0x01,0x62,0x83,0xC7,0xCD,0xA8,0xCE,0xFE,0x03,0x6C,0x58,0x81,0x38,
+ 0x80,0xA9,0x0E,0x75,0x5A,0x23,0x41,0x8C,0x20,0x62,0xCA,0x69,0x1A,0x33,0x62,0x0E,
+ 0x0B,0x80,0x6E,0xD5,0x85,0x21,0xEB,0xBE,0xC0,0x3E,0x6A,0xC7,0x92,0x51,0xE6,0xDE,
+ 0x74,0x63,0xB7,0x01,0x22,0x59,0xB5,0x39,0x8B,0xF4,0xF1,0x0E,0x93,0xBB,0x7F,0xA8,
+ 0xAE,0xF2,0xD7,0x9B,0x5A,0xA4,0xFC,0x2D,0xE5,0x5F,0x0D,0x46,0x3F,0x1D,0x2B,0x5A,
+ 0xBC,0x3A,0x7D,0x00,0xAC,0x2B,0x38,0x75,0xF6,0x05,0x70,0xFE,0xD7,0xC6,0x2E,0x19,
+ 0xF9,0xDF,0xFA,0x77,0xB3,0x24,0xB8,0xB5,0xFF,0xD3,0x2C,0x06,0xD4,0xEA,0x25,0xDB,
+ 0x85,0xC9,0xC3,0x59,0x2C,0x90,0x42,0xB9,0x1F,0xB1,0xAB,0x64,0x55,0x25,0xAD,0xD3,
+ 0x82,0x5F,0x1B,0x53,0x2B,0xF1,0x51,0x7C,0xFE,0x64,0x02,0x0C,0x72,0x5F,0xD0,0x51,
+ 0x36,0x58,0xA3,0xED,0x44,0x59,0x3F,0xDD,0x75,0xEE,0x21,0xF0,0x97,0x3D,0x05,0x38,
+ 0x7D,0x29,0xF9,0x88,0x3A,0x42,0x76,0x0C,0x67,0x6F,0x61,0xAB,0x99,0xA2,0x87,0x58,
+ 0x74,0x2A,0x1F,0x42,0x25,0x02,0xED,0xD5,0x46,0x99,0x13,0xAE,0x91,0x86,0xAF,0x71,
+ 0x95,0xE6,0x93,0x12,0x51,0xF2,0xAA,0x44,0x79,0xE9,0x5B,0x58,0xEE,0xE0,0x9F,0x25,
+ 0x25,0xCC,0x39,0xA1,0xCF,0x11,0x98,0xBB,0x2E,0x82,0xBD,0xDE,0x05,0x53,0x50,0x1A,
+ 0xFC,0x1B,0x9C,0xE5,0x58,0xF4,0x8D,0xF7,0x6A,0xA3,0xDD,0x68,0xF2,0x42,0x3A,0xE2,
+ 0xF3,0xAB,0x7D,0x12,0x5F,0x0C,0x92,0x30,0xE8,0xB5,0xD7,0x22,0xEB,0x70,0x37,0x8B,
+ 0x5B,0xE4,0x6D,0x90,0x28,0x14,0x79,0xE4,0x32,0xDD,0x2F,0xE4,0x87,0x8F,0xE8,0xAD,
+ 0x3F,0x32,0x42,0x88,0x23,0xA5,0xB6,0x1A,0xD6,0xD4,0x7C,0x47,0xB4,0x19,0x91,0x5D,
+ 0x40,0xE6,0xAA,0x32,0x65,0x85,0xD6,0x95,0x77,0x57,0x65,0x82,0xCF,0x40,0xFF,0x02,
+ 0x50,0x30,0x81,0x5B,0xD1,0xDE,0x41,0x5D,0xB9,0x7B,0xC7,0xE1,0xFD,0xE0,0x41,0xFF,
+ 0x9B,0x66,0x42,0x4A,0x1B,0xAE,0xF6,0xA1,0x2A,0x4F,0x55,0x94,0x73,0x3C,0x05,0x24,
+ 0x4E,0xDE,0xBF,0xCC,0xCE,0x02,0xD1,0x16,0xFD,0x81,0x16,0x57,0xF4,0x94,0x2C,0x6C,
+ 0x85,0xAC,0x85,0x27,0xB1,0x1E,0x00,0x07,0x63,0xE0,0x0B,0x01,0xFC,0xCF,0x2D,0x40,
+ 0x44,0xB2,0x74,0x5B,0xAA,0xED,0x7A,0xB8,0x91,0x3B,0xBE,0x92,0x10,0x02,0xC9,0xDF,
+ 0xDB,0x1A,0x87,0x84,0x14,0x61,0x89,0x34,0xD1,0xB4,0x67,0xC2,0x3A,0xBC,0x36,0xE3,
+ 0x76,0x3B,0xD3,0x0F,0xDB,0x49,0x1C,0x3D,0x87,0xB2,0x8A,0xF9,0x4D,0x19,0x29,0x07,
+ 0x36,0x1C,0x1D,0x8C,0xD8,0x61,0xD3,0xE3,0x7D,0xA7,0xC6,0xC1,0x64,0x07,0xA2,0xDE,
+ 0x87,0x0C,0x77,0x82,0xD0,0x96,0x28,0x17,0x82,0x3D,0x88,0xBB,0x4D,0x47,0x08,0xEF,
+ 0x29,0x42,0xA7,0x54,0x9F,0x12,0xB8,0xDD,0x89,0x61,0x94,0xB1,0xF4,0xCE,0x5D,0x42,
+ 0x1F,0x23,0xC0,0x88,0xDB,0xDA,0xCC,0xA3,0x17,0x20,0x1B,0x8B,0x17,0xE7,0xE5,0x7F,
+ 0xC4,0x4D,0xBA,0xC8,0xE4,0x89,0x99,0x00,0x06,0x7E,0x45,0xF3,0x96,0x09,0xB8,0x3E,
+ 0xE4,0x74,0x58,0x91,0x7E,0xC7,0x92,0x18,0x28,0xC5,0xF2,0xA8,0x87,0x6E,0x11,0xFB,
+ 0x20,0x57,0xD7,0x33,0xBF,0x4B,0x47,0xE9,0xCD,0xF3,0xF2,0xE4,0x95,0xBE,0x5A,0x0B,
+ 0x70,0x70,0x51,0x71,0x4E,0x85,0x1E,0x46,0xE5,0xE4,0xE8,0xF2,0x60,0xC0,0x72,0xDA,
+ 0x1C,0x74,0xFE,0xC1,0xD8,0x37,0xB4,0x54,0xBE,0xA9,0x59,0x79,0xFD,0xFD,0x31,0xC6,
+ 0xB3,0x57,0x9D,0xAB,0xAE,0x0E,0x42,0x91,0x2C,0x44,0x41,0x61,0x07,0x0D,0x7B,0x4F,
+ 0x8A,0xD3,0x35,0x1F,0x7E,0x53,0x96,0xE5,0x2A,0x0E,0x02,0xEF,0x87,0xEB,0x83,0xD6,
+ 0xA7,0xA2,0xF8,0x16,0x26,0x96,0x60,0xA1,0x3F,0x60,0xD1,0x6C,0x4A,0x69,0x99,0xD8,
+ 0x91,0x40,0xCB,0x76,0x5C,0x59,0x62,0x4A,0xF1,0xF0,0x87,0xF5,0x0D,0x15,0x54,0xBD,
+ 0x54,0x1D,0xB5,0x7A,0x22,0xE2,0x5F,0x7A,0x35,0x7E,0x6C,0x77,0x56,0x93,0xEE,0x24,
+ 0x74,0xFF,0xDB,0x63,0x16,0x80,0x62,0x48,0xD5,0x51,0x45,0xAB,0xDA,0xFF,0x6F,0xAF,
+ 0x93,0xDA,0x47,0xD4,0xCA,0x17,0x47,0x28,0x29,0x40,0x83,0x05,0xD0,0x52,0xB5,0xF7,
+ 0x74,0x6F,0x0D,0xAB,0x36,0xD3,0x89,0x1F,0x52,0x97,0xFE,0x02,0x8E,0x2D,0x44,0x14,
+ 0x52,0x25,0x64,0xAB,0x22,0xE1,0x16,0x16,0x32,0x4A,0xBF,0x09,0x06,0x5C,0x55,0x85,
+ 0x02,0x44,0x16,0x02,0x68,0x46,0x74,0x2E,0x53,0x59,0x18,0xA3,0x4B,0x55,0x49,0xF5,
+ 0x3A,0x74,0x1F,0x4B,0x90,0x56,0xE4,0x72,0xB2,0x54,0x16,0x41,0x8F,0x52,0x9D,0x7D,
+ 0xC8,0x19,0xCC,0xE8,0xC7,0x9C,0x1A,0x30,0x67,0x93,0x3E,0x38,0x7E,0xC4,0x1F,0x73,
+ 0x0C,0x1D,0xF4,0x37,0x4C,0x57,0x52,0xDA,0x88,0xB3,0xBB,0x85,0x1F,0x1D,0x6D,0x0B,
+ 0x6E,0x6A,0xB8,0x7D,0xEB,0x6B,0x46,0x22,0x9A,0x2E,0x39,0x86,0x0B,0x50,0x95,0xCE,
+ 0xB1,0x8F,0xDE,0x34,0x7B,0x1D,0xD7,0x8E,0xD8,0xEB,0x78,0xBE,0x8B,0x3D,0x1A,0xDA,
+ 0x83,0xD4,0x8F,0xBD,0xCE,0xF9,0xD9,0x1D,0xE1,0x06,0x16,0xDA,0xA3,0xFD,0x91,0x5E,
+ 0x02,0xB5,0xFD,0x2A,0x26,0x5A,0x1E,0xA4,0x06,0x39,0xD4,0xF7,0x17,0x44,0xC9,0x93,
+ 0x54,0x59,0x0C,0x8C,0xF2,0xC9,0xFD,0xF3,0x29,0xE2,0x18,0x3D,0xA5,0x7A,0x7F,0x3F,
+ 0x27,0x16,0xAE,0x89,0x08,0x7D,0xAA,0xDE,0x2F,0x2D,0xA8,0x45,0xC6,0xAC,0xD0,0xCD,
+ 0x92,0x83,0x18,0x53,0x00,0xC7,0x41,0x7D,0xF2,0x80,0xF8,0x5D,0x83,0xE4,0xA1,0x61,
+ 0x64,0xC5,0x1F,0x8D,0xA8,0x3F,0xF9,0x4A,0x0B,0xB0,0x45,0xB9,0xD1,0xBA,0x79,0xCB,
+ 0xD9,0x67,0x86,0x2C,0x5A,0xB2,0xAE,0xAE,0xE6,0x0A,0x9F,0xA6,0x13,0x00,0x54,0x21,
+ 0xEF,0x0E,0x34,0x51,0xE7,0xF3,0x20,0x8C,0x65,0x42,0xE4,0xAC,0x3A,0xC4,0xD2,0x8B,
+ 0x54,0x04,0x3C,0xB4,0xDB,0x9F,0x0D,0x13,0x5E,0x66,0x0A,0x0D,0xAD,0x8E,0x97,0x36,
+ 0x8E,0x06,0x6F,0x1A,0x63,0x3F,0x84,0xAE,0xC1,0x84,0xF5,0x7D,0xC9,0xCB,0x49,0x79,
+ 0xF6,0x64,0x0A,0xC0,0x70,0x60,0x84,0x53,0x71,0x11,0xA8,0x3B,0xB2,0xA2,0xA0,0x05,
+ 0x5E,0x9D,0x30,0x64,0x5F,0xB7,0x21,0x7B,0xAE,0x78,0x88,0x5B,0xE9,0xE6,0xCA,0xBF,
+ 0x0B,0x5D,0xFF,0x81,0x5F,0xA6,0xDA,0x3D,0xC3,0xCD,0xB8,0xFF,0x7E,0x4A,0x57,0xAB,
+ 0x97,0x33,0xE8,0x3E,0x28,0x15,0xF5,0x7E,0x68,0x71,0xDA,0x02,0xEC,0x2A,0x8D,0xF9,
+ 0x20,0x78,0xF5,0x56,0x22,0x4B,0x0C,0x50,0xCE,0x5F,0x42,0xEA,0xB7,0x74,0x3D,0xC2,
+ 0x5B,0xE2,0x10,0x49,0x62,0x33,0x69,0xAA,0x7A,0x83,0x5D,0x81,0x48,0x43,0x7E,0x26,
+ 0xE2,0x34,0xF4,0x24,0x70,0xDC,0x77,0x11,0xFF,0x48,0xC9,0x5F,0x89,0x33,0x61,0xFD,
+ 0x18,0xB3,0x98,0x3B,0x17,0x8E,0xB1,0xB1,0xF7,0x14,0xBD,0x73,0x95,0x42,0x02,0x6A,
+ 0x27,0xDA,0x44,0x42,0xE2,0xE0,0x2C,0x6B,0x20,0xC5,0x82,0x09,0x22,0x99,0x4A,0xE1,
+ 0x85,0xAA,0xD0,0x36,0x11,0x81,0x18,0x70,0x29,0x22,0x27,0xD5,0x04,0x3E,0x7B,0xD3,
+ 0xFB,0x12,0x97,0xFA,0xBD,0xF5,0xF0,0xBA,0xBB,0x20,0x7B,0xC8,0x1E,0x30,0xA9,0x96,
+ 0xC6,0xD9,0x95,0x6B,0xD1,0x4C,0xA7,0x46,0x58,0x3C,0x29,0x2B,0x8F,0x77,0x39,0xB5,
+ 0x9D,0x48,0x73,0x35,0x16,0x82,0xEC,0xBC,0x17,0x2C,0xA5,0x6B,0xA8,0x86,0x28,0x94,
+ 0x81,0x5F,0x47,0xC9,0xE1,0x7B,0x5B,0x26,0x42,0xC2,0xF9,0xF7,0x43,0xA4,0x97,0x37,
+ 0x2C,0x36,0x82,0x23,0xAD,0x85,0x4B,0xE4,0xBC,0x2B,0xAA,0xCB,0xD4,0xD5,0xFF,0x78,
+ 0xCA,0xD9,0x19,0x76,0xEA,0x63,0x81,0x1B,0x24,0xC4,0xD8,0x06,0x61,0xEF,0x4F,0x8B,
+ 0x3D,0x7A,0x28,0x33,0x5D,0x83,0x8F,0xB1,0x8E,0x57,0x10,0xAB,0x84,0x2A,0x48,0xC8,
+ 0x8C,0x95,0xE2,0x4B,0x67,0x87,0xD8,0x17,0x5A,0x06,0xAB,0xF9,0xD4,0xC8,0xBB,0xAC,
+ 0xF2,0xBA,0xD2,0x71,0x23,0x3B,0xDA,0x82,0xE0,0x0F,0x40,0x6E,0x8D,0x84,0x7D,0x6C,
+ 0xA8,0xFE,0xFC,0x3B,0xE1,0xA4,0x66,0x2A,0x8F,0x46,0xDC,0x89,0x6B,0x3A,0x15,0xEB,
+ 0xB1,0x3F,0x09,0x8C,0x2F,0xE7,0x74,0x8C,0x13,0x4B,0x5B,0xA7,0xCC,0x72,0xF1,0x26,
+ 0xA9,0x32,0xF6,0x9C,0xC2,0xF6,0x57,0x6A,0xEB,0x42,0xAD,0x31,0xF2,0xD8,0x02,0x2D,
+ 0x13,0xA4,0x1C,0xE4,0x02,0x08,0x5D,0x89,0x12,0xCF,0x7E,0xF8,0xFA,0xB1,0xBB,0x79,
+ 0xA0,0x8A,0x24,0x7C,0xB9,0xCE,0x07,0x3D,0x6E,0x85,0x53,0xCC,0x66,0xC9,0xAE,0xC0,
+ 0x94,0x75,0xAB,0xFF,0xE3,0x2D,0x2C,0xC9,0xC9,0x37,0x17,0x17,0xDF,0x4D,0xED,0x56,
+ 0xE3,0x6E,0xF4,0xF9,0x24,0xDC,0x42,0x99,0xCD,0x6A,0x61,0x4F,0x54,0x9E,0xCC,0x98,
+ 0x91,0x73,0xD4,0x7F,0x22,0xB6,0xB0,0x40,0x09,0x18,0x10,0x60,0x9A,0x3C,0xFB,0x53,
+ 0x1A,0x0A,0x4A,0x78,0xA8,0x91,0x64,0x6F,0x2F,0x5F,0x44,0xD8,0x27,0x87,0xB7,0x51,
+ 0x43,0xE5,0x00,0xAC,0xC2,0x79,0x33,0xC9,0xC1,0xD1,0xD8,0x37,0x85,0xDC,0x80,0xFD,
+ 0xF0,0xB2,0x89,0x93,0xE0,0x71,0x5B,0x72,0x3C,0xD3,0xBB,0xE4,0xA5,0xCD,0x13,0x7C,
+ 0x90,0x5D,0x17,0x81,0x8D,0xD4,0x50,0x26,0x4D,0x16,0x19,0xB8,0x47,0x93,0x14,0x24,
+ 0xBB,0x13,0xA5,0x30,0xEA,0xCA,0xCA,0x24,0x5F,0x5F,0x12,0xB7,0x1B,0xD4,0x1C,0xCA,
+ 0x3F,0x2B,0x37,0x2F,0x05,0x80,0x4C,0xEA,0x5C,0x74,0x59,0xA7,0x44,0xFE,0x3C,0x32,
+ 0xE7,0xDF,0x3F,0xAC,0x48,0x72,0x43,0x7A,0x73,0xCB,0x3B,0x45,0x9C,0x08,0xA0,0xB2,
+ 0xD1,0x99,0xE6,0x7E,0xEE,0x86,0xF3,0x8F,0xB9,0x9C,0xA7,0xC6,0x69,0xBF,0xD6,0x1C,
+ 0x0F,0xDF,0x50,0x96,0x82,0x69,0x28,0x13,0x28,0xA0,0x8F,0xB1,0x9D,0x65,0x0C,0xC9,
+ 0xC5,0xE2,0xA8,0x6D,0x3C,0xA3,0xA5,0x07,0x8C,0xDF,0x66,0xAE,0xF2,0x06,0x64,0x66,
+ 0xC8,0x1C,0xB8,0x9B,0x09,0xC5,0xA2,0x95,0x3B,0x14,0x5B,0x58,0x42,0x17,0x23,0x5A,
+ 0x2E,0x1E,0x27,0x22,0xD7,0x72,0x92,0x15,0x2F,0x53,0xAA,0x73,0xCE,0x8C,0xB3,0xBD,
+ 0xA7,0x4A,0x6C,0x46,0x2F,0x22,0x6B,0x61,0xAF,0x2B,0xD8,0xAB,0x51,0x8A,0x9A,0x84,
+ 0x48,0x30,0xFF,0x84,0x09,0xEA,0x00,0x27,0xC6,0x24,0x7D,0xC2,0x4C,0x32,0x23,0xCF,
+ 0x59,0x01,0x67,0xA3,0x7A,0x80,0x09,0x69,0x7E,0xEC,0x42,0x51,0x09,0xD2,0xEC,0xAC,
+ 0x8D,0x67,0x7E,0x1D,0xC4,0x1E,0xFE,0x80,0x83,0x94,0xFC,0x25,0x8D,0x84,0x55,0x97,
+ 0xC8,0x4A,0xD5,0x10,0x73,0x10,0xAF,0x74,0x51,0x27,0xA4,0x4C,0x2E,0x42,0x4F,0x69,
+ 0xFC,0x02,0xEA,0x03,0x66,0xDC,0x19,0xC3,0x5D,0x12,0x70,0xED,0x69,0x57,0x91,0xCB,
+ 0xCB,0xE3,0xFC,0x09,0x31,0x68,0x1A,0x34,0xDE,0xEB,0x9D,0x20,0x9E,0x26,0xED,0x13,
+ 0x2E,0xB8,0x62,0xCF,0x8B,0x93,0xB9,0x2F,0xDF,0xC4,0x73,0x5E,0x1C,0x08,0xAC,0x32,
+ 0x5F,0xA1,0x56,0xE9,0xBE,0x0C,0x5D,0xD3,0x6F,0x96,0x24,0x40,0xBA,0x37,0xAE,0x10,
+ 0x89,0xE5,0x3F,0x51,0x2A,0xC9,0x34,0xB2,0xA9,0x23,0xAE,0xE1,0x5B,0xA4,0xF2,0xFD,
+ 0x9F,0xDA,0x22,0xFF,0x37,0x2F,0x4A,0x96,0x24,0x20,0xCE,0xA6,0xBD,0x42,0x79,0x5D,
+ 0xE5,0x55,0xFC,0xDB,0x9D,0x83,0xDD,0xB9,0xC7,0xC9,0xDE,0xEF,0x94,0x02,0xC6,0xDD,
+ 0xFC,0x71,0x22,0x73,0xD2,0x71,0xA7,0xFF,0x5C,0x0A,0x66,0xDE,0xD2,0x8E,0x35,0x3E,
+ 0xD6,0x8A,0x26,0x94,0x6D,0x44,0x41,0xF3,0x1D,0x7F,0x6E,0xFC,0x45,0x60,0x98,0xB5,
+ 0x4A,0x47,0x03,0xF1,0x65,0x9D,0x7C,0x36,0x36,0xF8,0xE9,0x2F,0x67,0x1A,0x73,0xE6,
+ 0x11,0x58,0x9D,0xC6,0x85,0x91,0x3A,0xC1,0x3F,0x95,0xE1,0x34,0xE7,0x84,0x1B,0x8A,
+ 0x47,0x05,0xCC,0x4E,0xAA,0xA3,0x89,0x6B,0x17,0xC0,0xB3,0xB2,0x99,0x2B,0xE4,0x71,
+ 0x50,0xA1,0x9A,0x99,0x32,0x00,0x63,0xBB,0x96,0x80,0x9E,0xE6,0x67,0x31,0x26,0x83,
+ 0x1D,0x7E,0x5A,0x07,0x2E,0x1C,0x79,0x8E,0x0B,0x6C,0x8A,0x86,0x8B,0x22,0x71,0x68,
+ 0xBD,0x54,0x36,0xD9,0x2E,0xF4,0xF3,0xAE,0xEF,0x9C,0x8A,0x63,0x63,0x65,0x74,0xB1,
+ 0x4D,0xDD,0xE1,0xFA,0x38,0x49,0x5A,0xF9,0x89,0x9C,0x85,0xF7,0xB8,0x19,0xA9,0x78,
+ 0xEF,0x68,0x9E,0xAE,0x49,0xC8,0x8A,0x18,0xC6,0xAF,0xE0,0x40,0xCA,0xC8,0x1E,0x24,
+ 0x55,0xBE,0x9C,0x80,0x18,0x7D,0x08,0xBA,0x51,0x79,0x68,0x5C,0x78,0x0C,0xC7,0x83,
+ 0x62,0x8B,0xCC,0x2D,0x01,0xA0,0xC5,0xE3,0x30,0xE5,0xDB,0x3F,0x53,0xD8,0x76,0xE6,
+ 0x1C,0x27,0x19,0xE0,0x52,0xEE,0x2D,0x4C,0x7D,0x30,0x88,0xEB,0x56,0x72,0xCA,0xD5,
+ 0xA8,0xD5,0xDF,0x54,0x00,0x7C,0x19,0x44,0x2A,0xB6,0xA4,0x10,0x48,0x3E,0xDB,0xF6,
+ 0xA0,0x90,0xFB,0x75,0xF5,0x36,0xF1,0x8B,0xC8,0xE5,0x89,0x25,0x8A,0x63,0xA0,0xF1,
+ 0xB2,0xD3,0xEB,0x9C,0xF6,0x55,0xC0,0xE6,0x42,0x7C,0x93,0x9A,0xC5,0xE4,0x91,0x93,
+ 0xBB,0x60,0xA2,0x92,0x1A,0x6E,0x97,0x5C,0xD8,0x64,0x96,0xE6,0x87,0xFA,0x19,0x93,
+ 0xA9,0x03,0x38,0x50,0x2F,0xBF,0x22,0x81,0x3C,0xA9,0x45,0xE9,0xF7,0x28,0x27,0x3A,
+ 0x4A,0xAD,0xE9,0x57,0xA6,0xD1,0xD2,0x66,0x05,0xFE,0xC8,0x92,0x40,0x24,0x06,0x29,
+ 0x3A,0xBC,0x97,0x8F,0x9D,0x74,0x61,0xE6,0xBB,0x8D,0x8F,0xE7,0xA2,0xDC,0x0C,0xFC,
+ 0xD1,0xD3,0x3B,0x89,0x60,0x0F,0xA6,0x83,0xDF,0xB1,0xEE,0xE7,0x9B,0xA4,0xF6,0x74,
+ 0x1C,0x8A,0x36,0x3D,0x70,0xEC,0xE3,0xF5,0x25,0xE7,0x1A,0xD9,0x65,0x20,0x9A,0x52,
+ 0x8A,0x01,0x00,0xE2,0x52,0x56,0x1D,0xDB,0x22,0xF0,0x5A,0xE7,0x63,0xE4,0x52,0xF2,
+ 0x0C,0xFA,0xA8,0x5C,0xA1,0xBC,0xF0,0xA7,0xDF,0x29,0x97,0x25,0x92,0xD8,0x38,0x37,
+ 0xAE,0xE0,0xA4,0xD9,0x03,0x74,0x69,0x88,0xF5,0x53,0xED,0x05,0xE2,0x6B,0x8A,0x89,
+ 0x75,0x25,0xCC,0x91,0x3C,0x09,0x37,0xCC,0xF6,0x6A,0x1C,0xDE,0x5C,0xCD,0x48,0x7E,
+ 0x23,0xD8,0xEC,0xEB,0x68,0x8C,0x40,0xF0,0x12,0xAD,0x79,0x53,0x68,0xB7,0xB4,0x31,
+ 0xF6,0x79,0xCE,0x90,0x9D,0x06,0xC5,0xD4,0x37,0x2A,0x4D,0xD9,0xF8,0x71,0xBA,0x8B,
+ 0x7B,0xEB,0xDE,0x3F,0xD5,0x63,0x1C,0x8C,0xFF,0xB6,0xA1,0xF6,0xCD,0xB1,0xEA,0x5E,
+ 0x65,0xCF,0xEE,0xDA,0x59,0x72,0x69,0x7A,0x67,0x99,0x80,0x31,0x7E,0x88,0xCA,0x1B,
+ 0x9E,0x6E,0x41,0xBE,0x52,0xFA,0xC3,0x46,0x6D,0x1C,0x2F,0xA1,0xF8,0xC3,0x0A,0xF8,
+ 0x5E,0x01,0x2E,0x75,0xFD,0xA5,0x34,0xA5,0xEF,0xD7,0xB6,0x5A,0x91,0xC0,0x48,0x9E,
+ 0xE1,0xA6,0x18,0xC4,0xAF,0x34,0xF7,0x5E,0xBD,0xC0,0x10,0x8D,0xC6,0x81,0x37,0x9F,
+ 0x5C,0xE9,0x26,0x86,0x31,0x25,0xB3,0x4C,0x1B,0x84,0xBB,0xB6,0xBC,0xF2,0x2E,0x64,
+ 0xD4,0x68,0x21,0xC1,0x1A,0x21,0x13,0xBC,0x07,0xD0,0x91,0xC3,0x8A,0x42,0x3F,0xFF,
+ 0x66,0xCE,0x55,0x2B,0x34,0x4F,0x17,0x7E,0xCF,0x00,0x5E,0x6E,0xD3,0xB0,0xDC,0x3F,
+ 0x6B,0x33,0xD9,0x3B,0x9C,0x5F,0x4E,0x3E,0xB2,0x5B,0x8C,0x72,0xC2,0xCE,0xFF,0xF0,
+ 0xC8,0xAC,0xB3,0x6D,0x49,0x8D,0x13,0x7C,0x71,0xA9,0x69,0x9C,0x23,0xA5,0x6C,0xC4,
+ 0xD9,0x36,0x67,0x31,0xE0,0x30,0x9A,0x0F,0x4E,0x57,0x0F,0x49,0x0B,0x34,0xEC,0x7C,
+ 0xDD,0xFE,0xCA,0x9A,0x6F,0x64,0xB7,0x8E,0xB7,0x8E,0xD8,0xE1,0x58,0xE5,0xBB,0x05,
+ 0x27,0x4F,0x4A,0x16,0xD7,0x7D,0x51,0x37,0x00,0x68,0xAB,0x6B,0x5C,0xF0,0x72,0xAD,
+ 0x4F,0x4F,0x8F,0x59,0x9A,0x47,0xC0,0xAF,0xB2,0x6D,0x5F,0x13,0x20,0x91,0xFB,0x77,
+ 0xC6,0x8B,0xB3,0xD2,0x69,0xAF,0x44,0xA5,0x5D,0x0C,0x46,0x62,0x00,0xEA,0x2D,0x8F,
+ 0x7B,0x87,0x9C,0xDC,0x45,0xFB,0xCC,0x03,0xD6,0xFF,0xC6,0x79,0x10,0x4D,0x77,0x41,
+ 0xBF,0xA8,0x79,0xE0,0xFC,0xB4,0xAF,0x8C,0x64,0xFE,0x13,0x92,0x04,0x55,0x99,0x56,
+ 0xFB,0x5D,0xEE,0x5A,0x35,0x77,0xB0,0x96,0xC0,0x61,0xA8,0xB2,0xCA,0x7D,0x6C,0x84,
+ 0x26,0x70,0x91,0xA9,0x6F,0xC5,0xBE,0x80,0xA8,0x43,0x21,0x74,0x59,0xF6,0xE5,0xAA,
+ 0xB4,0x4C,0xC3,0x4B,0xE3,0xEB,0xE1,0xA9,0xFF,0xE2,0x3B,0x5C,0x35,0xA5,0xCB,0xF6,
+ 0x24,0x6B,0x3F,0x0F,0xED,0x94,0x93,0xDE,0x08,0x55,0xAC,0x20,0x1A,0x28,0xEE,0xFE,
+ 0xD7,0x77,0x3F,0x62,0x88,0x1F,0x19,0x65,0x66,0x30,0x64,0xB0,0x90,0xDD,0x50,0x46,
+ 0x88,0xFA,0xE0,0x7F,0x10,0xC8,0x84,0xC3,0xA0,0xC0,0x15,0x5B,0x20,0xA7,0x93,0x1D,
+ 0xA2,0xA3,0xFF,0x05,0x9C,0xB2,0xE8,0x59,0x3B,0xFC,0x57,0xF8,0xD3,0xF9,0xDD,0xFD,
+ 0xD9,0xCA,0x65,0xC2,0xC7,0xDE,0xD8,0xB5,0x8C,0xE3,0x5B,0x20,0x22,0x4B,0xC9,0x3D,
+ 0xE5,0x38,0x0B,0xC5,0x6B,0x08,0x43,0x6B,0x49,0xFF,0xF0,0x08,0x09,0xE4,0x71,0x53,
+ 0x00,0xD4,0x13,0x39,0x18,0xED,0x40,0x64,0x22,0x05,0x2B,0x86,0x4E,0x3C,0xC0,0xD7,
+ 0x1C,0xC0,0x88,0x2E,0x98,0xEF,0xC3,0x9B,0xF3,0xB6,0x6D,0x69,0xDF,0x65,0x89,0x09,
+ 0xFA,0xA9,0x49,0xE9,0xF9,0x04,0x3F,0x79,0xCF,0x7B,0x4F,0x23,0xB7,0x53,0xD8,0xDE,
+ 0xDC,0xE7,0xAF,0xE3,0x8B,0x0C,0x68,0x3B,0x60,0xD8,0xC5,0xFA,0x6D,0x54,0x69,0x7B,
+ 0xDC,0x34,0x84,0x99,0x0B,0xB9,0x41,0xBB,0x30,0x76,0x78,0xD4,0x68,0x08,0x35,0x8A,
+ 0xA0,0x90,0xBF,0xC8,0xF0,0x74,0xA4,0xAD,0x8D,0xA5,0xB2,0xAA,0x29,0x2C,0xF4,0x20,
+ 0x12,0x06,0x09,0x9A,0xB7,0x82,0xB6,0xA2,0x5F,0x6D,0xAA,0x23,0x44,0xBB,0x19,0x7C,
+ 0x01,0x7D,0xCC,0xEE,0x9F,0x8F,0x4F,0x3D,0x81,0x05,0x38,0xFF,0xD5,0x1C,0xF2,0x0E,
+ 0xBC,0x0F,0x76,0x87,0x2B,0xD3,0xD0,0x5C,0x5D,0x38,0x7B,0x57,0x63,0xA1,0x81,0xB7,
+ 0xCF,0x45,0xE4,0xA8,0xE9,0x63,0x1F,0xDD,0xC7,0xCB,0xFD,0x31,0x85,0x88,0xD8,0x1C,
+ 0x94,0x10,0x47,0xD2,0x38,0x12,0xD8,0xEC,0xBB,0x92,0x29,0x93,0x2C,0xD0,0x08,0xB8,
+ 0x2B,0x3E,0x5C,0x8D,0x5B,0xF5,0x20,0x97,0xF5,0x31,0xA6,0x16,0x88,0xE4,0x62,0xE0,
+ 0xC7,0x13,0x2B,0x33,0xE5,0xD3,0xB5,0x1A,0xAF,0x56,0x02,0xE6,0xA4,0xDF,0x67,0xC5,
+ 0x44,0x65,0xBD,0x50,0xF0,0x16,0xE8,0x7E,0x6A,0x20,0x4E,0xC8,0x28,0xD6,0x76,0x01,
+ 0x83,0x84,0xDE,0xC1,0xA1,0xF5,0xE2,0x3F,0xA1,0x93,0x25,0x4F,0x6D,0x4D,0x87,0xE5,
+ 0xDB,0x7D,0x36,0x93,0xF1,0xC8,0x45,0x6A,0x2B,0x37,0x81,0x75,0x44,0x74,0xD2,0xD0,
+ 0xC0,0x29,0x2F,0xBD,0xB8,0xA3,0xB8,0x72,0x0E,0x90,0xDE,0xFF,0xF2,0x32,0x0A,0x9F,
+ 0xBD,0x55,0xA7,0xB5,0xFC,0xED,0x6A,0x8B,0x77,0xB2,0x84,0x54,0x21,0xF7,0xC7,0x55,
+ 0x61,0x68,0x44,0x2A,0x77,0x49,0xA2,0x50,0xE7,0xFB,0x88,0x5B,0x09,0xC6,0x3B,0x3F,
+ 0x5F,0x81,0x35,0x64,0x71,0x31,0xF6,0xAF,0xF2,0x8A,0x12,0xC6,0x3C,0xDA,0x78,0x6C,
+ 0xFC,0x36,0x23,0xAF,0x54,0xBD,0x66,0x5E,0x93,0xA9,0xA3,0xE9,0x68,0x95,0x39,0x42,
+ 0xE0,0xBD,0xC5,0x67,0xAE,0x98,0x2E,0x38,0x91,0x6E,0x34,0xFD,0x5D,0xBD,0xC1,0xCB,
+ 0xE0,0x92,0x1B,0xE1,0x5A,0x91,0x64,0x76,0xF1,0x3B,0x0B,0xD7,0x64,0x30,0x1D,0x07,
+ 0x7D,0x7B,0x6F,0xAE,0x72,0x1B,0x57,0xB0,0xD2,0x7E,0xC8,0xC7,0x78,0x0C,0xD7,0xB5,
+ 0xB5,0xCE,0x62,0x15,0xA1,0x1C,0x7C,0x18,0xB5,0xA9,0x07,0x7B,0xEC,0xEF,0x3F,0xF5,
+ 0xB0,0xDF,0x50,0xB4,0xA9,0x67,0xD3,0xCC,0xD8,0x9D,0x92,0xE0,0x64,0xE1,0xFD,0xF1,
+ 0x89,0x81,0xCD,0xE1,0x0A,0x89,0xAD,0x54,0xBD,0x74,0x54,0xE3,0x65,0x34,0x32,0x5D,
+ 0x6E,0xAC,0xB3,0x68,0x39,0xDA,0x6E,0xB2,0xD3,0xB8,0x5D,0xB6,0x7A,0xBA,0xE1,0x38,
+ 0x9A,0xBD,0xD0,0x06,0xF9,0xEA,0x7F,0xA3,0x18,0xB6,0xDD,0xDE,0xEC,0x1A,0x8E,0x70,
+ 0x85,0xA1,0xC5,0x35,0xE8,0x7E,0x06,0x7F,0xEE,0x62,0x65,0x92,0xAF,0xC2,0x44,0x2A,
+ 0xA4,0x41,0x3D,0xAD,0x1F,0x37,0x24,0xAC,0x4D,0x6A,0x6E,0x99,0x3D,0xAA,0x4B,0xE2,
+ 0x3A,0x7A,0x2C,0x28,0x53,0x18,0x84,0x32,0x30,0x02,0x0B,0x05,0xF9,0xEF,0xD5,0x25,
+ 0x6A,0xA5,0x1A,0x04,0x18,0xEB,0x1B,0xAF,0x12,0x15,0x01,0x8C,0x44,0x44,0xA5,0xB7,
+ 0x66,0xDE,0x59,0x41,0x1F,0x9A,0x2D,0x7D,0xEB,0xDF,0x17,0x81,0xC7,0x3A,0x20,0xA1,
+ 0x33,0xCB,0x57,0xC9,0x72,0xDD,0xB9,0x76,0x3A,0x90,0x05,0xDC,0x83,0xCF,0x04,0x39,
+ 0x1C,0x5E,0x13,0x58,0x70,0x32,0x66,0x0D,0x36,0xEE,0x75,0x42,0x4B,0xFF,0xE7,0x5D,
+ 0x1B,0x38,0x7D,0xD8,0x2A,0xFC,0x83,0xF4,0xD2,0x9B,0xF9,0x2C,0xE2,0x2E,0xF8,0x60,
+ 0x2C,0xAE,0xFC,0xF9,0x48,0x80,0xC0,0xC7,0x80,0xC6,0xCE,0xBA,0x12,0xDC,0xCD,0x14,
+ 0xD2,0xF6,0xB0,0x16,0x2F,0x4A,0xBB,0x30,0xDA,0x32,0xBF,0x01,0x6F,0x6E,0x8B,0x09,
+ 0x01,0xD2,0x28,0x9E,0xEA,0xF3,0xCA,0xB6,0xBC,0x5A,0x66,0x93,0xF0,0x62,0xA5,0x62,
+ 0x8F,0x20,0x63,0x06,0x42,0x06,0x1E,0xD0,0x97,0x6C,0x11,0xA0,0x36,0x0F,0xCB,0xC7,
+ 0xCA,0x15,0x88,0xE3,0x50,0x77,0xDC,0xAA,0x81,0x57,0xFD,0x70,0x88,0xF2,0x45,0x89,
+ 0x41,0x2F,0x6C,0x2F,0x92,0x03,0x31,0x29,0x93,0x17,0x51,0xFD,0x15,0x49,0xC4,0xEE,
+ 0xCA,0xC8,0x62,0xE1,0xCC,0x08,0x12,0x37,0x3E,0x84,0x11,0x66,0x01,0x7F,0xE5,0x21,
+ 0x92,0x79,0xF0,0x56,0xAA,0x2B,0xFA,0xAE,0x16,0xA0,0xAB,0xFC,0x97,0x8E,0x50,0xEE,
+ 0x49,0xDD,0x8E,0xC1,0x07,0x2B,0xEB,0x14,0x80,0xFF,0x0B,0x35,0x40,0xC9,0x5E,0x2F,
+ 0x98,0x0D,0xBA,0x72,0x3B,0x3B,0xB1,0x50,0x79,0x9F,0x88,0x2A,0x7D,0x35,0x7B,0xBF,
+ 0x9A,0x98,0xE1,0xFE,0xF0,0xFF,0x5D,0xE5,0xB0,0xB7,0x75,0x89,0xA4,0x05,0xDC,0xE2,
+ 0xD0,0xC3,0xCA,0xC1,0x08,0x80,0x0D,0x6F,0x0A,0xBB,0x72,0x8A,0x8D,0x01,0xEB,0x37,
+ 0x8E,0xA6,0x90,0x30,0x48,0xE6,0x52,0x54,0x8F,0xF4,0x5C,0x22,0xF8,0x58,0xEC,0xCC,
+ 0x45,0x40,0xDD,0x80,0x10,0xC5,0xB0,0x35,0xB4,0xFB,0x69,0xF9,0x36,0x29,0xFD,0x95,
+ 0xC4,0x5B,0xCB,0x13,0x8A,0x1D,0xD7,0xE9,0x76,0x08,0x69,0x73,0x63,0xF7,0xAE,0x1B,
+ 0x42,0x8C,0x0F,0xC1,0xA0,0x7A,0xEE,0x1D,0xC0,0xD6,0x32,0xC7,0x2E,0xDA,0xC4,0x6F,
+ 0x29,0x3B,0xB5,0xC8,0x47,0xDB,0x7D,0x7D,0xA9,0x36,0x4D,0x3B,0x89,0x09,0x05,0x60,
+ 0x1F,0x50,0xD7,0x52,0x99,0xEA,0x16,0x45,0x64,0xAE,0x49,0xF4,0xE4,0x76,0xB3,0xB5,
+ 0x50,0xFB,0x52,0x05,0x0F,0xA5,0x5B,0x7D,0x5A,0xC3,0x42,0x36,0xAD,0xCB,0x65,0xAE,
+ 0xBC,0x1E,0x46,0x8F,0xAB,0x14,0x3F,0x30,0x7A,0xA2,0x28,0x7C,0xA9,0x36,0xCE,0xAB,
+ 0x8F,0x62,0x6B,0xCC,0xD2,0x28,0x0C,0x78,0x9F,0x3C,0x01,0xCB,0x9B,0xBA,0x47,0xA7,
+ 0xF5,0x84,0x8C,0xE6,0xC0,0x00,0xFC,0x08,0x49,0x59,0x4B,0x17,0x33,0xB0,0xD3,0xBA,
+ 0xC3,0x0B,0xD8,0xCF,0x7B,0x70,0x7A,0xE3,0x6B,0xD8,0xCC,0xD5,0xFA,0xA6,0x99,0x3A,
+ 0x06,0xF5,0x86,0xE0,0xE6,0xA5,0x16,0x92,0x09,0x6E,0x6B,0xBA,0x59,0xD2,0x74,0x2A,
+ 0x5B,0x27,0x81,0x64,0x60,0x8D,0x9F,0x34,0xAC,0x23,0x4A,0x52,0xFF,0xEB,0x98,0x9A,
+ 0x88,0x06,0x6D,0xB7,0x23,0x05,0xE6,0xB6,0x4C,0x73,0x24,0x53,0x5E,0xB7,0xEF,0x87,
+ 0x00,0xB1,0xB6,0xE1,0x4E,0x41,0xB1,0x2C,0x1D,0xF2,0x68,0x54,0x55,0xA3,0xE8,0x52,
+ 0x9C,0x43,0x98,0x25,0x77,0x42,0xDD,0x8B,0x78,0x55,0x40,0x38,0x69,0xDB,0x2E,0xAE,
+ 0x3F,0xDC,0x5F,0xDE,0xEF,0x49,0x23,0xB3,0x05,0xE5,0x64,0x53,0x5A,0xA1,0xCD,0xD2,
+ 0xCC,0xDB,0x53,0x0A,0x98,0x99,0x01,0x06,0x48,0xFD,0x2F,0x97,0x44,0xED,0xEC,0x53,
+ 0xF5,0x16,0x44,0xAB,0xE2,0xC6,0x2E,0xF9,0x0B,0x84,0x40,0x0B,0xCC,0xAE,0x1A,0xA1,
+ 0x25,0xDC,0xC0,0xEA,0x1A,0x31,0x30,0x60,0x94,0x73,0x87,0xEA,0xE7,0x5F,0xCC,0x5F,
+ 0xF9,0x21,0xAB,0x65,0x0C,0xB5,0x02,0xF0,0x9B,0x9B,0xDA,0x06,0x40,0x91,0x7A,0xFD,
+ 0xED,0xBF,0xFB,0xD6,0x98,0xF7,0x97,0x6E,0x40,0xF9,0x83,0x33,0xF1,0xD4,0x50,0xAB,
+ 0x54,0x15,0x12,0x42,0x5C,0xDB,0x31,0x4E,0x6D,0x5C,0xA7,0x06,0x7C,0xD7,0xA6,0x40,
+ 0xF0,0xF4,0x14,0x8E,0xF3,0x0E,0x56,0x75,0x20,0x40,0x1C,0xA8,0xCF,0x75,0x46,0x83,
+ 0x8A,0xB4,0xF1,0xBA,0x5C,0x7D,0x7B,0x91,0x4F,0x1E,0xE8,0x88,0x84,0xC5,0x04,0x55,
+ 0x6B,0x82,0x35,0x03,0xD3,0x23,0x2F,0x9C,0x5C,0x55,0x13,0x83,0xD1,0x2C,0xB5,0x31,
+ 0x07,0x0D,0x6F,0x1E,0x11,0x24,0x3E,0x9F,0x45,0x34,0xB2,0xCB,0xAA,0x12,0xFD,0x06,
+ 0x2C,0x31,0x2A,0x8D,0x7E,0x02,0xE6,0xAF,0x3B,0x93,0x89,0x50,0x98,0xA3,0x01,0xD4,
+ 0x07,0x75,0x65,0x87,0x0B,0x5F,0xC6,0x83,0x47,0x5A,0x7A,0x87,0x0C,0xCC,0x37,0x10,
+ 0x27,0xF2,0x42,0xD8,0xBA,0x58,0x62,0xF8,0xBA,0x14,0x14,0xD3,0x41,0x20,0xB8,0x40,
+ 0xD3,0xB8,0x2E,0x1E,0x53,0x42,0xB4,0x40,0xD0,0x99,0xF4,0xAC,0xD8,0xF3,0x6C,0x85,
+ 0x27,0xE9,0x56,0xDA,0xF2,0x4E,0x9F,0xA4,0x3B,0xCE,0xA1,0x7A,0x2D,0x18,0x2D,0x1B,
+ 0x48,0x70,0xBE,0x17,0x10,0x47,0xB0,0x0F,0x89,0x8A,0x58,0xAB,0x75,0x5A,0xE8,0x4F,
+ 0xB3,0xAC,0x28,0x13,0x31,0x63,0x09,0x7D,0xE8,0xDB,0xF5,0x85,0xEE,0xC1,0x6F,0x08,
+ 0xD7,0x51,0x5C,0xD6,0xC6,0x4C,0x0A,0x40,0xEA,0x53,0x46,0x9B,0xE2,0x81,0xA9,0x9D,
+ 0x66,0x80,0x32,0x7A,0x83,0x5A,0x7C,0xCC,0xEE,0xAA,0x5F,0xA5,0x0C,0x35,0x3D,0x20,
+ 0xFC,0xD7,0x54,0xE9,0x58,0x19,0x81,0xFB,0xE0,0x47,0x30,0xA8,0x95,0xE1,0x74,0x37,
+ 0xFE,0x37,0xF0,0x63,0x7A,0x2E,0x74,0xF3,0x3A,0x77,0xE6,0xA1,0xCA,0x16,0x07,0xE8,
+ 0x94,0x9E,0xFE,0xA5,0x25,0x5D,0x08,0x02,0xEE,0x64,0x43,0x02,0x51,0xBB,0xD4,0xF2,
+ 0xD2,0xBF,0x80,0x65,0xA0,0xC5,0x55,0x23,0x1E,0xD5,0x2C,0x06,0x95,0x9A,0xFE,0xCD,
+ 0xDA,0x3B,0x9D,0x34,0xFB,0xF8,0xEF,0x78,0x7B,0xF4,0x46,0xEE,0xE5,0xC9,0xA8,0x90,
+ 0xA5,0x38,0xEC,0x21,0x46,0x95,0x8C,0xE7,0x54,0xE9,0x93,0xF4,0x44,0x38,0x04,0x6E,
+ 0x01,0x7D,0xB9,0xA1,0xBB,0x49,0x3B,0x11,0xDD,0x0D,0x54,0x90,0xCD,0x9C,0xBF,0xBE,
+ 0xC0,0x98,0x6A,0xC7,0x30,0x0B,0x30,0xD6,0x19,0x0C,0x02,0x1C,0x24,0xC4,0xAF,0x7C,
+ 0x10,0xF8,0x67,0x07,0x2E,0x84,0x7F,0x0A,0x25,0xE0,0x1C,0x5E,0x4F,0x51,0x9F,0x66,
+ 0x99,0x3C,0x93,0x55,0x3F,0x47,0x98,0x7A,0x99,0x2D,0x15,0x0C,0x22,0xD4,0x9A,0x82,
+ 0x4F,0x43,0x3B,0x49,0x6A,0x83,0x2A,0x36,0xEE,0xAD,0xDE,0xEE,0x75,0xEC,0xF5,0x7E,
+ 0xF4,0xA8,0xEB,0xD1,0x70,0x51,0x47,0xB6,0x4C,0xEA,0x66,0x88,0x82,0x02,0x19,0xF5,
+ 0x53,0xAF,0xC4,0x41,0xC1,0x89,0x72,0x42,0x2C,0xB2,0x32,0xCC,0x09,0x12,0x0B,0xA9,
+ 0x7F,0x8A,0x97,0x6C,0xB3,0x58,0xA5,0x2D,0xE4,0xE8,0x91,0xD6,0xFD,0xB7,0xC9,0x7E,
+ 0x32,0x3B,0x8A,0x9F,0x93,0xFA,0xBA,0x4A,0x26,0x2A,0x0E,0x83,0xDD,0x4E,0x55,0x12,
+ 0xA8,0xC7,0x73,0xCB,0xBC,0x78,0xD3,0xB5,0x7C,0x73,0x77,0x1F,0x6F,0x9E,0xB7,0xDA,
+ 0x7F,0xBA,0xD9,0x91,0xE9,0x52,0x63,0x1F,0x36,0xC5,0x5D,0x2C,0x18,0x7A,0x7E,0xF4,
+ 0x1A,0x16,0xBA,0x79,0xEF,0xE2,0x20,0x03,0xEA,0x20,0xED,0x65,0x73,0x90,0x11,0x2F,
+ 0xD7,0x92,0x90,0x7C,0xE4,0xA1,0xA8,0x6A,0x24,0x6D,0x66,0x6D,0xDC,0xB2,0x61,0xDC,
+ 0xCD,0xA5,0xB5,0xFF,0x2F,0x3A,0xF2,0xF0,0x22,0x02,0x2B,0x84,0xB1,0x49,0xA1,0xE7,
+ 0xC8,0x7C,0x85,0x28,0xB8,0xCD,0x00,0x17,0x22,0x3E,0x23,0x44,0x8E,0xCA,0xD3,0xDC,
+ 0x77,0x5A,0xA7,0x6D,0xDF,0x67,0xFB,0x41,0x33,0x92,0x11,0xB3,0x87,0x02,0xA4,0x89,
+ 0x09,0xF6,0xBF,0xE0,0x81,0x99,0x07,0xF9,0x3D,0x18,0x5A,0xE2,0x7C,0x5D,0xBD,0x16,
+ 0xE8,0xA4,0x92,0x2A,0xAC,0x3A,0xFE,0xB0,0x75,0x17,0xD5,0x48,0x9A,0xCA,0x81,0x9F,
+ 0x27,0x06,0x19,0x81,0xE3,0x6C,0xCA,0x12,0x0C,0x6D,0xC9,0x88,0xFF,0x10,0x55,0x77,
+ 0x1A,0x49,0xE7,0xEB,0xFB,0x6D,0xD8,0x6C,0x31,0x58,0x8D,0x13,0x7A,0xFE,0x3C,0x39,
+ 0x90,0x1D,0x59,0x51,0x62,0xBE,0xA2,0xEC,0xE9,0x05,0x65,0x95,0xF9,0xBC,0x2C,0x2B,
+ 0x8F,0xA6,0x98,0xA4,0x99,0x03,0xB9,0x7C,0xAC,0x50,0x69,0x3B,0x8D,0x3E,0x96,0x60,
+ 0x47,0xE9,0x2C,0x96,0xAB,0x35,0x68,0xBC,0xA0,0x1B,0xB1,0x1C,0x93,0x9A,0x8D,0xFE,
+ 0x9A,0x48,0x6B,0xC2,0x7C,0xA7,0xE9,0xC0,0x37,0x53,0xBA,0xCB,0xF2,0x3D,0x7C,0xC1,
+ 0xD6,0xAE,0xAD,0x66,0x69,0x96,0xB9,0x3B,0x7D,0x9F,0x72,0x11,0xC9,0xC0,0xA7,0x9C,
+ 0xCD,0xA1,0x06,0x71,0xB5,0x19,0x7D,0x76,0x3E,0xAD,0x87,0xDB,0x63,0xCD,0x00,0x86,
+ 0x6B,0x71,0x62,0x5E,0xB5,0x78,0xB6,0x1E,0x22,0xAA,0x2B,0x23,0x37,0xBF,0xCC,0xEC,
+ 0x90,0x08,0x9C,0xBE,0xBD,0x4B,0x8D,0xC3,0x58,0xF5,0xA6,0x6C,0x99,0xC0,0xCE,0x61,
+ 0x34,0xBE,0x6F,0xB1,0x47,0xA8,0x12,0x2A,0x67,0x6D,0x37,0x64,0x37,0xA2,0x0D,0xD7,
+ 0x40,0xF8,0x9B,0xC0,0xEA,0x94,0x45,0xD1,0xD6,0x88,0xFF,0xD9,0x0A,0x91,0xC3,0xE6,
+ 0xFF,0xD6,0xF6,0x4D,0x20,0xFD,0xFF,0x63,0x54,0xBC,0x3E,0x2C,0x09,0x04,0x4D,0xE3,
+ 0xC8,0xD7,0xCA,0x44,0x47,0x99,0x24,0x63,0xAB,0xFA,0x8D,0x3B,0xFF,0x3F,0xE2,0x91,
+ 0xC8,0xEA,0xF2,0xCC,0x25,0xF7,0xB2,0xEE,0xCB,0xD8,0x84,0x8A,0xA4,0xF9,0x16,0x6B,
+ 0xF7,0xC0,0xA7,0x0A,0xAD,0xD7,0x21,0x2C,0x99,0xB5,0x0D,0x3D,0xA3,0xEC,0x3C,0xB1,
+ 0xD9,0x27,0xC5,0x1E,0x09,0xB8,0xCB,0x3B,0x6E,0x6A,0xBF,0xCE,0xF3,0x8E,0x5C,0xBF,
+ 0xCC,0xBD,0xFB,0x8E,0x5B,0x0E,0xA3,0x23,0xED,0x91,0x40,0x03,0x47,0x75,0x7E,0xF3,
+ 0x54,0x8A,0x10,0xD7,0x91,0x2B,0x59,0x88,0x8B,0xF7,0x27,0xB2,0x07,0xFE,0x3C,0xB3,
+ 0x72,0xFA,0x56,0xE0,0x52,0x6B,0x4E,0x7E,0x5F,0xF8,0xCB,0x04,0x7D,0x16,0xA9,0x24,
+ 0x6F,0xDA,0x0B,0x20,0x39,0x4B,0x3E,0x63,0xF5,0xF7,0xDB,0xCA,0xF9,0xE9,0x1F,0xE7,
+ 0xB4,0xE1,0xAA,0xF6,0xD0,0x39,0x21,0x43,0x15,0x71,0x6D,0xB0,0x0D,0x74,0x8A,0xE4,
+ 0xCC,0xB2,0xCD,0x36,0x20,0x7C,0xF2,0x4E,0x22,0x0F,0xBF,0x79,0xD1,0x89,0x1A,0xC6,
+ 0x95,0x0E,0xA5,0xF4,0x37,0x51,0x9B,0xF2,0xCA,0xD6,0xDE,0x25,0xD8,0x24,0x57,0x1E,
+ 0xF4,0x2B,0xBE,0x68,0x7D,0xE4,0xB2,0xC7,0x61,0x0E,0x67,0xD4,0x8C,0x3B,0xD8,0x9D,
+ 0x24,0xAD,0x14,0x55,0xF8,0x4A,0x9B,0xB9,0x4A,0x7C,0x11,0xA5,0x6F,0x1B,0xB2,0x94,
+ 0x56,0x18,0xC7,0x1C,0xB2,0xF9,0xBA,0x36,0xDF,0xB6,0x3A,0x84,0xB0,0x7E,0xF0,0xAB,
+ 0x6D,0xFD,0x7C,0x60,0x62,0xEE,0x2A,0x0C,0x89,0xF8,0x5E,0xDE,0xA6,0xDF,0x86,0x75,
+ 0x18,0x03,0x84,0xDE,0x77,0x25,0x85,0x57,0xAE,0x11,0x88,0xE6,0xC3,0x18,0x0D,0xE2,
+ 0x9F,0x58,0x89,0x20,0xB5,0xAC,0x17,0xD6,0x67,0x36,0xE3,0x11,0x92,0xF8,0xA2,0xE2,
+ 0x78,0xDE,0x63,0x06,0x67,0x0D,0x24,0x4D,0x6F,0x84,0x10,0x79,0xBD,0x68,0x11,0x3D,
+ 0xA9,0xA7,0x20,0x5B,0x39,0xCE,0xEE,0xDF,0x80,0xDC,0xC8,0xB8,0xF1,0xFA,0x01,0xA2,
+ 0x62,0x01,0xB2,0x5D,0x1E,0xEF,0x07,0x00,0x0E,0xA3,0x3A,0x8F,0x69,0xDF,0x36,0xF8,
+ 0x1F,0x55,0x34,0x30,0x09,0x22,0x7E,0x7C,0x83,0xFC,0xDA,0x7A,0x36,0x1A,0xEF,0x70,
+ 0x33,0xC6,0xF5,0x00,0x0F,0x01,0x49,0x7E,0xA0,0xF9,0x26,0x9A,0xD9,0x6E,0x19,0xC6,
+ 0x16,0x3C,0xB0,0xC2,0x4C,0xBF,0xDA,0xAD,0xDC,0xE3,0x93,0x87,0x9D,0xE6,0x11,0x8A,
+ 0x86,0xCA,0x03,0x6A,0x83,0x4E,0x86,0x6F,0x00,0x76,0xE6,0x9D,0xCF,0xF3,0x24,0x5B,
+ 0x04,0xD8,0xCE,0xD5,0xF2,0x1A,0x49,0x66,0xB3,0xD0,0x9F,0x25,0x79,0x4B,0x2C,0xF5,
+ 0x36,0x85,0xAB,0x80,0x5F,0x27,0x36,0xBD,0x84,0xD0,0x19,0xA1,0xD2,0x92,0xB5,0x98,
+ 0x62,0x5F,0x9A,0x32,0xC8,0x2A,0xEC,0x6A,0x5D,0x72,0x75,0x73,0x56,0x99,0xE8,0x77,
+ 0x15,0x9C,0x7D,0xA0,0x7A,0x18,0x3D,0x55,0x1F,0x66,0xFC,0xA6,0x74,0x47,0xE6,0xA2,
+ 0x10,0xBD,0xC3,0xC3,0x8F,0x85,0xE1,0xFB,0xA8,0x0C,0x61,0x78,0xA3,0x53,0x4D,0xF7,
+ 0x75,0x3E,0xAB,0x8B,0x2D,0x3B,0x1B,0x2C,0x40,0xE6,0xB5,0xFC,0x3C,0x64,0x22,0xF6,
+ 0x3B,0xB8,0x65,0x93,0x5A,0xE9,0xD6,0x95,0xAD,0xCB,0xC6,0x8E,0xD5,0x27,0xBC,0xD3,
+ 0xEC,0x65,0x61,0xE8,0xF7,0x68,0x91,0x61,0x24,0xB2,0x22,0xF3,0x64,0xE4,0xF8,0x8C,
+ 0x24,0x0E,0xE4,0xC4,0x4A,0xAD,0xCC,0x6B,0x5D,0xA5,0x42,0x46,0x3E,0xDF,0x3D,0xEE,
+ 0x8D,0x2E,0x22,0xEC,0x41,0x52,0xAE,0xB1,0xFF,0xE4,0xF2,0xE7,0xF2,0xC4,0xCF,0x4D,
+ 0x3D,0x76,0x3A,0xF0,0xC5,0x89,0x7F,0xFF,0xCB,0xAC,0x7F,0xCD,0xB3,0x60,0x61,0x18,
+ 0xDF,0x0B,0x8B,0x45,0x05,0x89,0xB6,0x6A,0x4E,0x23,0x10,0xF4,0xD2,0x75,0xF0,0x25,
+ 0xB0,0x7F,0x4E,0xF2,0x68,0x44,0x7E,0x34,0x9C,0xE5,0x02,0xC4,0xA0,0xB2,0x65,0x65,
+ 0xA9,0xD5,0xD4,0xE9,0xAE,0xC4,0x84,0x07,0x4D,0x76,0x85,0x44,0xEF,0x75,0xD3,0xCF,
+ 0x8E,0x5A,0xF9,0x5B,0x71,0x9E,0x31,0xEC,0xBC,0xFE,0x7B,0xD8,0x7E,0x32,0xE0,0xB0,
+ 0x44,0x72,0x4A,0xFD,0x6D,0xFA,0xBE,0xAC,0x67,0x59,0x53,0x87,0xBF,0x43,0x71,0xCE,
+ 0x49,0x09,0xD9,0x0D,0x99,0xE4,0xE6,0xDC,0x55,0xFA,0xF4,0x43,0x9C,0x00,0x36,0x0D,
+ 0xF5,0xE4,0x28,0xD5,0x98,0x36,0x6A,0x20,0xB0,0x81,0x7C,0x39,0x6A,0x9F,0x09,0xBD,
+ 0x13,0xC0,0x3B,0x6A,0xD4,0xF8,0x79,0x3A,0x53,0x6F,0xBF,0xC9,0x1F,0xE5,0x3C,0x70,
+ 0x35,0xD1,0x31,0x61,0x5C,0xA8,0x99,0x62,0xA2,0x6B,0x83,0xB6,0x7B,0x01,0x93,0x88,
+ 0x65,0x4C,0x59,0x95,0xEB,0x68,0x45,0xCD,0xDC,0xEC,0x8F,0x6B,0x6A,0xF0,0x79,0xF4,
+ 0xC1,0xD8,0x68,0xE6,0x93,0x60,0xC0,0xA8,0xE8,0xE6,0xE0,0x12,0x51,0xDC,0x88,0xDF,
+ 0x1B,0xE8,0xE2,0xFB,0x19,0x91,0xD4,0xE1,0xB8,0x6F,0x27,0x17,0x84,0xB4,0x78,0x4E,
+ 0x9D,0x7A,0x0A,0x78,0x52,0x4A,0x4D,0x03,0x87,0xAF,0x94,0xF8,0xF8,0xA7,0x24,0x12,
+ 0xD7,0xF0,0xF2,0xA5,0x56,0xF0,0x24,0xC6,0x48,0x40,0x29,0x3A,0xE2,0x8F,0x0D,0x60,
+ 0x0A,0xA8,0x0D,0x23,0x74,0xC9,0x03,0x1B,0xD3,0x37,0x15,0x04,0x61,0xE8,0x2F,0x05,
+ 0x9F,0x29,0x22,0x6F,0x22,0xBB,0xB6,0x0C,0x11,0xB1,0xC0,0x43,0x68,0xC0,0x08,0x23,
+ 0xB2,0xCD,0x32,0xA2,0xBB,0xAE,0x6E,0x88,0xE5,0xDD,0x27,0xE5,0x78,0xEF,0x62,0x35,
+ 0x67,0x72,0x89,0xE3,0xB3,0xC3,0xF9,0xE8,0xA6,0x33,0x74,0x42,0x3D,0xB5,0x1F,0xAE,
+ 0x18,0xF5,0x1E,0xD4,0x9D,0x58,0x74,0xE9,0x0F,0x9A,0x62,0x25,0xFD,0x73,0x6F,0x9D,
+ 0x29,0x60,0x0A,0xB4,0xBA,0x06,0xFE,0x82,0x5A,0xA7,0x23,0x02,0x77,0x06,0x66,0x10,
+ 0x1C,0x36,0x10,0x37,0xAC,0x73,0x66,0x98,0x36,0x4A,0xCD,0x6F,0x15,0xA4,0x19,0xF7,
+ 0x4A,0xD7,0x03,0x38,0x9B,0xE4,0x61,0x1E,0x5D,0x79,0x87,0x2D,0xA2,0xA5,0xB1,0x2D,
+ 0xCE,0x91,0x2D,0xD0,0xAE,0x20,0xAB,0xA9,0x37,0x5F,0x22,0x0A,0x2C,0x34,0xEC,0xB3,
+ 0xE0,0xC7,0xD3,0xC4,0x68,0x49,0x42,0x1E,0x1C,0x2D,0xCD,0x54,0xEA,0x91,0xB0,0x6F,
+ 0x6D,0xC6,0x25,0x06,0x31,0x0C,0x76,0xA1,0x56,0x10,0x5E,0x77,0x57,0x80,0xEE,0x57,
+ 0x45,0x2A,0x00,0x22,0x6B,0x60,0xDD,0x8E,0xB8,0x06,0x8F,0x9F,0x03,0x64,0xC4,0x88,
+ 0x63,0xEF,0xCE,0x35,0x95,0xC1,0xA9,0x15,0xB1,0xC7,0x69,0xE2,0xD0,0x58,0xD5,0x57,
+ 0xE0,0x5E,0x87,0x51,0xFB,0x22,0xF4,0xFA,0x80,0xFF,0xA0,0x01,0x0C,0x3F,0x82,0xFB,
+ 0xC6,0x31,0x02,0xB0,0xBD,0xA3,0x78,0x59,0x32,0xC3,0x3E,0xC1,0x92,0xBD,0xE6,0x7C,
+ 0x1D,0xF9,0x5F,0x34,0x06,0x39,0xA0,0x11,0x91,0x19,0x16,0x39,0x10,0x90,0x56,0xE8,
+ 0xF2,0x99,0x03,0xDE,0x9F,0x7C,0x12,0x5C,0x11,0x66,0x25,0x23,0x44,0x81,0xB2,0x08,
+ 0xFB,0xC1,0x13,0x5B,0x76,0x64,0x91,0x93,0x2A,0x6A,0x2C,0x04,0xAD,0x5E,0x19,0xD4,
+ 0x33,0x4A,0x6D,0x2B,0x28,0x64,0xEE,0xA6,0xFD,0xFB,0x12,0xC8,0x99,0x75,0x42,0x09,
+ 0x35,0xC4,0xF7,0xDF,0x10,0x5D,0xBF,0xDB,0xBE,0xA6,0x72,0x14,0x82,0xF7,0xE4,0x92,
+ 0x18,0x10,0x5B,0x03,0xA3,0xDC,0x13,0x03,0x82,0x76,0x57,0x64,0x2D,0x20,0xAB,0xBB,
+ 0x3A,0x88,0xF5,0x53,0xC2,0x16,0x20,0xA3,0x31,0xEB,0xC2,0x4E,0xDC,0xA7,0xE4,0x38,
+ 0x4A,0x24,0xED,0x2F,0xCF,0x5E,0xF8,0x33,0xA2,0xA4,0x46,0x5A,0x5B,0x8A,0x6D,0xA7,
+ 0xB1,0xA6,0xD6,0x0E,0x02,0x11,0xB8,0xD6,0x7C,0x81,0xAE,0x72,0x22,0x60,0x35,0x09,
+ 0xAE,0x88,0xE5,0x4C,0x3D,0x08,0x70,0x7E,0x3C,0x8A,0x79,0xAD,0xFC,0xF3,0x73,0x2E,
+ 0x6B,0x10,0xA1,0x74,0xCC,0x4D,0x5D,0xAA,0x9C,0x73,0x67,0x7A,0xEA,0x8E,0x6F,0x9F,
+ 0x75,0x88,0xD8,0xB3,0x39,0xE2,0xF9,0x55,0x82,0xFA,0x30,0x7F,0x6F,0xA3,0x6F,0x94,
+ 0x0D,0x6E,0xBD,0xBF,0xA7,0x48,0x49,0x68,0x22,0x04,0x34,0x6C,0x5A,0xAD,0x4B,0x68,
+ 0x10,0x6B,0x5B,0xE1,0xD8,0x4A,0x45,0x4A,0x5F,0x1D,0xF6,0x49,0x2A,0xDA,0x87,0xC1,
+ 0x72,0x04,0x6F,0x34,0x2B,0x6E,0x22,0x97,0xBD,0xD4,0x52,0x76,0xF0,0x1D,0xEA,0x85,
+ 0x4B,0x39,0x55,0xE6,0x86,0xAA,0x40,0x42,0xC8,0xC7,0xAF,0x47,0x14,0xD1,0xC2,0x40,
+ 0xA2,0x1F,0xA7,0x58,0xEE,0xD9,0xFC,0x12,0xD8,0x47,0x99,0xEF,0xA1,0x5A,0xE0,0x23,
+ 0x1C,0xA4,0x3A,0x19,0x1A,0xCA,0x08,0xAA,0x04,0xF4,0x7D,0x16,0x8B,0xCA,0xD1,0x4A,
+ 0xF4,0x9D,0x8C,0x7A,0xBC,0x26,0x33,0x78,0xD4,0x5D,0x6F,0x7C,0xC9,0x69,0x7C,0x3E,
+ 0xB3,0xC6,0x18,0x05,0x3C,0xB7,0x3D,0x84,0xBB,0x0F,0xAB,0x0B,0xEA,0x85,0x6F,0xD7,
+ 0x4E,0x83,0x35,0x61,0x40,0x5A,0x14,0xD3,0x08,0x8B,0x36,0xE3,0xC4,0x81,0x84,0xA5,
+ 0x80,0x3E,0x92,0x6C,0x1E,0x64,0xB9,0xDD,0x3D,0xFF,0xFC,0xC6,0x82,0x14,0xA7,0x4C,
+ 0xB1,0x04,0xCA,0xCA,0x94,0xA0,0x5B,0x7A,0xBA,0x50,0xC1,0xA6,0x60,0x5B,0x67,0x71,
+ 0xBB,0xE1,0x5C,0xAA,0xB3,0x4D,0xA9,0x20,0x2F,0xE0,0x44,0x37,0x8D,0xFC,0xF4,0x5A,
+ 0xFE,0x14,0xDE,0xAD,0x6C,0x81,0x95,0x3A,0x3B,0x49,0xDC,0x40,0x28,0x8A,0x2D,0xC5,
+ 0x78,0x6A,0x03,0x51,0x3B,0xA8,0xA6,0x96,0x7D,0x34,0xE9,0x83,0x2B,0x65,0x85,0x98,
+ 0xBB,0x71,0x02,0xE0,0x95,0xC2,0x2E,0xCD,0x22,0xBB,0xE6,0xBD,0x58,0x97,0x57,0x81,
+ 0x57,0x38,0x24,0x49,0x84,0x20,0xC9,0x67,0xCC,0x27,0x4E,0xEF,0x15,0xE5,0xAD,0x49,
+ 0x12,0x23,0x4C,0x09,0x91,0x1C,0x85,0xB6,0x3D,0x30,0x68,0xCD,0xC3,0x08,0x49,0xD6,
+ 0x32,0x9F,0xD1,0xC3,0xC7,0x0E,0x41,0xEB,0xB7,0x83,0xAB,0x2A,0x2B,0xE3,0x96,0x87,
+ 0x01,0x5A,0xEA,0x67,0x03,0x4B,0x75,0x23,0x5A,0xD8,0x1F,0x07,0x29,0x38,0x0A,0xB4,
+ 0x27,0xD6,0xBA,0x8A,0xF3,0xE5,0xF1,0x73,0x62,0x24,0xA3,0xB2,0x99,0x96,0x8C,0xD3,
+ 0x10,0x06,0x1E,0x3A,0xED,0x51,0x83,0xD8,0xD4,0x71,0x48,0x64,0x4A,0x59,0x4A,0xBF,
+ 0xC3,0xB1,0x77,0x69,0xC2,0x85,0xFD,0xF4,0x78,0x44,0xA9,0x52,0xAF,0x70,0xF4,0x5B,
+ 0x0B,0x83,0x8F,0x4C,0x29,0x2C,0x08,0xE3,0x92,0xDE,0x38,0x05,0x25,0xEF,0x64,0x03,
+ 0x52,0xC1,0x40,0x50,0x94,0x75,0xF0,0xB1,0x3C,0xEE,0x3E,0x27,0x76,0x40,0x81,0x97,
+ 0x12,0xBC,0x9E,0x8E,0x55,0x1C,0x97,0xF2,0x3B,0xC7,0xDB,0xFB,0x12,0x38,0xA1,0xA8,
+ 0xE8,0xD0,0xAC,0x4F,0xBE,0x91,0x29,0x4B,0x06,0xC5,0x35,0xE7,0xA4,0x7B,0x1B,0x9C,
+ 0x8D,0xD5,0xD5,0x4B,0xA4,0x17,0xA2,0xF4,0x49,0xFC,0xCD,0x89,0xEB,0xA1,0xEA,0x95,
+ 0xBC,0xF6,0x85,0x45,0xF9,0x42,0x56,0x8B,0x4E,0x28,0x55,0xE6,0x3D,0x91,0xC6,0x66,
+ 0x1E,0x1A,0x85,0xF4,0x50,0xEA,0x77,0x28,0x66,0x9D,0x7E,0x20,0xFC,0x6E,0x5D,0x8A,
+ 0xF9,0xA4,0x95,0x52,0x89,0xF9,0x68,0x06,0x65,0x0E,0x37,0xD5,0xB0,0x0C,0x7D,0x6E,
+ 0x73,0x4A,0x6E,0xA0,0xA2,0x96,0xDD,0xE4,0xB2,0xA8,0x2D,0xFE,0xD0,0x55,0xF5,0x06,
+ 0x62,0x60,0x1E,0x0E,0xD7,0x83,0xB7,0xE1,0x36,0xFE,0x2C,0x1E,0x4D,0x84,0x15,0x6F,
+ 0xFC,0x80,0x9A,0x05,0xB9,0x1D,0xF7,0xFC,0x3F,0x2C,0x36,0x78,0x6E,0x62,0x03,0xA1,
+ 0x30,0xF4,0xA8,0xA9,0x79,0x4E,0xB7,0x09,0xC9,0x5E,0xEA,0x77,0x2C,0xFB,0x55,0xB2,
+ 0x2D,0x29,0x8E,0x0C,0x5A,0xF5,0x5C,0x00,0x3B,0x2E,0x89,0x01,0xAA,0xB2,0xE2,0x15,
+ 0x30,0x9B,0x45,0xA3,0x12,0xFA,0x88,0xA5,0x0D,0x61,0xC3,0xBC,0xB8,0xA1,0xC0,0x59,
+ 0x19,0x1B,0xA6,0xA9,0x91,0x5E,0x8D,0xC3,0x49,0x76,0xDE,0x1B,0x51,0x12,0xD2,0xB2,
+ 0x46,0x5B,0x2A,0xE6,0xD4,0x7C,0x9F,0xA1,0x50,0xA9,0xAE,0x92,0x8F,0x0B,0xFC,0xEB,
+ 0x0B,0xCB,0xE5,0xE6,0xAE,0xD4,0x7D,0xAB,0x60,0x2D,0x2A,0xAC,0x71,0x23,0xD1,0x91,
+ 0xB5,0x99,0xCF,0x34,0x2C,0xD3,0xC6,0xF0,0xED,0x38,0xA2,0x70,0x50,0xA3,0xDF,0xCD,
+ 0xA0,0x0F,0x28,0x41,0xB2,0x74,0x43,0x69,0x33,0x42,0xEB,0x43,0x3E,0x46,0x7C,0x1C,
+ 0x24,0xD0,0xCF,0x19,0xAF,0x86,0xB8,0xE4,0x1C,0x96,0x02,0xAF,0xEB,0xD1,0xCC,0x67,
+ 0x74,0x07,0x42,0xF7,0x8D,0x69,0x58,0x42,0x7D,0xDE,0x09,0x65,0x79,0xEE,0x86,0xC2,
+ 0x81,0x3D,0x96,0x3D,0x38,0xF0,0xB2,0x4C,0x29,0xB3,0xAB,0xA5,0x91,0x93,0x4C,0x9A,
+ 0xD7,0x78,0xCD,0xFA,0x7C,0x4E,0x4B,0x58,0x87,0x49,0x1D,0xED,0xC0,0x4E,0x48,0x63,
+ 0x01,0x0A,0xE0,0x68,0x9E,0xFB,0x50,0xCE,0xE4,0x16,0xCF,0xAC,0xAA,0x33,0x4E,0x5C,
+ 0xAC,0x57,0xE6,0xE2,0xFE,0x56,0x8E,0xFD,0xC4,0x5D,0x34,0x2F,0x3E,0x10,0x47,0xC6,
+ 0x47,0xF9,0xA8,0xDA,0xA2,0x13,0xF2,0x20,0x71,0x4C,0x2A,0x24,0x90,0x5D,0xC1,0x00,
+ 0xA4,0xB3,0x87,0x75,0xF2,0xF1,0x8A,0x2A,0x38,0x81,0xC6,0x6B,0x4D,0xEA,0xEE,0xAC,
+ 0x2E,0xEA,0xAE,0xF6,0x73,0x9B,0x59,0xF9,0xD1,0x0F,0x0C,0xF8,0x92,0x95,0xB8,0xAA,
+ 0xC3,0xAE,0x80,0x9A,0xD1,0x32,0x4A,0xC0,0x02,0xCD,0xA3,0xDE,0x4D,0xC5,0x2D,0x0E,
+ 0x9D,0x6C,0xE2,0x24,0x91,0x98,0x9D,0xE2,0x3B,0x8A,0x95,0xC7,0x33,0xC1,0xB1,0x46,
+ 0x62,0x12,0x12,0x64,0xC8,0x1E,0x39,0x01,0x5C,0x10,0x05,0x6F,0x37,0xE3,0xCC,0xC8,
+ 0xA3,0x38,0xD3,0xB9,0xE8,0xAD,0xD4,0x6C,0xCB,0x7A,0x1D,0x5A,0x9B,0x5C,0xB1,0x8F,
+ 0x04,0xAB,0xAE,0x82,0xA7,0xA2,0xC8,0x02,0x5D,0x82,0xF9,0xBC,0x70,0xF1,0x19,0x93,
+ 0x24,0xAD,0x14,0x33,0x92,0x0D,0x7E,0x14,0xDE,0xB1,0x1A,0x7B,0x64,0x85,0x5F,0x11,
+ 0xD0,0x90,0x7F,0xE3,0xB6,0xAE,0x1D,0x41,0xF4,0xC3,0x21,0x92,0x77,0x59,0xE5,0x95,
+ 0xDD,0xD5,0x53,0xB3,0x13,0x67,0x67,0xAC,0x59,0xB3,0x85,0x2A,0x01,0xB7,0xEA,0x21,
+ 0x8D,0x00,0xDB,0x14,0xE2,0x97,0x0B,0xB5,0x53,0x8A,0x9C,0x51,0x10,0x8A,0x11,0xE6,
+ 0xDA,0x78,0x8F,0xCD,0xFF,0x9D,0x4B,0x15,0x4F,0xC7,0x5C,0x6F,0x87,0x3C,0xDE,0x31,
+ 0x86,0xEA,0x21,0x6A,0xB0,0x67,0x19,0xF4,0xFD,0x45,0xF3,0x74,0x8D,0x4C,0x89,0x0E,
+ 0x21,0x24,0x35,0xD0,0x48,0xE8,0xD0,0x18,0x49,0xFC,0x99,0x02,0xB4,0xFC,0x5E,0x81,
+ 0x2E,0x53,0xD7,0xBF,0xD3,0x6A,0x3F,0x96,0x2D,0xB3,0x90,0x7B,0x50,0xAA,0xD6,0x86,
+ 0x1D,0x9D,0x33,0xA6,0x23,0xD6,0xBA,0xDF,0xA7,0x1B,0x12,0x91,0x24,0x29,0x56,0xC0,
+ 0xFD,0x4F,0x1B,0x7B,0xB3,0x81,0x9F,0x08,0x52,0xDC,0x48,0xDE,0x02,0x1E,0xEC,0xD6,
+ 0x3B,0xCC,0xB9,0x19,0x6C,0xC8,0x21,0x26,0x25,0xD1,0x29,0x55,0x4D,0xE1,0xD7,0xA0,
+ 0x6E,0x63,0x4A,0xB9,0xC7,0x71,0x2D,0x25,0xAF,0x8E,0x1E,0x20,0xC7,0xFA,0xA2,0xAE,
+ 0xC5,0x7F,0x84,0x04,0x30,0x9D,0x01,0x46,0xB7,0xAD,0x28,0x49,0x7C,0x75,0x5D,0x2F,
+ 0xAB,0x0F,0xB6,0x02,0x55,0x70,0x0F,0x8D,0xC0,0x84,0x52,0x87,0xDF,0xED,0xE4,0x85,
+ 0xC6,0xED,0x56,0x6E,0x50,0xE3,0xB7,0x95,0xFB,0xAA,0x03,0x4E,0x60,0x8E,0x02,0x2C,
+ 0x8E,0xEC,0x35,0x02,0x77,0xF9,0xCE,0xD1,0x27,0xA2,0x8A,0x00,0x6C,0x97,0xAA,0x7B,
+ 0xAD,0x7E,0x2D,0xBF,0xCB,0x99,0x54,0xCB,0xAD,0x3C,0x76,0xCA,0x01,0xA0,0x7F,0xF8,
+ 0xC1,0x93,0x91,0x1A,0xE7,0x5D,0xF5,0x3D,0x61,0x30,0x24,0x17,0x9B,0x5B,0xC2,0x59,
+ 0x7B,0xA1,0x4A,0x68,0x47,0xC0,0xD0,0x2C,0xDB,0xC1,0x5F,0x61,0x71,0xBE,0xAF,0x6E,
+ 0x8F,0xCB,0x5B,0xD0,0xAE,0xF1,0x75,0xE4,0x94,0x92,0xB0,0x9E,0x43,0xE8,0x1C,0x50,
+ 0xEE,0x31,0xA8,0xA2,0x5E,0xC1,0x44,0xF6,0x85,0x02,0x29,0x07,0x2B,0x9A,0x59,0xEA,
+ 0x94,0x55,0xC5,0xB6,0xB3,0x4F,0xF8,0xE3,0x97,0xF6,0x32,0x09,0x08,0x0D,0x84,0x1B,
+ 0xA6,0x01,0x8E,0xB0,0x66,0xA7,0xB2,0xE8,0x66,0xFA,0x5C,0x05,0x14,0x59,0xB4,0x31,
+ 0xC5,0xB7,0xF4,0xF4,0xFB,0xB1,0x93,0x3E,0xA7,0x81,0x56,0x95,0xD9,0x7D,0xDE,0x3E,
+ 0x04,0x16,0xD2,0x74,0x90,0x77,0x4E,0x79,0x36,0x3D,0x11,0xBE,0x3D,0xBC,0x81,0x1E,
+ 0xD6,0x39,0x05,0x87,0xB6,0x77,0xFD,0x12,0x27,0x47,0x25,0x24,0x78,0xB2,0xC7,0x88,
+ 0xA9,0xB0,0xBE,0x80,0x52,0xCF,0x0E,0x04,0xB7,0x0B,0xC3,0x9F,0x9E,0x4F,0xAC,0x06,
+ 0x16,0x63,0x04,0xD6,0x97,0xEE,0x26,0x94,0x94,0x54,0xB4,0x4B,0xE8,0x8D,0x6F,0x6C,
+ 0x37,0x60,0x45,0x27,0x23,0xF2,0x0E,0x9E,0x82,0x4E,0x7A,0x2A,0x6F,0xE0,0xF9,0xDE,
+ 0x0A,0xDA,0x44,0x39,0x2A,0x89,0x4A,0x14,0x53,0xFC,0xE4,0x3A,0xFD,0x27,0x5E,0x79,
+ 0x04,0x25,0x87,0xC0,0xD5,0x0A,0xBE,0xAC,0x3D,0x98,0xAA,0x90,0x08,0x12,0x39,0xEB,
+ 0x9B,0x88,0x89,0xE9,0xC2,0xB7,0xD4,0x84,0x05,0x73,0x6F,0xA1,0xC7,0x23,0x64,0x9E,
+ 0xD8,0xBA,0xFB,0xBF,0xD7,0xDA,0x2F,0x94,0xEE,0x2C,0xF5,0x16,0x4C,0xF0,0x1A,0xB9,
+ 0x96,0xAE,0xB5,0x52,0xA5,0xF3,0xBB,0x12,0x5E,0x43,0x03,0xC0,0xB8,0x70,0x54,0xDA,
+ 0xAA,0x6D,0x4F,0xDF,0x8B,0x42,0x6D,0xC4,0x3B,0x6E,0x9A,0xFE,0x8A,0x6F,0xE2,0x9D,
+ 0xB7,0x1D,0xFF,0xE5,0x10,0x96,0x0E,0xA9,0x68,0x6A,0xDA,0xAE,0x6C,0x95,0xF0,0xF2,
+ 0xC3,0xC0,0x8B,0x60,0xE4,0xF1,0xCF,0x39,0x2B,0x44,0xE7,0x92,0x08,0x09,0x24,0xD7,
+ 0x98,0x94,0x43,0x59,0x57,0x30,0x75,0xF5,0x10,0x38,0x51,0x7F,0x33,0x2B,0x01,0x97,
+ 0x9A,0x87,0x68,0x57,0x5A,0xB8,0xF6,0x8A,0xC2,0xEB,0xE4,0x6D,0x24,0x61,0x24,0x6B,
+ 0x12,0xEF,0xB6,0x82,0xC1,0x3F,0x5B,0x37,0xFC,0xBC,0x01,0x4D,0x8D,0x8A,0x25,0xA2,
+ 0xBA,0x1B,0x3C,0xD4,0x10,0xEF,0x72,0x5B,0xDD,0xAD,0xAB,0x57,0x50,0x00,0x8D,0x3A,
+ 0x34,0xCC,0xB7,0xF2,0xA6,0xA4,0x75,0x90,0x51,0x91,0xA9,0x28,0xE1,0x0B,0x8C,0xBC,
+ 0x8E,0xA9,0x9C,0x06,0x6E,0xDF,0x56,0xD0,0x25,0x09,0x0C,0x00,0x59,0x89,0xDF,0x56,
+ 0x41,0x09,0x96,0xC7,0xFA,0x1F,0x92,0x39,0x61,0x9C,0x62,0xF9,0x47,0x0E,0x89,0x9E,
+ 0xF4,0x72,0xC6,0x8C,0x8B,0xD5,0x40,0x94,0x32,0x98,0xEC,0xE8,0x52,0xBB,0xE6,0x32,
+ 0xA1,0x74,0xD0,0x2B,0xF1,0x39,0x37,0xF1,0x77,0x97,0x4D,0x1C,0x20,0xD2,0xA5,0x30,
+ 0x7C,0xA9,0x0D,0xD4,0x64,0x47,0xEB,0xD2,0x5D,0x09,0x0D,0x12,0x5D,0xA1,0x37,0xED,
+ 0xCC,0xB1,0x52,0x09,0x2E,0x64,0xED,0x90,0x4D,0xDA,0xCB,0xB7,0x7D,0xEC,0xEB,0xBC,
+ 0x8F,0xDB,0x19,0xA8,0xF0,0x13,0xE2,0x3D,0xE9,0x21,0xA1,0xCD,0x3A,0x80,0x79,0x44,
+ 0xC0,0x18,0x2F,0xC6,0x5C,0x57,0x6E,0x77,0x60,0x7B,0x89,0x71,0xFF,0x2C,0xEB,0x3D,
+ 0x40,0x68,0x9F,0x2E,0xF6,0x70,0xA5,0xD9,0x83,0x05,0x6A,0xF3,0xE3,0x4B,0x80,0x04,
+ 0xA4,0x88,0xCC,0x15,0x10,0x77,0xC9,0x6C,0xF4,0x7A,0x50,0x2C,0xB0,0x3A,0x6D,0x01,
+ 0x04,0x65,0x67,0x24,0x99,0xA5,0x08,0x34,0x5F,0xC1,0x53,0xA2,0x4C,0x5D,0x79,0xF7,
+ 0x62,0x1A,0xC2,0x04,0x28,0x0B,0x6D,0x67,0xD4,0xCF,0xC8,0xED,0xC8,0xCE,0x16,0x12,
+ 0x8E,0x0C,0xBF,0xF0,0x06,0xEE,0xBE,0x38,0xED,0x96,0xF5,0xED,0xC9,0x5D,0xD1,0xBB,
+ 0x8A,0xFC,0x2C,0x3C,0x5B,0xEA,0x80,0x5E,0xDC,0x11,0xA6,0x9A,0x24,0x16,0xB8,0x3E,
+ 0xD2,0x50,0x37,0x15,0xC7,0x68,0xB6,0x53,0xA3,0x31,0xE7,0xD3,0x3A,0x37,0xE9,0x3B,
+ 0x7E,0xAC,0x05,0x03,0x68,0x47,0x30,0x45,0x59,0x5A,0x59,0x2B,0xED,0xB1,0x66,0xE6,
+ 0x34,0x82,0x7D,0xD2,0x2C,0x1C,0x26,0xFB,0x89,0x8B,0x8A,0x31,0x3F,0x5D,0x6B,0x25,
+ 0xE4,0x21,0x5F,0xEB,0xC2,0x51,0x2A,0xD0,0xDF,0x41,0x2E,0x7E,0x42,0x16,0xAD,0x6A,
+ 0x86,0xB6,0x3D,0xC0,0x0D,0xF0,0x83,0x5B,0x17,0xA8,0x00,0x24,0xD7,0xD9,0x04,0x04,
+ 0xA3,0xAE,0x81,0xAA,0x20,0x30,0x2C,0xA3,0x84,0x2D,0x5A,0x1C,0x9D,0xA1,0x81,0x74,
+ 0x5C,0x79,0x3E,0x0B,0xE1,0xA9,0x69,0xE3,0x45,0xF6,0xAE,0x32,0x49,0x18,0xB2,0xCB,
+ 0x73,0x5E,0x27,0xA1,0x31,0x97,0x38,0x49,0xB1,0x7B,0x75,0x91,0xCC,0x92,0xD0,0xB5,
+ 0xA1,0xBB,0xDB,0x7C,0x89,0x51,0x38,0xAA,0xCC,0xCF,0x7F,0xCA,0xE7,0xA0,0x98,0x7B,
+ 0xA8,0xEE,0x58,0x36,0x23,0x2C,0x98,0xC4,0x42,0xAB,0x42,0x0F,0x09,0x42,0xDE,0xB2,
+ 0xFB,0x28,0x1D,0xCA,0x85,0xB6,0x31,0xA2,0xD3,0x19,0x53,0x3A,0xA3,0x0B,0xB6,0x13,
+ 0x64,0x46,0x14,0xD1,0x09,0xB5,0xFC,0x69,0x5B,0xD2,0x16,0x59,0xF0,0xCA,0x8F,0x33,
+ 0x1B,0x2E,0xA5,0xB6,0x2A,0x76,0xAD,0xB5,0x31,0x53,0xAD,0x12,0xF5,0xFE,0x5E,0x5C,
+ 0x58,0x93,0xFE,0x77,0x3B,0x92,0x25,0xF2,0x15,0x8D,0xE5,0xAF,0x21,0xAA,0x56,0x2E,
+ 0xF5,0x1A,0xF9,0x0D,0x49,0x4E,0x24,0xCF,0xD8,0x21,0xC6,0x30,0x96,0xBF,0xD5,0x0B,
+ 0x2D,0xCE,0xC2,0x76,0x35,0xBD,0x5D,0xF2,0x88,0x12,0x8F,0x2A,0xB6,0xDB,0xB1,0x8C,
+ 0x3C,0x07,0x95,0x9C,0x6D,0xA6,0x0E,0xF7,0x29,0x22,0x4A,0x26,0xF7,0x50,0x6A,0x2B,
+ 0x27,0x98,0x49,0xB1,0xED,0x22,0xF3,0x61,0xCE,0x10,0x1C,0x64,0x61,0x1B,0xDE,0x54,
+ 0xF7,0x20,0xAB,0x87,0x01,0x17,0xA4,0x52,0xC7,0xF6,0x8F,0x67,0xFD,0x12,0x51,0x4E,
+ 0x9F,0x6E,0xCD,0x7B,0xB9,0xCB,0x55,0xDC,0xFD,0xAB,0x59,0x96,0xCC,0xB8,0x98,0x3C,
+ 0x48,0xB0,0x47,0x2C,0xF1,0xC4,0xCF,0xC4,0xDC,0xED,0x4E,0x0C,0xF4,0x63,0xA9,0x7F,
+ 0x0B,0x83,0x39,0x50,0xD7,0xAD,0xB6,0x34,0x2F,0x31,0x57,0x93,0xCD,0xCE,0x68,0x4D,
+ 0x9B,0x92,0x12,0x4E,0xF9,0x76,0xB1,0xE7,0x32,0x95,0x66,0x1F,0xAD,0x3D,0x24,0x10,
+ 0x45,0xF7,0x3D,0xFC,0x01,0x66,0x71,0x35,0x43,0xDF,0x60,0xDE,0x57,0xBD,0x77,0x1E,
+ 0xD5,0x81,0x1C,0x6A,0xDE,0xCE,0x27,0xAC,0xCA,0x7F,0xD6,0x70,0x70,0xA5,0x83,0xBD,
+ 0xCF,0x24,0x72,0x99,0xE2,0x78,0x32,0x97,0x1A,0xCA,0x00,0xE3,0x45,0x28,0xC1,0x76,
+ 0xC9,0x65,0xE7,0x5E,0x20,0xC3,0x0B,0xE9,0x88,0x87,0x93,0x90,0x58,0x97,0x2F,0x00,
+ 0x7D,0x7C,0x42,0x45,0xE2,0x25,0xFA,0xAF,0x22,0x34,0x8B,0xBF,0x4E,0x68,0x6C,0x47,
+ 0xC0,0x20,0x00,0x4A,0x84,0xF9,0x6A,0xC0,0x75,0xF9,0x78,0x7A,0x41,0xAC,0x7A,0xFF,
+ 0xF6,0x03,0x4A,0x23,0xA5,0x02,0x9C,0x8F,0x7A,0x31,0xE5,0xC9,0x49,0xB3,0x62,0xCB,
+ 0xBA,0xC4,0x8B,0x57,0x0D,0x83,0xB0,0xA1,0x70,0x5E,0x84,0x44,0x7D,0x0C,0xD6,0x9D,
+ 0x12,0xEF,0x94,0xFF,0x08,0xED,0xFC,0xEF,0xD9,0x4B,0x19,0x4B,0x34,0xC8,0x71,0x34,
+ 0x67,0x73,0xA0,0x96,0xCC,0x8B,0x5D,0x1B,0x2A,0xA6,0x96,0x55,0xAE,0x28,0x18,0x3C,
+ 0x8A,0xCD,0x16,0x63,0xD7,0x4C,0x3E,0xF0,0xEC,0xAE,0x3A,0xFE,0x2C,0x34,0x5E,0xAD,
+ 0x3F,0xDE,0x18,0xAE,0xE2,0xEB,0x2C,0x6E,0xE4,0xCC,0x53,0xD6,0x50,0xD7,0xFE,0xCD,
+ 0x62,0x72,0x4D,0x9E,0x47,0x3B,0x29,0xF2,0x88,0x0E,0x1C,0x5D,0xD5,0xB6,0x9C,0x0F,
+ 0x43,0x27,0x8F,0x46,0x21,0x76,0xAC,0x1E,0x2A,0xE0,0xA5,0x71,0x24,0x91,0x2F,0xD9,
+ 0xD7,0x9B,0x37,0xB3,0x4F,0x92,0xC1,0x66,0x54,0x54,0x3F,0x38,0xD7,0xEE,0x75,0x54,
+ 0x12,0x9B,0x62,0x0D,0xFB,0x7D,0xD3,0x02,0x42,0x47,0x81,0x46,0x91,0x4D,0x8A,0xE5,
+ 0xF7,0xD0,0xFB,0xCC,0x0D,0x9D,0x2D,0xEF,0x6D,0x24,0x5F,0xC5,0x75,0x6C,0xB6,0xD4,
+ 0xF2,0xE1,0x90,0xBB,0xAD,0x90,0xC2,0x63,0xE7,0x93,0x21,0x76,0xFB,0xD5,0x9D,0x78,
+ 0x2F,0xD5,0xC3,0x08,0x9E,0xAE,0xB3,0xE7,0x21,0xCE,0x57,0x6F,0x12,0x57,0x01,0x44,
+ 0xB1,0x44,0x89,0x88,0x89,0x15,0xDB,0x66,0xD7,0x7F,0x8F,0x26,0x4D,0x7C,0x95,0xC3,
+ 0xEE,0xAE,0x79,0xED,0xD9,0xBE,0x82,0xC8,0xFA,0xEE,0x8A,0x9D,0x57,0xFC,0x11,0xCA,
+ 0xF0,0x94,0x95,0x76,0xDF,0x52,0x86,0x05,0xF4,0x6C,0x0E,0x2C,0xA0,0xF2,0xC6,0x72,
+ 0x39,0x2F,0xA9,0xF1,0x69,0x8D,0x2D,0xC9,0xC3,0xE2,0xBC,0x2B,0x36,0xFB,0xCB,0xD6,
+ 0xED,0x8D,0x2D,0xD6,0xFE,0x26,0x98,0xC9,0x87,0x95,0x63,0xDF,0xBE,0xD4,0x8B,0x4A,
+ 0x1F,0x9A,0x51,0x92,0x51,0x57,0xDB,0xFD,0x17,0x73,0xE5,0x6D,0x21,0x62,0x01,0xA2,
+ 0x94,0xEE,0xFF,0xCE,0x15,0xD8,0x68,0x62,0x31,0x56,0xD3,0x74,0x34,0xF1,0xC7,0x75,
+ 0x23,0xAC,0xE7,0x91,0x50,0x15,0x15,0x4E,0xE8,0xA8,0xD8,0xFE,0xE5,0x2A,0xCF,0x84,
+ 0x7E,0xB0,0x48,0x9F,0xF4,0xC0,0x86,0xA2,0x9D,0xD6,0x1F,0x90,0xCC,0x55,0x24,0x1E,
+ 0xDE,0x29,0x1E,0x91,0xCD,0xBF,0x62,0x00,0x86,0xF6,0xDB,0xB6,0x58,0x91,0x26,0x8C,
+ 0xD4,0xAA,0xF5,0xCE,0xC0,0x8B,0xED,0x70,0xF9,0x5F,0x47,0x7B,0x04,0xDE,0x8D,0x2E,
+ 0x3E,0x29,0x9D,0x55,0x8A,0x2A,0x3B,0x3E,0x99,0x51,0xE9,0xF2,0xC1,0xC3,0x77,0x55,
+ 0x36,0xC2,0x8C,0xD6,0x1F,0x58,0xB3,0x64,0xC9,0x83,0x53,0x76,0x20,0xC7,0x5D,0x3B,
+ 0xFA,0xE9,0x3D,0x8A,0xBA,0xDC,0x04,0x15,0x7C,0x21,0x6E,0x43,0x6B,0xD1,0xB9,0xAE,
+ 0xBF,0x6C,0x8B,0xCC,0xBF,0x83,0x2D,0x5F,0xD7,0x04,0x01,0x92,0x37,0xA1,0x09,0xF4,
+ 0xAD,0x11,0x8B,0x26,0x08,0x8D,0x8D,0x7F,0x13,0xA6,0x8E,0xE7,0xD4,0x36,0xB0,0x6A,
+ 0x71,0x92,0x18,0x6B,0x87,0x85,0x02,0x34,0x9B,0x63,0xF9,0xA2,0x65,0x15,0x4E,0xE6,
+ 0x8F,0xBB,0x54,0x22,0x91,0x81,0x0F,0x5C,0xA7,0xCE,0x02,0x26,0xA0,0x8D,0x11,0x1F,
+ 0x83,0x87,0xB3,0xEF,0x6A,0x03,0x49,0x02,0x13,0x32,0x25,0x55,0x43,0x8C,0xED,0x27,
+ 0x8D,0x13,0xD1,0x92,0x4F,0xBD,0x04,0xA2,0x0E,0x07,0x32,0x58,0x6F,0x46,0xC2,0xFA,
+ 0x37,0x1D,0xF1,0xC2,0x7C,0xE7,0x63,0xBC,0x92,0xB0,0x9A,0xCC,0x18,0x6E,0x7C,0xD9,
+ 0x36,0x2F,0x8B,0xF1,0xEB,0x8B,0x4E,0xDA,0x9E,0xC1,0x8D,0xE2,0x94,0xF0,0x7C,0xA4,
+ 0x1D,0x70,0xDB,0x87,0x2A,0xFC,0x0D,0xDE,0xC2,0xFF,0x00,0x3F,0x19,0x80,0xDA,0xFB,
+ 0x66,0xF0,0xB5,0xFD,0x22,0x76,0xE5,0xBD,0x36,0x15,0xBD,0xC7,0xCD,0x90,0xB7,0xC5,
+ 0x1B,0x01,0xCA,0xC8,0x9B,0x8E,0x83,0x4E,0xE6,0xA5,0x25,0xCE,0x56,0x18,0x3E,0xB1,
+ 0x4A,0xC2,0xC6,0x35,0xA1,0x4A,0x91,0x53,0x1E,0x92,0x25,0x6B,0x0E,0xD0,0x97,0x9C,
+ 0xCF,0xAA,0xA4,0x3C,0xAE,0xE8,0x74,0xB0,0xC8,0x43,0xFE,0xFA,0x04,0x11,0xA5,0xBB,
+ 0x23,0x15,0x4F,0x90,0x40,0x90,0xB6,0x67,0x26,0x5B,0x6A,0x4A,0x3A,0x2E,0xA9,0x8C,
+ 0x82,0x51,0x20,0x86,0x00,0x2B,0x61,0x38,0x53,0x35,0x0E,0xFC,0x59,0x3C,0xE1,0xA4,
+ 0x15,0xB6,0x6A,0xFE,0x2F,0x80,0xE6,0x6D,0xBC,0xD4,0xAB,0x34,0xAD,0x2B,0x35,0x99,
+ 0x62,0x66,0x35,0x5A,0xFF,0xE8,0x40,0x34,0xBE,0xDE,0x8A,0xB2,0x78,0x24,0x90,0x46,
+ 0x7F,0x27,0x00,0xE7,0x31,0x30,0x1C,0x5E,0x16,0x1E,0x78,0xE5,0x9F,0x80,0x66,0x97,
+ 0xEE,0xF3,0x49,0x24,0xD8,0x22,0x2C,0x8B,0x68,0xA5,0x38,0x37,0xA4,0x46,0x59,0x3A,
+ 0x51,0x7B,0xB4,0x77,0x91,0x72,0xDA,0x83,0x95,0xF5,0xC9,0xEB,0x40,0x70,0xA1,0xE6,
+ 0x8B,0x92,0xE7,0xEF,0x5E,0x80,0xB4,0x39,0xCC,0x47,0x28,0x98,0x12,0x81,0xEE,0x8E,
+ 0x8F,0xF6,0x76,0x5D,0xB7,0x25,0x4A,0x9C,0xE1,0x4B,0xD1,0xAA,0x0C,0xF0,0xC8,0x2B,
+ 0xE3,0x24,0xAB,0x58,0x7B,0x95,0x71,0x57,0x26,0xC4,0x1C,0x12,0x27,0x75,0x59,0x41,
+ 0xC5,0xF1,0x10,0xDE,0xEC,0x4F,0xE1,0x17,0xCA,0x38,0xB5,0x7D,0x17,0xC4,0x7C,0x0B,
+ 0xDF,0xEB,0x9F,0xF8,0xDB,0x34,0xE4,0xC6,0x6F,0x70,0x68,0xEF,0xA7,0x4B,0x11,0x33,
+ 0xEC,0xF8,0x01,0x91,0x35,0x0A,0x24,0x6F,0xB2,0x5A,0xED,0x67,0x70,0x4B,0x0F,0x11,
+ 0xE9,0x5E,0x66,0x47,0x51,0x2A,0xF6,0xED,0xB5,0xA6,0x7A,0x5E,0x24,0xB3,0xC9,0x18,
+ 0x46,0x1C,0x15,0x6D,0x2C,0x6B,0xE4,0x01,0x21,0x8D,0x36,0xDB,0x0A,0xDB,0x59,0xFB,
+ 0x5F,0x14,0x8B,0x3A,0x3E,0x59,0x8E,0x1D,0x9F,0x3C,0xAA,0x67,0x49,0x44,0xD7,0xA4,
+ 0x32,0x9E,0xE6,0x3D,0xCA,0x5B,0x59,0xEF,0xBB,0x0B,0xB5,0x14,0x97,0x9A,0x49,0x84,
+ 0x3D,0xC5,0xE0,0x3D,0xFB,0xEA,0xD2,0x03,0xC4,0xE7,0xC3,0x26,0x28,0xC5,0xCC,0x5E,
+ 0xAA,0x00,0xB8,0x66,0x38,0xF3,0x23,0xC4,0x00,0xD8,0x39,0x90,0xBF,0x16,0x72,0x85,
+ 0xFA,0xA9,0xEF,0x5C,0xD5,0x02,0x93,0x89,0xC4,0x3B,0x28,0x5A,0x8E,0xFC,0x36,0x32,
+ 0x46,0x2A,0x26,0x74,0x3D,0xD3,0x3D,0x42,0x3A,0x71,0x5D,0x2F,0x63,0x83,0xB6,0x57,
+ 0x4A,0x24,0xB6,0x0C,0x2F,0xFA,0xAD,0x25,0xFF,0x43,0xE4,0x68,0xDD,0x60,0xDF,0x1C,
+ 0x60,0xB1,0xC2,0x00,0xE4,0xD8,0xDC,0x7E,0xDC,0x0E,0x85,0xDA,0x1E,0x94,0xE6,0x8D,
+ 0x9A,0x38,0x35,0xE5,0x3C,0x5C,0x81,0x68,0x66,0x5E,0x6C,0x71,0x36,0xFE,0xD8,0x1B,
+ 0x38,0x96,0x60,0x05,0x5D,0x35,0x27,0x02,0x17,0x03,0x34,0x34,0x9D,0x85,0xB9,0xA0,
+ 0x11,0x9F,0x49,0xF3,0x7A,0xB5,0xF1,0x78,0xAD,0x3A,0xE3,0x58,0x2C,0x08,0x15,0xBD,
+ 0xCE,0xB9,0x54,0xFA,0x84,0xB2,0x25,0xD0,0x6F,0x5F,0x62,0xF7,0xE1,0xD6,0x2A,0x15,
+ 0x14,0x3D,0xFC,0xAF,0xD5,0xFD,0x9F,0x80,0xB8,0xF6,0x37,0x63,0x52,0xB1,0x35,0xAF,
+ 0x00,0x2F,0xE8,0xB2,0x5A,0xAF,0x03,0x4F,0xE4,0x14,0x28,0x94,0xA1,0xE8,0x63,0x7B,
+ 0xA5,0x80,0x9E,0x37,0x10,0x94,0x1C,0x1F,0x77,0x76,0x1C,0x09,0x13,0x87,0x43,0x2F,
+ 0x55,0xDD,0x45,0x24,0x0C,0x58,0x2E,0x38,0x34,0x2F,0x97,0x08,0x17,0xB4,0x79,0xBD,
+ 0x88,0x2A,0x07,0x84,0xEB,0x6E,0x18,0x7A,0x54,0xA3,0xBF,0x65,0x8E,0x3E,0xBE,0xEC,
+ 0x93,0xB7,0xAF,0xCF,0xA8,0x93,0x82,0x30,0x2C,0x62,0xC0,0x68,0xA5,0x8F,0x36,0xA0,
+ 0xB6,0x49,0x4D,0x52,0x7D,0x6E,0xC3,0x60,0x88,0x12,0x9B,0x94,0x46,0x7F,0xFA,0x00,
+ 0xE4,0x45,0x00,0xB8,0x14,0x3C,0x7F,0x54,0xA8,0x71,0x9E,0x01,0xE4,0x67,0x00,0x98,
+ 0xBC,0x86,0xCA,0x58,0x82,0xD5,0xBB,0x35,0x51,0x35,0x4D,0x3B,0xB2,0x43,0xFD,0xBA,
+ 0x7F,0xDA,0x68,0x2C,0x0D,0xDB,0xC9,0xC2,0x79,0x8F,0x43,0xED,0x7B,0x5A,0x45,0xA9,
+ 0x89,0xE3,0xA7,0x5F,0x62,0xD9,0x29,0x48,0x6F,0x97,0x6A,0x2B,0xB6,0x96,0x0E,0x2F,
+ 0x8E,0xB5,0x25,0xBB,0x63,0x1E,0xB1,0x2F,0x6D,0xF5,0x7A,0xE9,0x1A,0x58,0xF7,0xA9,
+ 0xC9,0x98,0x35,0xCB,0xD9,0x98,0x20,0xDB,0x35,0x67,0xD3,0x55,0xF3,0xA9,0x32,0x31,
+ 0xBE,0x69,0x9A,0x4E,0x43,0xD6,0x9E,0x95,0x0F,0xB8,0x66,0x70,0x83,0xD6,0xC7,0x31,
+ 0x86,0x96,0x0D,0xCD,0x09,0x07,0x48,0xAC,0x02,0x6F,0xE1,0x00,0x57,0xBF,0xCD,0x64,
+ 0x71,0x71,0xED,0x3F,0xDC,0x7F,0x0E,0x80,0x08,0x68,0xE3,0xE3,0xAC,0xEC,0x13,0x90,
+ 0x9C,0xEC,0x8D,0x87,0x92,0x37,0xAF,0xC2,0xB6,0xDF,0x45,0xBE,0xA1,0x99,0x1F,0x9F,
+ 0x09,0xF4,0xA9,0x72,0xEB,0x87,0x77,0x7C,0x77,0x0F,0x7A,0xE7,0xFD,0x7C,0xC5,0xEB,
+ 0x83,0xCC,0x1C,0x56,0xA2,0xD3,0x67,0x19,0x80,0xD8,0x22,0xE8,0xC4,0x8A,0x31,0x2C,
+ 0x86,0x15,0xCD,0x9C,0x58,0xED,0x02,0xD8,0x06,0x9B,0x0A,0xEB,0x67,0x66,0x5C,0x28,
+ 0x4F,0xFE,0xC4,0x68,0xCA,0xB5,0x6B,0xDB,0x2D,0x91,0x72,0x6F,0x86,0x66,0x12,0x2F,
+ 0xA5,0xF0,0x16,0x47,0x84,0x0C,0xCC,0x42,0xFA,0xF7,0x9A,0x3B,0x43,0xCC,0x73,0x41,
+ 0x92,0xB4,0xB5,0x6E,0x18,0xDE,0x6F,0x4D,0x37,0x8C,0xCC,0x4C,0x02,0x8B,0x08,0xFC,
+ 0x2A,0x6C,0x9F,0x2C,0x8D,0x3C,0xD4,0x29,0x31,0x53,0xE0,0x24,0x8D,0xD6,0xAC,0x8C,
+ 0xA6,0x6E,0xFE,0xD1,0xFC,0xE4,0x9B,0x7D,0xC6,0x6D,0xD1,0x63,0x61,0xB7,0x30,0xCC,
+ 0xCF,0x33,0xA3,0x6A,0x51,0x6A,0x4A,0x1C,0x98,0x42,0xF1,0x73,0x6D,0xAB,0x4C,0x7C,
+ 0x14,0x62,0x02,0xBC,0x90,0xF8,0x4B,0x9E,0x61,0x64,0x38,0x88,0x2D,0x61,0x76,0x0E,
+ 0xF1,0x9E,0x6A,0x8F,0x37,0xC2,0x5C,0xD4,0xAE,0x54,0x67,0xAD,0x4F,0xF4,0xF8,0x59,
+ 0x26,0xE6,0x58,0xE6,0xEE,0x2D,0x3A,0x83,0x18,0x6D,0xE0,0x0D,0x72,0x42,0x4B,0x6A,
+ 0x59,0xDA,0x25,0xB4,0x8A,0xA9,0xA4,0xA3,0x87,0xDF,0x50,0x8F,0xEF,0x35,0x18,0xC3,
+ 0x12,0xCA,0xFD,0xBE,0x82,0x78,0xA9,0xF6,0x8B,0xA1,0x31,0xF7,0xA9,0x3F,0x26,0x59,
+ 0x4F,0xAF,0x91,0xFB,0x11,0x81,0xDF,0x47,0x3D,0xC8,0xA5,0x12,0x19,0x60,0xEF,0xAA,
+ 0xD5,0xFB,0x93,0xA8,0x4B,0x95,0xD6,0x6A,0xED,0x17,0x57,0x26,0xE2,0x27,0x55,0xAA,
+ 0x33,0x2A,0x92,0x81,0xA0,0x14,0xAB,0x12,0x83,0xA4,0xA1,0xE2,0x85,0x15,0x74,0x38,
+ 0x9D,0x95,0xE6,0x18,0xD5,0xCD,0x5E,0xC0,0xB9,0x66,0x3C,0x62,0x3A,0xF6,0xCC,0xBB,
+ 0xB1,0x88,0x04,0x41,0xAE,0x18,0xCE,0x80,0x61,0xF9,0x5A,0x89,0x8F,0x24,0x0E,0xFC,
+ 0x0A,0xDC,0xC2,0x57,0xEB,0x17,0xAE,0x2F,0x51,0x17,0x1F,0xE0,0x95,0xFE,0x08,0x5C,
+ 0x05,0xE8,0x0A,0x7D,0x15,0x7A,0xED,0x2E,0xD2,0x3C,0x0D,0x9C,0x4C,0xF3,0x79,0x91,
+ 0x86,0x84,0xD3,0x58,0x86,0x9E,0x05,0x16,0x9B,0xA8,0x1B,0x0B,0x2A,0x2F,0x77,0xFC,
+ 0xF2,0x0F,0xA0,0xBF,0x40,0xBF,0x15,0xCE,0x2D,0x70,0xD7,0xA4,0xB2,0x1C,0x76,0x08,
+ 0x7C,0xF6,0x09,0x3B,0xFF,0x3D,0x2F,0x7B,0x67,0x63,0xCC,0x31,0x27,0xBA,0x82,0xBF,
+ 0x59,0xA1,0x57,0x7C,0xBD,0x8F,0x98,0xC5,0x55,0xCA,0xC2,0x0A,0x60,0x7A,0x1D,0x5A,
+ 0xC7,0xA5,0xE0,0xFE,0x08,0xB8,0xED,0xB0,0xAA,0xB8,0x2E,0xB1,0x66,0x3D,0xE4,0x32,
+ 0xCA,0xF8,0x80,0x21,0xB6,0x2F,0x80,0x55,0x61,0x85,0xA9,0x31,0x69,0x69,0x36,0xA8,
+ 0x16,0xE3,0xC5,0xB2,0xA3,0xBB,0x9E,0x16,0x3B,0x3C,0xDE,0xB0,0xE3,0x3D,0x35,0x95,
+ 0x70,0x61,0x55,0x2D,0x97,0x78,0xB3,0xA0,0x59,0x40,0x9A,0x7F,0x0A,0xED,0xB8,0x4F,
+ 0x2D,0xFC,0x77,0xA2,0x9E,0x58,0x53,0x7C,0xA0,0x9A,0x8F,0xAF,0x3E,0x72,0x1E,0x59,
+ 0x31,0xC1,0xF5,0x5D,0x70,0xD8,0xDF,0x7D,0xB2,0x67,0xB0,0x3E,0x5A,0x9B,0xE1,0x67,
+ 0x0B,0xC8,0x32,0xD8,0xDF,0xE2,0x02,0x11,0x65,0x57,0xB9,0x4A,0xE4,0x64,0x69,0x36,
+ 0x95,0xA6,0xE1,0x0B,0x3D,0x14,0x2C,0x4D,0x88,0x3A,0x6A,0x7E,0x01,0x8C,0x08,0x62,
+ 0x0F,0x84,0x0B,0x63,0xC0,0x41,0x0B,0x65,0x17,0x22,0xDE,0x2F,0xFD,0xD6,0xD2,0xFE,
+ 0xBA,0x4E,0x2F,0xAC,0x88,0x2B,0x2E,0x98,0x8E,0xD2,0x71,0x3D,0xB2,0x3C,0x19,0x68,
+ 0x98,0x54,0x6D,0xD9,0x4C,0xD3,0x0D,0x77,0xE4,0x2B,0xBA,0x4E,0xC0,0xB6,0xD5,0x6D,
+ 0x37,0x65,0x7F,0x58,0xF6,0x29,0xDD,0x99,0x35,0x93,0x15,0xBA,0xD1,0x0C,0x31,0x6F,
+ 0xEC,0x2B,0x3B,0xAD,0xA9,0xAD,0xD8,0x7F,0x7D,0x91,0x96,0x2A,0x97,0x43,0xF3,0xC4,
+ 0x74,0xF6,0x5C,0x8E,0x9A,0xC0,0xA4,0x2A,0x20,0xD0,0xF0,0x6C,0x0E,0x25,0xD9,0x41,
+ 0xE2,0x6F,0xA8,0x54,0x65,0xE6,0xF5,0x0E,0xCD,0x8E,0x28,0xB4,0x4B,0x41,0xF5,0x29,
+ 0x9C,0xB5,0x84,0xD5,0x2E,0x33,0x82,0xD3,0x71,0x9F,0x94,0x21,0x65,0x06,0x68,0x68,
+ 0xDC,0xC4,0x0F,0xA7,0x68,0xAC,0x8E,0x3C,0x7C,0xA0,0x0D,0x34,0xB7,0x8B,0xD3,0x5C,
+ 0x05,0x5D,0xC9,0xB9,0x53,0xF8,0x01,0xED,0x9B,0xDC,0x61,0x72,0x38,0xE8,0x4C,0x9A,
+ 0x5A,0x9E,0xF9,0x2B,0x86,0x4B,0xB0,0xE4,0x32,0x16,0xC1,0x28,0xD3,0x7C,0x66,0x4D,
+ 0x7C,0x49,0x7A,0x39,0x08,0xF3,0x97,0x30,0x38,0xDC,0x5F,0x86,0x18,0xBB,0xB7,0x1E,
+ 0xFF,0xB5,0x88,0xF1,0xFF,0x06,0x25,0x8F,0x7C,0xE3,0x27,0xE9,0xF1,0x7C,0x01,0x75,
+ 0xCF,0x3B,0x99,0xE3,0x20,0x95,0x81,0xEA,0x9D,0x5E,0xBF,0x55,0x5B,0x5C,0xEE,0x13,
+ 0x26,0x84,0x45,0xF1,0x85,0x49,0x3F,0x33,0xC5,0xA0,0x6B,0x03,0x0D,0x26,0xE8,0xCB,
+ 0x49,0xF5,0x48,0xF1,0x0A,0x53,0x4C,0x33,0x0D,0x2A,0xAA,0x02,0xB0,0x08,0x38,0xC3,
+ 0x6D,0xD8,0x7E,0x8D,0xC1,0xCC,0xF5,0x3B,0xB6,0x0B,0x7D,0x82,0xF0,0x43,0x17,0xDF,
+ 0x68,0x4E,0xED,0x10,0xC4,0x1D,0x64,0x05,0x40,0x5F,0xC9,0x26,0xDF,0xE9,0xAB,0x49,
+ 0x2E,0x74,0x8E,0x8D,0xF7,0x18,0xD7,0x26,0xEA,0x38,0x28,0x43,0x14,0x35,0x9F,0xCF,
+ 0x72,0x59,0x27,0x4F,0x53,0x5D,0x9A,0xB7,0x96,0x8F,0x40,0xD6,0xC4,0x0C,0xB5,0x87,
+ 0x42,0xEA,0xD1,0x18,0xEB,0x2E,0xD8,0x39,0xEF,0xD4,0xF0,0x3D,0x74,0x83,0x4B,0x44,
+ 0x5E,0xEE,0x2A,0xBC,0x9C,0x73,0x07,0xF2,0xC7,0xEC,0xA5,0xAA,0xBB,0x18,0x92,0xCF,
+ 0x0E,0xF4,0xA9,0xBC,0xC6,0xCC,0x47,0x53,0x09,0x35,0x03,0xC9,0xDE,0x82,0x5C,0xAE,
+ 0x0C,0x5F,0x06,0x08,0x0E,0x6B,0xEE,0xAD,0x94,0xDD,0x65,0x94,0x46,0x56,0xD2,0xEB,
+ 0x39,0xE9,0x71,0x12,0x61,0xE2,0x50,0x64,0x81,0xAF,0xC8,0x16,0x65,0x55,0xF4,0x2C,
+ 0x94,0x9F,0x34,0xCA,0xBD,0x51,0x3C,0x68,0x9B,0x08,0x55,0x77,0x8D,0xEB,0x4E,0x31,
+ 0x81,0x2D,0xAC,0xB9,0x20,0xF7,0x87,0x2E,0x37,0x1C,0xCD,0x57,0x50,0xEB,0x33,0x60,
+ 0xE2,0x03,0x83,0x52,0x47,0xAD,0x0E,0x3F,0xEA,0x3B,0x58,0x11,0x80,0xF4,0x4D,0xCD,
+ 0xA6,0x3E,0x8A,0xF1,0x31,0xDC,0x07,0x97,0x74,0x6C,0xBF,0x8F,0x3A,0x74,0xA3,0xB5,
+ 0xEE,0x1D,0x0E,0xAC,0x4F,0xEB,0x09,0xB3,0x38,0x8B,0xBA,0xE7,0x54,0x0E,0x6F,0xD4,
+ 0xE7,0x9D,0xF3,0x1D,0xB7,0xE1,0x7B,0x46,0x03,0x19,0x16,0x2C,0x4C,0x0D,0x5B,0x98,
+ 0x27,0x6D,0x5D,0xA0,0x72,0x07,0x8F,0x39,0xCB,0x48,0x23,0xA2,0x60,0xD0,0x2B,0xFF,
+ 0x7A,0x0F,0x0A,0xBE,0xE7,0xCB,0xAF,0xCF,0xDB,0x99,0xB8,0x23,0x07,0xB8,0xC8,0xE8,
+ 0xC1,0x08,0xEF,0xD5,0xE8,0xA7,0xC1,0xD0,0x2C,0x48,0x5C,0xEC,0xA7,0x21,0x58,0x01,
+ 0x67,0x48,0xE3,0xF0,0xCE,0x46,0xAF,0x01,0x3F,0x7A,0x81,0xD1,0x38,0x10,0xA3,0xDA,
+ 0xEC,0xCE,0x52,0xFD,0x54,0x75,0x67,0x83,0x5A,0x9B,0xC7,0x4F,0x8B,0x75,0x7B,0x97,
+ 0x47,0x71,0xE3,0x16,0xBC,0x49,0xF7,0x9F,0x4C,0xF2,0xAF,0xE2,0xA6,0x8D,0xBF,0x18,
+ 0xA1,0x2D,0x15,0x07,0xB1,0xD6,0xCF,0x79,0xF1,0x16,0x97,0x1E,0xE0,0x6B,0x49,0xA3,
+ 0x98,0x86,0x30,0x23,0x0B,0xD0,0xAB,0xD0,0x0E,0x69,0xC1,0xCA,0x1C,0x22,0xA1,0x27,
+ 0xB3,0xC3,0x9A,0xC9,0xD5,0xB9,0x27,0x62,0xBD,0xC9,0xB1,0x55,0x66,0xA2,0x3C,0xC1,
+ 0x49,0x65,0x68,0x91,0x87,0xBE,0x77,0x83,0x3B,0x40,0x24,0x6F,0x6F,0xF3,0x29,0x79,
+ 0xEA,0xCE,0x29,0xB7,0x0C,0x51,0xB8,0xB4,0xFF,0x12,0xB9,0x8C,0x38,0x7B,0x07,0xEB,
+ 0x6F,0x96,0xBB,0xC9,0x86,0x25,0x6B,0x07,0x06,0x73,0xDA,0xE8,0x29,0xCD,0xFE,0xD8,
+ 0xDB,0x2D,0xC9,0x6C,0xAE,0x78,0x35,0x30,0x32,0xC6,0x11,0x16,0x19,0xD1,0xF4,0x69,
+ 0x7E,0xFC,0x8C,0xA7,0x26,0x3A,0x2D,0x46,0xD6,0xB8,0x34,0xC4,0x4F,0x68,0x5A,0x8B,
+ 0xAB,0x89,0x3A,0x7A,0x8C,0xED,0xA8,0x54,0x64,0xE6,0x43,0xD0,0xA6,0x2C,0x55,0x82,
+ 0xC8,0xB4,0x78,0xC1,0x65,0x1B,0x8B,0xB4,0x74,0xF6,0x0A,0xCD,0xE0,0x04,0x0B,0x8E,
+ 0xBB,0xB7,0x6A,0x6A,0x10,0x81,0xCC,0xA1,0x50,0x94,0xFE,0xE7,0xF6,0x33,0xC2,0x06,
+ 0x72,0x2C,0x31,0x6A,0x30,0xD6,0x6D,0x15,0x78,0xCD,0xBB,0xAD,0x8E,0x6D,0x18,0xAB,
+ 0xCF,0x33,0x81,0xF6,0xF3,0xB5,0x92,0x5D,0xF9,0x95,0xEF,0x34,0xA3,0x09,0xD1,0x3A,
+ 0x76,0xF6,0x6E,0x41,0x52,0xAB,0x69,0x03,0x84,0xE7,0x6D,0xE5,0x62,0xC9,0xB7,0xD1,
+ 0xB6,0x3F,0xA3,0x82,0x8E,0x37,0x34,0x2C,0xC2,0x6F,0xAD,0x49,0x5C,0xCB,0x3C,0x5B,
+ 0x38,0x30,0x6A,0xBB,0xB6,0x6E,0x12,0x7F,0x13,0x63,0xA0,0x0A,0x1D,0xE3,0xC6,0xD4,
+ 0x77,0x92,0x75,0x28,0x2E,0x73,0xB7,0x02,0x09,0x09,0xFB,0x31,0x10,0x32,0x4E,0x28,
+ 0xDD,0x9B,0x5B,0x22,0x37,0x75,0xB5,0x96,0x66,0x5C,0x33,0xA8,0xF9,0x26,0x2B,0x86,
+ 0x8A,0x79,0x3F,0x56,0x1C,0x5E,0x38,0x5E,0x7E,0xEF,0x48,0xDC,0x0D,0x0A,0x88,0x6C,
+ 0x3F,0xCB,0x91,0x6A,0xE1,0x4F,0xFC,0x6C,0x2F,0x87,0x89,0x7A,0x9C,0xFE,0x1A,0x88,
+ 0xBC,0x12,0x04,0x04,0x35,0x55,0x29,0x99,0xE8,0x0B,0x65,0xD7,0xAE,0x27,0x7C,0x8E,
+ 0x19,0x7F,0xD0,0x82,0x73,0x68,0xCF,0x68,0xF3,0x8B,0xCB,0x52,0xEC,0xFB,0xC4,0xDC,
+ 0x26,0xC6,0xCA,0x7D,0x10,0x46,0x53,0x0F,0xAA,0x92,0xDA,0x72,0x5D,0xBB,0x15,0x5E,
+ 0xF5,0x56,0x7B,0xB8,0xA1,0x34,0xD9,0x65,0xA1,0xC5,0xB0,0x3D,0xD8,0xDA,0x9F,0x81,
+ 0xBF,0x33,0xAB,0x7D,0x16,0x95,0xC3,0xC4,0x70,0x7F,0x82,0xDC,0x6C,0x78,0x08,0xBE,
+ 0x49,0x81,0xAB,0x81,0x78,0x60,0xDD,0xA3,0x75,0x1C,0xDB,0x97,0xBE,0x4B,0xCB,0x54,
+ 0xFD,0xF1,0xB5,0xDD,0xAB,0x81,0x80,0xD0,0x08,0x2D,0x88,0x94,0x07,0x4F,0x3A,0xCB,
+ 0x66,0xD0,0xF6,0x2F,0x87,0xAD,0x25,0x81,0x70,0x29,0x13,0xA0,0x83,0x5F,0x5B,0xD1,
+ 0x9B,0x00,0x00,0x91,0xB2,0xD7,0x33,0xB6,0x10,0x3A,0xF0,0x9C,0x19,0x13,0xC9,0xB1,
+ 0x18,0x1A,0x87,0xF6,0x5A,0xB8,0x80,0x6F,0x48,0x7A,0xC5,0x8D,0x42,0xDD,0x28,0x4B,
+ 0xDA,0x32,0xDF,0xEE,0x8D,0x6D,0xD3,0x18,0x3D,0x8C,0x48,0x90,0x87,0xBC,0x3F,0x37,
+ 0xCD,0x8A,0xDE,0xCD,0x55,0x0C,0x11,0x5E,0x50,0x92,0xB1,0x85,0x1D,0xD0,0x0A,0xF5,
+ 0x20,0xB3,0xC6,0xD1,0xD1,0x26,0xD5,0x8B,0x12,0x3D,0x86,0x0B,0xFA,0x7E,0x4D,0xEF,
+ 0xAD,0x72,0x38,0x5B,0xBD,0x97,0x48,0xD4,0x1E,0xB2,0x00,0x5C,0x90,0xE6,0xCD,0xA7,
+ 0x5A,0xCA,0x77,0x74,0x54,0xD6,0x34,0x47,0x06,0x48,0xE9,0xBC,0x56,0x13,0x4E,0xE7,
+ 0x17,0x6A,0x46,0x38,0xC9,0xDE,0x51,0x4A,0xFB,0x3B,0x73,0xAD,0xCE,0x8E,0x13,0xC9,
+ 0x33,0x26,0x50,0xD6,0x66,0xF3,0xA7,0xA2,0xB8,0x60,0x5E,0x14,0x11,0x99,0x4C,0x4D,
+ 0x66,0x19,0x99,0xFE,0x59,0x77,0x5B,0xE0,0x20,0xAB,0x36,0xCD,0x91,0x6F,0xE9,0xF8,
+ 0x95,0x43,0x41,0x61,0x7D,0xF8,0xFB,0xAD,0x75,0x33,0xA5,0xD2,0xC4,0x67,0x95,0x97,
+ 0x97,0xB6,0x07,0xD4,0xB0,0x1A,0x7E,0x4E,0x6D,0xCD,0xA9,0x05,0x00,0x54,0x83,0x6D,
+ 0x52,0xA3,0x17,0xD6,0x5A,0xA0,0x83,0xC6,0xD9,0x78,0x29,0xEF,0xD2,0x08,0xA1,0x28,
+ 0xCA,0xEA,0x70,0xA1,0x05,0x84,0xB4,0x48,0xF9,0x00
+};
+
diff --git a/drivers/video/wmt/bootanimation/animation.c b/drivers/video/wmt/bootanimation/animation.c
new file mode 100644
index 00000000..ce37c44d
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/animation.c
@@ -0,0 +1,459 @@
+/*++
+ linux/drivers/video/wmt/animation.c
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#include <linux/mm.h>
+#include <linux/timer.h>
+#include <linux/wait.h>
+#include <linux/sched.h>
+#include <linux/delay.h>
+#include <linux/kthread.h>
+#include <linux/version.h>
+#include <linux/module.h>
+#include <asm/io.h>
+#include <linux/proc_fs.h>
+#include <linux/ioport.h>
+#include "animation.h"
+#include "buffer.h"
+#include "LzmaDec.h"
+#include "../memblock.h"
+
+#include "anim_data.h"
+
+
+#define ANIM_STOP_PROC_FILE "kernel_animation"
+
+#define MAX_CLIP_COUNT 6
+
+extern int wmt_getsyspara(char *varname, unsigned char *varval, int *varlen);
+//extern void clear_animation_fb(void * p);
+//extern void flip_animation_fb(int pingpong);
+
+
+
+#define THE_MB_USER "Boot-Animation"
+
+#define DEFAULT_BUF_IMAGES 4 //default buffered image count
+
+#undef THIS_DEBUG
+//#define THIS_DEBUG
+
+
+
+#ifdef THIS_DEBUG
+#define LOG_DBG(fmt,args...) printk(KERN_INFO "[Boot Animation] " fmt , ## args)
+#define LOG_INFO(fmt,args...) printk(KERN_INFO "[Boot Animation] " fmt, ## args)
+#define LOG_ERROR(fmt,args...) printk(KERN_ERR "[Boot Animation] " fmt , ## args)
+#else
+#define LOG_DBG(fmt,args...)
+#define LOG_INFO(fmt,args...) printk(KERN_INFO "[Boot Animation] " fmt, ## args)
+#define LOG_ERROR(fmt,args...) printk(KERN_ERR "[Boot Animation] " fmt , ## args)
+#endif
+
+
+// MUST match Windows PC tool. Don't change it.
+struct animation_clip_header{
+ int xres;
+ int yres;
+ int linesize;
+ unsigned char x_mode;
+ unsigned char y_mode;
+ short x_offset;
+ short y_offset;
+ unsigned char repeat;
+ unsigned char reserved;
+ int interval;
+ int image_count;
+ int data_len;
+};
+
+// MUST match Windows PC tool. Don't change it.
+struct file_header {
+ int maigc;
+ unsigned short version;
+ unsigned char clip_count;
+ unsigned char color_format;
+ unsigned int file_len;
+};
+
+
+
+struct play_context {
+ struct animation_clip_header *clip;
+ int xpos; // top postion
+ int ypos; // left postion
+
+ volatile int play_thread;
+ animation_buffer buf;
+};
+
+
+// globe value to stop the animation loop
+static volatile int g_logo_stop = 0;
+static struct animation_fb_info fb;
+
+static void *SzAlloc(void *p, size_t size)
+{
+ void * add = (void *)mb_alloc(size);
+ LOG_DBG("alloc: size %d, add = %p \n", size, add);
+ return add;
+}
+
+static void SzFree(void *p, void *address) {
+ if (address != 0) {
+ LOG_DBG("free: address = %p \n", address);
+ mb_free((int)address);
+ }
+}
+
+static ISzAlloc g_Alloc = { SzAlloc, SzFree };
+
+
+static int show_frame(struct play_context *ctx, unsigned char *data)
+{
+ unsigned char * dest;
+ int linesize = fb.width * (fb.color_fmt + 1) * 2;
+ int i = 0;
+ struct animation_clip_header *clip = ctx->clip;
+ if (g_logo_stop)
+ return 0;
+
+ dest = fb.addr;
+
+// printk(KERN_INFO "dest = 0x%p src = 0x%p\n", dest, data);
+
+ if(data) {
+ LOG_DBG("dest %p, data %p (%d,%d) (%dx%d) linesize(%d,%d)", dest, data,
+ ctx->xpos, ctx->ypos, clip->xres, clip->yres, clip->linesize, linesize);
+
+ dest += ctx->ypos * linesize;
+ dest += ctx->xpos * (fb.color_fmt + 1) * 2;
+
+ for (i = 0; i < clip->yres; i++) {
+ memcpy(dest, data, clip->xres * (fb.color_fmt + 1) * 2);
+ dest += linesize;
+ data += clip->linesize;
+ }
+ }
+
+ LOG_DBG("show_frame data %p, fb.addr %p\n", data, fb.addr);
+ return 0;
+}
+
+static int decompress(struct play_context * ctx, unsigned char *src, unsigned int src_len)
+{
+ SRes res = 0;
+ CLzmaDec state;
+ size_t inPos = 0;
+ unsigned char * inBuf;
+ SizeT inProcessed;
+
+
+ // 1) read LZMA properties (5 bytes) and uncompressed size (8 bytes, little-endian) to header
+ UInt64 unpackSize = 0;
+ int i;
+
+ unsigned char * header = src;
+ for (i = 0; i < 8; i++)
+ unpackSize += (UInt64)header[LZMA_PROPS_SIZE + i] << (i * 8);
+
+ // 2) Allocate CLzmaDec structures (state + dictionary) using LZMA properties
+
+
+ LzmaDec_Construct(&state);
+ RINOK(LzmaDec_Allocate(&state, header, LZMA_PROPS_SIZE, &g_Alloc));
+ if (res != SZ_OK)
+ return res;
+
+ // 3) Init LzmaDec structure before any new LZMA stream. And call LzmaDec_DecodeToBuf in loop
+ LzmaDec_Init(&state);
+
+
+ inBuf = header + LZMA_PROPS_SIZE + 8;
+
+ for (;;)
+ {
+ unsigned int outSize;
+ unsigned char * outBuf = animation_buffer_get_writable(&ctx->buf, &outSize);
+
+ unsigned int frame_size = outSize;
+ unsigned int decoded = 0;
+ ELzmaFinishMode finishMode = LZMA_FINISH_ANY;
+ ELzmaStatus status;
+ while(1) {
+ inProcessed = src_len - LZMA_PROPS_SIZE - 8 - inPos;
+
+ res = LzmaDec_DecodeToBuf(&state, outBuf + frame_size - outSize, &outSize,
+ inBuf + inPos, &inProcessed, finishMode, &status);
+
+ inPos += inProcessed;
+ decoded += outSize;
+ unpackSize -= outSize;
+ outSize = frame_size - decoded;
+
+ LOG_DBG("Decoded %d bytes, inPos = %d\n", decoded, inPos);
+
+ if(res != SZ_OK)
+ break;
+
+ if (outSize == 0)
+ break;
+ }
+
+ animation_buffer_write_finish(&ctx->buf, outBuf);
+
+ if (res != SZ_OK || unpackSize == 0 || g_logo_stop)
+ break;
+ }
+
+ // 4) decompress finished, do clean job
+ LzmaDec_Free(&state, &g_Alloc);
+ return res;
+}
+
+static int animation_play(void * arg)
+{
+ unsigned char * data;
+ static int not_first_play;
+ struct play_context *ctx = (struct play_context *)arg;
+
+ LOG_DBG( "animation_play thread start...\n");
+
+ if(!not_first_play) {
+ msleep(500); // sleep a while to wait deocde few frames
+// clear_animation_fb(fb.addr);
+ not_first_play = 1;
+ }
+
+
+ // try to get a valid frame and show it
+ while(!g_logo_stop) {
+ data = animation_buffer_get_readable(&ctx->buf);
+ if(data) {
+ show_frame(ctx, data);
+ animation_buffer_read_finish(&ctx->buf, data);
+ }
+ else {
+ if( ctx->buf.eof ) //no data and reach eof
+ break;
+ LOG_DBG("animation_buffer_get_readable return NULL\n");
+ }
+
+ if(g_logo_stop)
+ break;
+
+ //else
+ msleep(ctx->clip->interval);
+ }
+
+ LOG_DBG( "animation_play thread End\n");
+ animation_buffer_stop(&ctx->buf);
+ ctx->play_thread = 0;
+ return 0;
+}
+
+
+static void decode_clip(struct animation_clip_header *clip, unsigned char * data, int index)
+{
+ // start timer for animation playback
+ struct play_context ctx;
+ int buf_images;
+
+ LOG_DBG("Start playing clip %d, %dx%d, linesize %d, image %d, data_len %d\n",
+ index, clip->xres, clip->yres, clip->linesize, clip->image_count, clip->data_len);
+
+ ctx.clip = clip;
+
+ // init the decompress buffer
+ if (clip->repeat == 0) {
+ buf_images = DEFAULT_BUF_IMAGES;
+ if(buf_images > clip->image_count)
+ buf_images = clip->image_count;
+ }
+ else {
+ // for the repeat clip, alloc a big memory to store all the frames
+ buf_images = clip->image_count;
+ }
+
+ if( 0 != animation_buffer_init(&ctx.buf, clip->linesize * clip->yres, buf_images, &g_Alloc)){
+ LOG_ERROR("Can't init animation buffer %dx%d\n", clip->linesize * clip->yres, buf_images);
+ return;
+ }
+
+
+ ctx.xpos = clip->x_mode * (fb.width / 2 - clip->xres / 2) + clip->x_offset;
+ ctx.ypos = clip->y_mode * (fb.height / 2 - clip->yres / 2) + clip->y_offset;
+
+ kthread_run(animation_play, &ctx, "wmt-boot-play");
+ ctx.play_thread = 1;
+
+ LOG_DBG("Start Decompressing ... \n");
+ decompress(&ctx, data, clip->data_len);
+
+ if (clip->repeat) {
+ while (!g_logo_stop) {
+ // Fake decompress for REPEAT mode. (Only decompress the clip once so we can save more CPU)
+ unsigned int outSize;
+ unsigned char * outBuf;
+
+ outBuf = animation_buffer_get_writable(&ctx.buf, &outSize);
+ animation_buffer_write_finish(&ctx.buf, outBuf);
+ }
+ }
+
+ LOG_DBG("Decompress finished!\n");
+ ctx.buf.eof = 1;
+ //wait the play thread exit
+ while(ctx.play_thread) {
+ msleep(10);
+ }
+
+ LOG_DBG("Play clip %d finished\n", index);
+ animation_buffer_release(&ctx.buf, &g_Alloc);
+}
+
+
+static int animation_main(void * arg)
+{
+ unsigned char * clip;
+ int i;
+
+ struct file_header *header = (struct file_header *)arg;
+ int clip_count = header->clip_count;
+ struct animation_clip_header clip_headers[MAX_CLIP_COUNT];
+ unsigned char * clip_datas[MAX_CLIP_COUNT];
+
+ if( clip_count > MAX_CLIP_COUNT)
+ clip_count = MAX_CLIP_COUNT;
+ LOG_DBG( "animation_main thread start, clip_cout = %d\n", clip_count);
+
+
+ clip = (unsigned char *)(header + 1);
+ for (i = 0; i< clip_count; i++){
+ memcpy(&clip_headers[i], clip, sizeof(struct animation_clip_header));
+ clip += sizeof(struct animation_clip_header);
+ clip_datas[i] = clip;
+ clip += clip_headers[i].data_len;
+ }
+
+ LOG_DBG( "Found %d clip(s)\n", clip_count);
+
+ for (i = 0; i < clip_count; i++) {
+ if (!g_logo_stop)
+ decode_clip(&clip_headers[i], clip_datas[i], i);
+ }
+
+
+ g_Alloc.Free(&g_Alloc, arg);
+ LOG_DBG( "animation_main thread finished \n");
+ return 0;
+}
+
+
+static int stop_proc_write( struct file *file,
+ const char *buffer,
+ unsigned long count,
+ void *data )
+{
+ /*
+ char value[20];
+ int len = count;
+ if( len >= sizeof(value))
+ len = sizeof(value) - 1;
+
+ if(copy_from_user(value, buffer, len))
+ return -EFAULT;
+
+ value[len] = '\0';
+
+ LOG_DBG("procfile_write get %s\n", value);
+ */
+
+ //anything will stop the boot animation
+ animation_stop();
+
+ return count; // discard other chars
+}
+
+struct proc_dir_entry *stop_proc_file;
+
+static struct proc_dir_entry * create_stop_proc_file(void)
+{
+ stop_proc_file = create_proc_entry(ANIM_STOP_PROC_FILE, 0644, NULL);
+
+ if( stop_proc_file != NULL )
+ stop_proc_file->write_proc = stop_proc_write;
+ else
+ LOG_ERROR("Can not create /proc/%s file", ANIM_STOP_PROC_FILE);
+ return stop_proc_file;
+}
+
+int animation_start(struct animation_fb_info *info)
+{
+ struct file_header *header;
+ unsigned char * buffer;
+
+ const void * animation_data = anim_data;
+ if (animation_data == NULL)
+ return -1;
+
+ if( !create_stop_proc_file() )
+ return -1;
+
+ header = (struct file_header *)animation_data;
+
+ if (header->maigc != 0x12344321) {
+ LOG_ERROR ("It's not a valid Animation file at 0x%p, first 4 bytes: 0x%x\n", animation_data, header->maigc);
+ return -1;
+ }
+
+ buffer = g_Alloc.Alloc(&g_Alloc, header->file_len);
+ if(!buffer) {
+ LOG_ERROR ("Can't alloc enough memory, length %d\n", header->file_len);
+ return -1;
+ }
+
+ memcpy(&fb, info, sizeof(fb));
+
+ //copy it to the new buffer and start the play thread
+ memcpy(buffer, header, header->file_len);
+ g_logo_stop = 0;
+
+
+ LOG_DBG("Start animation_main thread ...\n");
+ kthread_run(animation_main, buffer, "wmt-boot-anim");
+ return 0;
+}
+
+int animation_stop(void)
+{
+ LOG_INFO("animation_stop\n");
+ g_logo_stop = 1;
+
+ if( stop_proc_file ) {
+ remove_proc_entry(ANIM_STOP_PROC_FILE,stop_proc_file);
+ stop_proc_file = NULL;
+ }
+
+ return 0;
+}
+
+EXPORT_SYMBOL(animation_start);
+EXPORT_SYMBOL(animation_stop);
+
+
diff --git a/drivers/video/wmt/bootanimation/animation.h b/drivers/video/wmt/bootanimation/animation.h
new file mode 100644
index 00000000..f3d316df
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/animation.h
@@ -0,0 +1,34 @@
+/*++
+ linux/drivers/video/wmt/animation.h
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#ifndef _BOOTANIMATION_H_
+#define _BOOTANIMATION_H_
+
+
+struct animation_fb_info {
+ unsigned char * addr; // frame buffer start address
+ unsigned int width; // width
+ unsigned int height; // height
+ unsigned int color_fmt; // color format, 0 -- rgb565, 1 -- rgb888
+};
+
+int animation_start(struct animation_fb_info *info);
+int animation_stop(void);
+
+#endif
diff --git a/drivers/video/wmt/bootanimation/buffer.c b/drivers/video/wmt/bootanimation/buffer.c
new file mode 100644
index 00000000..5e6f1fe7
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/buffer.c
@@ -0,0 +1,102 @@
+/*++
+ linux/drivers/video/wmt/buffer.c
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+#include "buffer.h"
+
+#define assert(int )
+
+int animation_buffer_init(animation_buffer * buf, int size, int count, ISzAlloc *alloc)
+{
+ buf->buffer = buf->w_pos = buf->r_pos = alloc->Alloc(alloc, size * count);
+ if(!buf->buffer)
+ return -1;
+
+ buf->frame_count = count;
+ buf->frame_size = size;
+ buf->eof = 0;
+
+ sema_init(&buf->sem_writable, count);
+ sema_init(&buf->sem_readable, 0);
+
+ return 0;
+}
+
+int animation_buffer_release(animation_buffer * buf, ISzAlloc *alloc)
+{
+ if (buf->buffer)
+ alloc->Free(alloc, buf->buffer);
+ buf->buffer = buf->w_pos = buf->r_pos = NULL;
+ return 0;
+}
+
+unsigned char * animation_buffer_get_writable(animation_buffer * buf, unsigned int * pSize)
+{
+ down_interruptible(&buf->sem_writable);
+ *pSize = buf->frame_size;
+ return buf->w_pos;
+}
+
+int animation_buffer_stop(animation_buffer * buf)
+{
+ up(&buf->sem_readable);
+ /* up twice for safey*/
+ up(&buf->sem_writable);
+ up(&buf->sem_writable);
+ return 0;
+}
+
+void animation_buffer_write_finish(animation_buffer * buf, unsigned char * addr)
+{
+ assert(addr == buf->w_pos);
+
+ // printk(KERN_INFO "add one buffer 0x%p\n", addr);
+
+ buf->w_pos += buf->frame_size;
+ if (buf->w_pos - buf->buffer == buf->frame_size * buf->frame_count)
+ buf->w_pos = buf->buffer;
+
+ up(&buf->sem_readable);
+ // printk(KERN_INFO "write <<<<<<<======\n");
+ return;
+
+}
+
+
+unsigned char * animation_buffer_get_readable(animation_buffer * buf)
+{
+ int ret = down_trylock(&buf->sem_readable);
+ if (ret == 0) {
+ return buf->r_pos;
+ }
+ else {
+ return NULL;
+ }
+}
+
+void animation_buffer_read_finish(animation_buffer * buf, unsigned char * addr)
+{
+ // Integrity check
+ assert(addr == buf->r_pos);
+ // printk(KERN_INFO "read one buffer 0x%p\n", addr);
+ buf->r_pos += buf->frame_size;
+ if (buf->r_pos - buf->buffer == buf->frame_size * buf->frame_count)
+ buf->r_pos = buf->buffer;
+
+ up(&buf->sem_writable);
+}
+
diff --git a/drivers/video/wmt/bootanimation/buffer.h b/drivers/video/wmt/bootanimation/buffer.h
new file mode 100644
index 00000000..0203ad80
--- /dev/null
+++ b/drivers/video/wmt/bootanimation/buffer.h
@@ -0,0 +1,53 @@
+/*++
+ linux/drivers/video/wmt/buffer.h
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#ifndef ANIMATION_BUFFER_H_INCLUDED
+#define ANIMATION_BUFFER_H_INCLUDED
+
+#include "LzmaDec.h"
+#include <linux/semaphore.h>
+
+typedef struct
+{
+ unsigned char *buffer;
+ unsigned char *w_pos;
+ unsigned char *r_pos;
+ int frame_count;
+ int frame_size;
+ struct semaphore sem_writable;
+ struct semaphore sem_readable;
+ int eof;
+}animation_buffer;
+
+
+int animation_buffer_init(animation_buffer * buf, int size, int count, ISzAlloc *alloc);
+
+int animation_buffer_stop(animation_buffer * buf);
+int animation_buffer_release(animation_buffer * buf, ISzAlloc *alloc);
+
+
+unsigned char * animation_buffer_get_writable(animation_buffer * buf, unsigned int * pSize);
+void animation_buffer_write_finish(animation_buffer * buf, unsigned char * addr);
+
+unsigned char * animation_buffer_get_readable(animation_buffer * buf);
+void animation_buffer_read_finish(animation_buffer * buf, unsigned char * addr);
+
+
+#endif /* ANIMATION_BUFFER_H_INCLUDED */
+
diff --git a/drivers/video/wmt/cec.c b/drivers/video/wmt/cec.c
new file mode 100644
index 00000000..8d2115fb
--- /dev/null
+++ b/drivers/video/wmt/cec.c
@@ -0,0 +1,323 @@
+/*++
+ * linux/drivers/video/wmt/cec.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define CEC_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "cec.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define CEC_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx cec_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in cec.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int cec_xxx; *//*Example*/
+int cec_logical_addr;
+int cec_physical_addr;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void cec_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+/*---------------------------- CEC COMMON API -------------------------------*/
+
+#ifdef WMT_FTBLK_CEC
+/*---------------------------- CEC HAL --------------------------------------*/
+void wmt_cec_tx_data(char *buf, int size)
+{
+ int i;
+ unsigned int addr, reg;
+ int wait_idle;
+
+#ifdef DEBUG
+ DPRINT("[CEC] tx data(%d):", size);
+ for (i = 0; i < size; i++)
+ DPRINT(" 0x%x", buf[i]);
+ DPRINT("\n");
+#endif
+
+ if (size > 16) {
+ DPRINT("[CEC] *W* size max 16\n");
+ return;
+ }
+
+ wait_idle = 0;
+ while (vppif_reg32_read(CEC_WR_START)) { /* wait idle */
+ if (wait_idle >= 10) {
+ DPRINT("[CEC] wait idle timeout\n");
+ return;
+ }
+ wait_idle++;
+ mdelay(10);
+ return;
+ }
+
+ addr = REG_CEC_ENCODE_HEADER;
+ for (i = 0; i < size; i++) {
+ reg = (buf[i] << 4) + 0x1;
+ if (i == (size - 1))
+ reg |= BIT1;
+ vppif_reg32_out(addr, reg);
+#ifdef DEBUG
+ DPRINT("[CEC] wr 0x%x = 0x%x\n", addr, reg);
+#endif
+ addr += 4;
+ }
+ vppif_reg32_write(CEC_WR_NUM, (size * 10) + 0x1);
+ vppif_reg32_write(CEC_WR_START, 1);
+}
+
+int wmt_cec_rx_data(char *buf)
+{
+ int i, size;
+ unsigned int addr, reg;
+
+ addr = REG_CEC_DECODE_HEADER;
+ for (i = 0; i < 16; i++) {
+ reg = vppif_reg32_in(addr);
+ buf[i] = (reg & 0xFF0) >> 4;
+ if (reg & BIT1) /* EOM */
+ break;
+ addr += 4;
+ }
+ vppif_reg32_write(CEC_FINISH_RESET, 1);
+ mdelay(1);
+ vppif_reg32_write(CEC_FINISH_RESET, 0);
+ size = i + 1;
+#ifdef DEBUG
+ DPRINT("[CEC] rx data(%d):\n", size);
+ for (i = 0; i < size; i++)
+ DPRINT(" 0x%x", buf[i]);
+ DPRINT("\n");
+#endif
+ return size;
+}
+
+void wmt_cec_set_clock(void)
+{
+ #define CEC_CLOCK (1000000 / 7984)
+
+ vppif_reg32_out(REG_CEC_WR_START_SET0, 370 * CEC_CLOCK); /* 3.7 ms */
+ vppif_reg32_out(REG_CEC_WR_START_SET1, 450 * CEC_CLOCK); /* 4.5 ms */
+ vppif_reg32_out(REG_CEC_WR_LOGIC0_SET0, 150 * CEC_CLOCK); /* 1.5 ms */
+ vppif_reg32_out(REG_CEC_WR_LOGIC0_SET1, 240 * CEC_CLOCK); /* 2.4 ms */
+ vppif_reg32_out(REG_CEC_WR_LOGIC1_SET0, 60 * CEC_CLOCK); /* 0.6 ms */
+ vppif_reg32_out(REG_CEC_WR_LOGIC1_SET1, 240 * CEC_CLOCK); /* 2.4 ms */
+ vppif_reg32_out(REG_CEC_RD_START_L_SET0, 350 * CEC_CLOCK); /* 3.5 ms */
+ vppif_reg32_out(REG_CEC_RD_START_R_SET0, 390 * CEC_CLOCK); /* 3.9 ms */
+ vppif_reg32_out(REG_CEC_RD_START_L_SET1, 430 * CEC_CLOCK); /* 4.3 ms */
+ vppif_reg32_out(REG_CEC_RD_START_R_SET1, 470 * CEC_CLOCK); /* 4.7 ms */
+ vppif_reg32_out(REG_CEC_RD_LOGIC0_L_SET0, 130 * CEC_CLOCK); /* 1.3 ms */
+ vppif_reg32_out(REG_CEC_RD_LOGIC0_R_SET0, 170 * CEC_CLOCK); /* 1.7 ms */
+ vppif_reg32_out(REG_CEC_RD_LOGIC0_L_SET1, 205 * CEC_CLOCK); /* 2.05 ms*/
+ vppif_reg32_out(REG_CEC_RD_LOGIC0_R_SET1, 275 * CEC_CLOCK); /* 2.75 ms*/
+ vppif_reg32_out(REG_CEC_RD_LOGIC1_L_SET0, 40 * CEC_CLOCK); /* 0.4 ms */
+ vppif_reg32_out(REG_CEC_RD_LOGIC1_R_SET0, 80 * CEC_CLOCK); /* 0.8 ms */
+ vppif_reg32_out(REG_CEC_RD_LOGIC1_L_SET1, 205 * CEC_CLOCK); /* 2.05 ms*/
+ vppif_reg32_out(REG_CEC_RD_LOGIC1_R_SET1, 275 * CEC_CLOCK); /* 2.75 ms*/
+ vppif_reg32_out(REG_CEC_RD_L_SET0_ERROR, 182 * CEC_CLOCK); /* 1.82 ms */
+ vppif_reg32_out(REG_CEC_RD_R_SET1_ERROR, 238 * CEC_CLOCK); /* 2.38 ms */
+ vppif_reg32_out(REG_CEC_RD_L_ERROR, 287 * CEC_CLOCK); /* 2.87 ms */
+ vppif_reg32_out(REG_CEC_RX_SAMPLE_L_RANGE, 85 * CEC_CLOCK); /* 0.85 ms*/
+ vppif_reg32_out(REG_CEC_RX_SAMPLE_R_RANGE, 125 * CEC_CLOCK); /*1.25 ms*/
+ vppif_reg32_out(REG_CEC_WR_SET0_ERROR, 225 * CEC_CLOCK); /* 2.25 ms */
+ vppif_reg32_out(REG_CEC_WR_SET1_ERROR, 225 * CEC_CLOCK); /* 2.25 ms ?*/
+}
+
+void wmt_cec_set_logical_addr(int no, char addr, int enable)
+{
+ unsigned int mask;
+
+ if (no > 5) {
+ DPRINT("[CEC] *W* max 5\n");
+ return;
+ }
+ mask = 0xF << (4 * no);
+ vppif_reg32_write(REG_CEC_LOGICAL_ADDR, mask, 4 * no, addr);
+ mask = BIT24 << no;
+ vppif_reg32_write(REG_CEC_LOGICAL_ADDR, mask, 24 + no, enable);
+ DBGMSG("[CEC] set logical addr %d,0x%x\n", no, addr);
+}
+
+void wmt_cec_rx_enable(int enable)
+{
+ vppif_reg32_write(CEC_REJECT_NEXT_DECODE, (enable) ? 0 : 1);
+ /* GPIO4 disable GPIO function */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x40, BIT4, 4, (enable) ? 0 : 1);
+}
+
+void wmt_cec_enable_int(int no, int enable)
+{
+ vppif_reg32_write(REG_CEC_INT_ENABLE, 0x1 << no, no, enable);
+}
+
+void wmt_cec_clr_int(int sts)
+{
+ vppif_reg32_out(REG_CEC_STATUS, sts);
+}
+
+int wmt_cec_get_int(void)
+{
+ int reg;
+
+ reg = vppif_reg32_in(REG_CEC_STATUS);
+ return reg;
+}
+
+void wmt_cec_enable_loopback(int enable)
+{
+ /* 1 : read self write and all dest data */
+ vppif_reg32_write(CEC_RD_ENCODE_ENABLE, enable);
+}
+
+void wmt_cec_init_hw(void)
+{
+ wmt_cec_set_clock();
+ vppif_reg32_write(CEC_WR_RETRY, 3);
+ vppif_reg32_out(REG_CEC_RX_TRIG_RANGE, 2);
+
+ vppif_reg32_write(CEC_FREE_3X, 3);
+ vppif_reg32_write(CEC_FREE_5X, 5);
+ vppif_reg32_write(CEC_FREE_7X, 7);
+
+ vppif_reg32_write(CEC_COMP_DISABLE, 1);
+ vppif_reg32_write(CEC_ERR_HANDLE_DISABLE, 0);
+ vppif_reg32_write(CEC_NO_ACK_DISABLE, 0);
+ vppif_reg32_write(CEC_DECODE_FULL_DISABLE, 0);
+ vppif_reg32_write(CEC_STATUS4_START_DISABLE, 1);
+ vppif_reg32_write(CEC_STATUS4_LOGIC0_DISABLE, 1);
+ vppif_reg32_write(CEC_STATUS4_LOGIC1_DISABLE, 1);
+ /* 1 : read self write and all dest data */
+ vppif_reg32_write(CEC_RD_ENCODE_ENABLE, 0);
+}
+
+/*---------------------------- CEC API --------------------------------------*/
+void wmt_cec_reg_dump(void)
+{
+ DPRINT("========== CEC register dump ==========\n");
+ vpp_reg_dump(REG_CEC_BEGIN, REG_CEC_END - REG_CEC_BEGIN);
+
+ DPRINT("---------- CEC Tx ----------\n");
+ DPRINT("wr start %d,wr num %d\n",
+ vppif_reg32_read(CEC_WR_START),
+ vppif_reg32_read(CEC_WR_NUM));
+ DPRINT("wr header ack %d,EOM %d,data 0x%x\n",
+ vppif_reg32_read(CEC_WR_HEADER_ACK),
+ vppif_reg32_read(CEC_WR_HEADER_EOM),
+ vppif_reg32_read(CEC_WR_HEADER_DATA));
+ DPRINT("wr data ack %d,EOM %d,data 0x%x\n",
+ vppif_reg32_read(CEC_WR_DATA_ACK),
+ vppif_reg32_read(CEC_WR_DATA_EOM),
+ vppif_reg32_read(CEC_WR_DATA));
+ DPRINT("finish reset %d,wr retry %d\n",
+ vppif_reg32_read(CEC_FINISH_RESET),
+ vppif_reg32_read(CEC_WR_RETRY));
+ DPRINT("---------- CEC Rx ----------\n");
+ DPRINT("rd start %d,all ack %d,finish %d\n",
+ vppif_reg32_read(CEC_RD_START),
+ vppif_reg32_read(CEC_RD_ALL_ACK),
+ vppif_reg32_read(CEC_RD_FINISH));
+ DPRINT("rd header ack %d,EOM %d,data 0x%x\n",
+ vppif_reg32_read(CEC_RD_HEADER_ACK),
+ vppif_reg32_read(CEC_RD_HEADER_ACK),
+ vppif_reg32_read(CEC_RD_HEADER_DATA));
+ DPRINT("rd data ack %d,EOM %d,data 0x%x\n",
+ vppif_reg32_read(CEC_RD_DATA_ACK),
+ vppif_reg32_read(CEC_RD_DATA_EOM),
+ vppif_reg32_read(CEC_RD_DATA));
+
+ DPRINT("---------- Logical addr ----------\n");
+ DPRINT("addr1 0x%x,valid %d\n",
+ vppif_reg32_read(CEC_LOGICAL_ADDR1),
+ vppif_reg32_read(CEC_ADDR_VALID1));
+ DPRINT("addr2 0x%x,valid %d\n",
+ vppif_reg32_read(CEC_LOGICAL_ADDR2),
+ vppif_reg32_read(CEC_ADDR_VALID2));
+ DPRINT("addr3 0x%x,valid %d\n",
+ vppif_reg32_read(CEC_LOGICAL_ADDR3),
+ vppif_reg32_read(CEC_ADDR_VALID3));
+ DPRINT("addr4 0x%x,valid %d\n",
+ vppif_reg32_read(CEC_LOGICAL_ADDR4),
+ vppif_reg32_read(CEC_ADDR_VALID4));
+ DPRINT("addr5 0x%x,valid %d\n",
+ vppif_reg32_read(CEC_LOGICAL_ADDR5),
+ vppif_reg32_read(CEC_ADDR_VALID5));
+
+ DPRINT("---------- Misc ----------\n");
+ DPRINT("free 3x %d,5x %d,7x %d\n", vppif_reg32_read(CEC_FREE_3X),
+ vppif_reg32_read(CEC_FREE_5X), vppif_reg32_read(CEC_FREE_7X));
+ DPRINT("reject next decode %d,comp disable %d\n",
+ vppif_reg32_read(CEC_REJECT_NEXT_DECODE),
+ vppif_reg32_read(CEC_COMP_DISABLE));
+ DPRINT("err handle disable %d,no ack disable %d\n",
+ vppif_reg32_read(CEC_ERR_HANDLE_DISABLE),
+ vppif_reg32_read(CEC_NO_ACK_DISABLE));
+ DPRINT("r1 enc ok %d,r1 dec ok %d,r1 err %d\n",
+ vppif_reg32_read(CEC_R1_ENCODE_OK),
+ vppif_reg32_read(CEC_R1_DECODE_OK),
+ vppif_reg32_read(CEC_R1_ERROR));
+ DPRINT("r1 arb fail %d,r1 no ack %d\n",
+ vppif_reg32_read(CEC_R1_ARB_FAIL),
+ vppif_reg32_read(CEC_R1_NO_ACK));
+ DPRINT("dec full disable %d,self rd enable %d\n",
+ vppif_reg32_read(CEC_DECODE_FULL_DISABLE),
+ vppif_reg32_read(CEC_RD_ENCODE_ENABLE));
+}
+
+#ifdef CONFIG_PM
+static unsigned int *wmt_cec_pm_bk;
+void wmt_cec_do_suspend(void)
+{
+ /* Suspend GPIO output high */
+ vppif_reg32_write(GPIO_BASE_ADDR+0xC0, BIT23, 23, 0);
+ wmt_cec_pm_bk = vpp_backup_reg(REG_CEC_BEGIN,
+ (REG_CEC_END - REG_CEC_BEGIN));
+}
+
+void wmt_cec_do_resume(void)
+{
+ vppif_reg32_out(REG_VPP_SWRST2_SEL, 0x1011111);
+ /* disable GPIO function */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x40, BIT4, 4, 0);
+ /* GPIO4 disable GPIO out */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x80, BIT4, 4, 0);
+ /* GPIO4 disable pull ctrl */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x480, BIT4, 4, 0);
+ /* Suspend GPIO output enable */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x80, BIT23, 23, 1);
+ /* Suspend GPIO output high */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0xC0, BIT23, 23, 1);
+ /* Wake3 disable pull ctrl */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x480, BIT19, 19, 0);
+ vpp_restore_reg(REG_CEC_BEGIN,
+ (REG_CEC_END - REG_CEC_BEGIN), wmt_cec_pm_bk);
+ wmt_cec_pm_bk = 0;
+}
+#endif
+#endif /* WMT_FTBLK_CEC */
+
diff --git a/drivers/video/wmt/cec.h b/drivers/video/wmt/cec.h
new file mode 100644
index 00000000..d11d27b3
--- /dev/null
+++ b/drivers/video/wmt/cec.h
@@ -0,0 +1,65 @@
+/*++
+ * linux/drivers/video/wmt/cec.h
+ * WonderMedia HDMI CEC driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp-osif.h"
+#include "./hw/wmt-vpp-hw.h"
+#include "vpp.h"
+#include "com-cec.h"
+
+#ifdef WMT_FTBLK_CEC
+
+#ifndef CEC_H
+#define CEC_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifdef CEC_C
+#define EXTERN
+#else
+#define EXTERN extern
+#endif
+
+EXTERN void wmt_cec_tx_data(char *buf, int size);
+EXTERN void wmt_cec_clr_int(int sts);
+EXTERN int wmt_cec_get_int(void);
+
+EXTERN void wmt_cec_enable_loopback(int enable);
+EXTERN void wmt_cec_tx_data(char *buf, int size);
+EXTERN int wmt_cec_rx_data(char *buf);
+EXTERN void wmt_cec_hotplug_notify(int plug_status);
+EXTERN void wmt_cec_do_hotplug_notify(int no, int plug_status);
+
+EXTERN void wmt_cec_init_hw(void);
+EXTERN void wmt_cec_set_logical_addr(int no, char addr, int enable);
+EXTERN void wmt_cec_rx_enable(int enable);
+EXTERN void wmt_cec_do_suspend(void);
+EXTERN void wmt_cec_do_resume(void);
+
+#undef EXTERN
+#ifdef __cplusplus
+}
+#endif
+#endif /* CEC_H */
+#endif /* WMT_FTBLK_CEC */
diff --git a/drivers/video/wmt/com-cec.h b/drivers/video/wmt/com-cec.h
new file mode 100644
index 00000000..80f1ce16
--- /dev/null
+++ b/drivers/video/wmt/com-cec.h
@@ -0,0 +1,63 @@
+/*++
+ * linux/drivers/video/wmt/com-cec.h
+ * WonderMedia HDMI CEC driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef COM_CEC_H
+#define COM_CEC_H
+
+#if defined(__cplusplus)
+extern "C" {
+#endif
+
+/* define for notify AP */
+#define USER_PID 1
+#define WP_PID 2
+#define USER_PIX_MAX 2
+#define NETLINK_CEC_TEST (MAX_LINKS - 2)
+
+#define DEVICE_RX_DATA 0
+#define DEVICE_PLUG_IN 2
+#define DEVICE_PLUG_OUT 3
+#define DEVICE_STREAM 4
+
+#define MAX_MSG_BYTE 16
+#define MSG_ABORT 0xff
+
+typedef struct wmt_cec_msg {
+ char msglen;
+ char msgdata[MAX_MSG_BYTE];
+} wmt_cec_msg_t;
+
+typedef struct wmt_phy_addr {
+ unsigned int phy_addr;
+} wmt_phy_addr_t;
+
+#define WMT_CEC_IOC_MAGIC 'c'
+#define WMT_CEC_IOC_MAXNR 3
+#define CECIO_TX_DATA _IOW(WMT_CEC_IOC_MAGIC, 0, wmt_cec_msg_t)
+#define CECIO_TX_LOGADDR _IO(WMT_CEC_IOC_MAGIC, 1)
+#define CECIO_RX_PHYADDR _IOR(WMT_CEC_IOC_MAGIC, 2, wmt_phy_addr_t)
+
+#if defined(__cplusplus)
+}
+#endif
+#endif
diff --git a/drivers/video/wmt/com-mb.h b/drivers/video/wmt/com-mb.h
new file mode 100644
index 00000000..2045a8ea
--- /dev/null
+++ b/drivers/video/wmt/com-mb.h
@@ -0,0 +1,78 @@
+/*++
+ * WonderMedia Memory Block driver
+ *
+ * Copyright c 2010 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef COM_MB_H
+/* To assert that only one occurrence is included */
+#define COM_MB_H
+
+/*-------------------- MODULE DEPENDENCY -------------------------------------*/
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+
+/*------------------------------------------------------------------------------
+ Macros below are used for driver in IOCTL
+------------------------------------------------------------------------------*/
+#define MB_IOC_MAGIC 'M'
+
+/* O: physical address I: size */
+#define MBIO_MALLOC _IOWR(MB_IOC_MAGIC, 0, unsigned long)
+/* O: ummap size I: user address if map, physical address if not map */
+#define MBIO_FREE _IOWR(MB_IOC_MAGIC, 1, unsigned long)
+/* O: ummap size I: user address */
+#define MBIO_UNMAP _IOWR(MB_IOC_MAGIC, 2, unsigned long)
+/* O: mb size I: phys address */
+#define MBIO_MBSIZE _IOWR(MB_IOC_MAGIC, 3, unsigned long)
+/* O: max free mba size */
+#define MBIO_MAX_AVAILABLE_SIZE _IOR(MB_IOC_MAGIC, 4, unsigned long)
+/* advance use only */
+/* I: user address */
+#define MBIO_GET _IOW(MB_IOC_MAGIC, 5, unsigned long)
+/* I: user address */
+#define MBIO_PUT _IOW(MB_IOC_MAGIC, 6, unsigned long)
+/* O: virt address I: user address */
+#define MBIO_USER_TO_VIRT _IOWR(MB_IOC_MAGIC, 7, unsigned long)
+/* O: phys address I: user address */
+#define MBIO_USER_TO_PHYS _IOWR(MB_IOC_MAGIC, 8, unsigned long)
+/* I: size */
+#define MBIO_PREFETCH _IOW(MB_IOC_MAGIC, 9, unsigned long)
+/* O: static mba size */
+#define MBIO_STATIC_SIZE _IOR(MB_IOC_MAGIC, 10, unsigned long)
+/* O: use counter I: physical address */
+#define MBIO_MB_USER_COUNT _IOWR(MB_IOC_MAGIC, 11, unsigned long)
+#define MBIO_FORCE_RESET _IO (MB_IOC_MAGIC,12)
+/* I: phys address */
+#define MBIO_GET_BY_PHYS _IOW(MB_IOC_MAGIC, 13, unsigned long)
+/* I: phys address */
+#define MBIO_PUT_BY_PHYS _IOW(MB_IOC_MAGIC, 14, unsigned long)
+/* I: user address */
+#define MBIO_SYNC_CACHE _IOW(MB_IOC_MAGIC, 15, unsigned long)
+/* O: mb driver version */
+#define MBIO_GET_VERSION _IOR(MB_IOC_MAGIC, 16, unsigned long)
+/* O: property of MB */
+#define MBIO_SET_CACHABLE _IOW(MB_IOC_MAGIC, 17, unsigned long)
+/* O: property of MB */
+#define MBIO_CLR_CACHABLE _IOW(MB_IOC_MAGIC, 18, unsigned long)
+
+#endif /* ifndef COM_MB_H */
+
+/*=== END com-mb.h ==========================================================*/
+
diff --git a/drivers/video/wmt/com-vpp.h b/drivers/video/wmt/com-vpp.h
new file mode 100644
index 00000000..52447ffb
--- /dev/null
+++ b/drivers/video/wmt/com-vpp.h
@@ -0,0 +1,680 @@
+/*++
+ * linux/drivers/video/wmt/com-vpp.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef COM_VPP_H
+#define COM_VPP_H
+
+#ifdef __KERNEL__
+ #include <linux/types.h>
+ #include <linux/fb.h>
+ #include <asm/io.h>
+ #include <mach/common_def.h>
+ #include <mach/com-video.h>
+#else
+ #include "com-video.h"
+ #include "com-fb.h"
+#endif
+
+#define VPP_OLD_API
+
+#define VPP_NEW_FBUF_MANAGER
+
+#define VPP_AHB_CLK 250000000
+
+#ifdef CONFIG_MAX_RESX
+ #define VPP_HD_MAX_RESX CONFIG_MAX_RESX
+#else
+ #define VPP_HD_MAX_RESX 1920
+#endif
+
+#ifdef CONFIG_MAX_RESY
+ #define VPP_HD_MAX_RESY CONFIG_MAX_RESY
+#else
+ #define VPP_HD_MAX_RESY 1200
+#endif
+
+#ifdef CONFIG_DEFAULT_RESX
+ #define VPP_HD_DISP_RESX CONFIG_DEFAULT_RESX
+#else
+ #define VPP_HD_DISP_RESX 1024
+#endif
+
+#ifdef CONFIG_DEFAULT_RESY
+ #define VPP_HD_DISP_RESY CONFIG_DEFAULT_RESY
+#else
+ #define VPP_HD_DISP_RESY 768
+#endif
+
+#ifdef CONFIG_DEFAULT_FPS
+ #define VPP_HD_DISP_FPS CONFIG_DEFAULT_FPS
+#else
+ #define VPP_HD_DISP_FPS 60
+#endif
+
+#define VPP_HD_DISP_PIXCLK 65000000
+
+#define BIT0 0x00000001
+#define BIT1 0x00000002
+#define BIT2 0x00000004
+#define BIT3 0x00000008
+#define BIT4 0x00000010
+#define BIT5 0x00000020
+#define BIT6 0x00000040
+#define BIT7 0x00000080
+#define BIT8 0x00000100
+#define BIT9 0x00000200
+#define BIT10 0x00000400
+#define BIT11 0x00000800
+#define BIT12 0x00001000
+#define BIT13 0x00002000
+#define BIT14 0x00004000
+#define BIT15 0x00008000
+#define BIT16 0x00010000
+#define BIT17 0x00020000
+#define BIT18 0x00040000
+#define BIT19 0x00080000
+#define BIT20 0x00100000
+#define BIT21 0x00200000
+#define BIT22 0x00400000
+#define BIT23 0x00800000
+#define BIT24 0x01000000
+#define BIT25 0x02000000
+#define BIT26 0x04000000
+#define BIT27 0x08000000
+#define BIT28 0x10000000
+#define BIT29 0x20000000
+#define BIT30 0x40000000
+#define BIT31 0x80000000
+
+#define VPP_YUV_BLACK 0x00, 0x80, 0x80 /* Y, Cr, Cb */
+#define VPP_YUV_WHITE 0xff, 0x80, 0x80
+#define VPP_YUV_RED 0x51, 0xf0, 0x5a
+#define VPP_YUV_GREEN 0x90, 0x22, 0x35
+#define VPP_YUV_BLUE 0x28, 0x6d, 0xf0
+#define VPP_RGB32_BLACK 0x00, 0x00, 0x00, 0x00
+
+#define VPP_COL_RGB32_BLACK 0x000000
+#define VPP_COL_RGB32_WHITE 0xFFFFFF
+#define VPP_COL_RGB32_RED 0xFF0000
+#define VPP_COL_RGB32_GREEN 0x00FF00
+#define VPP_COL_RGB32_BLUE 0x0000FF
+
+#define VPP_COL_BLACK 0x008080 /* Y, Cr, Cb */
+#define VPP_COL_WHITE 0xff8080
+#define VPP_COL_RED 0x41d464
+#define VPP_COL_GREEN 0x902235
+#define VPP_COL_BLUE 0x2372d4
+
+#define VPP_MAGNUM(s, e) ((2^((s)-(e)+1))-1)
+
+#define WMT_FB_COLFMT(a) (a & 0xFF)
+
+typedef enum {
+ VPP_FBUF_GOVR_1,
+ VPP_FBUF_GOVR_2,
+ VPP_FBUF_SCLR_1,
+ VPP_FBUF_SCLR_2,
+ VPP_FBUF_MAX
+} vpp_fbuf_t;
+
+typedef enum {
+ VPP_FLAG_NULL = 0,
+ VPP_FLAG_ENABLE = 1,
+ VPP_FLAG_DISABLE = 0,
+ VPP_FLAG_TRUE = 1,
+ VPP_FLAG_FALSE = 0,
+ VPP_FLAG_ZERO = 0,
+ VPP_FLAG_ONE = 1,
+ VPP_FLAG_SUCCESS = 1,
+ VPP_FLAG_ERROR = 0,
+ VPP_FLAG_RD = 1,
+ VPP_FLAG_WR = 0,
+} vpp_flag_t;
+
+typedef enum {
+ VPP_MOD_GOVRH2,
+ VPP_MOD_GOVRH,
+ VPP_MOD_DISP,
+ VPP_MOD_GOVW,
+ VPP_MOD_GOVM,
+ VPP_MOD_SCL,
+ VPP_MOD_SCLW,
+ VPP_MOD_VPU,
+ VPP_MOD_VPUW,
+ VPP_MOD_PIP,
+ VPP_MOD_VPPM,
+ VPP_MOD_LCDC,
+ VPP_MOD_CURSOR,
+ VPP_MOD_MAX
+} vpp_mod_t;
+
+typedef enum {
+ VPP_VOUT_SDA = 0,
+ VPP_VOUT_SDD = 1,
+ VPP_VOUT_LCD = 2,
+ VPP_VOUT_DVI = 3,
+ VPP_VOUT_HDMI = 4,
+ VPP_VOUT_DVO2HDMI = 5,
+ VPP_VOUT_LVDS = 6,
+ VPP_VOUT_VGA = 7,
+ VPP_VOUT_FBDEV = 8,
+ VPP_VOUT_MAX
+} vpp_vout_t;
+
+typedef enum {
+ VPP_OUTDEV_TV_NORMAL, /* NTSC 720x480 or PAL 720x576 */
+ VPP_OUTDEV_VGA,
+ VPP_OUTDEV_DVO,
+ VPP_OUTDEV_MAX
+} vpp_output_device_t;
+
+typedef enum {
+ VPP_ALPHA_VIDEO,
+ VPP_ALPHA_GE,
+ VPP_ALPHA_MAX,
+} vpp_alpha_t;
+
+typedef enum {
+ VPP_DISP_FMT_FRAME, /* Progressive */
+ VPP_DISP_FMT_FIELD, /* Interlace */
+ VPP_DISP_FMT_MAX,
+} vpp_display_format_t;
+
+typedef enum {
+ VPP_MEDIA_FMT_MPEG,
+ VPP_MEDIA_FMT_H264,
+ VPP_MEDIA_FMT_MAX,
+} vpp_media_format_t;
+
+typedef enum {
+ VPP_PATH_NULL = 0,
+ VPP_PATH_ALL = 0xffffffff,
+ /* in */
+ VPP_PATH_GOVM_IN_VPU = BIT0,
+ VPP_PATH_GOVM_IN_SCL = BIT1,
+ VPP_PATH_GOVM_IN_GE = BIT2,
+ VPP_PATH_GOVM_IN_PIP = BIT3,
+ VPP_PATH_GOVM_IN_SPU = BIT4,
+ VPP_PATH_GOVM_IN_CUR = BIT5,
+ /* out */
+ VPP_PATH_SCL_OUT_REALTIME = BIT10,
+ VPP_PATH_SCL_OUT_RECURSIVE = BIT11,
+} vpp_path_t;
+
+typedef enum { /* don't change this order */
+ VPP_CSC_YUV2RGB2_MIN,
+ VPP_CSC_YUV2RGB_SDTV_0_255 = VPP_CSC_YUV2RGB2_MIN,
+ VPP_CSC_YUV2RGB_SDTV_16_235,
+ VPP_CSC_YUV2RGB_HDTV_0_255,
+ VPP_CSC_YUV2RGB_HDTV_16_235,
+ VPP_CSC_YUV2RGB_JFIF_0_255,
+ VPP_CSC_YUV2RGB_SMPTE170M,
+ VPP_CSC_YUV2RGB_SMPTE240M,
+ VPP_CSC_RGB2YUV_MIN,
+ VPP_CSC_RGB2YUV_SDTV_0_255 = VPP_CSC_RGB2YUV_MIN,
+ VPP_CSC_RGB2YUV_SDTV_16_235,
+ VPP_CSC_RGB2YUV_HDTV_0_255,
+ VPP_CSC_RGB2YUV_HDTV_16_235,
+ VPP_CSC_RGB2YUV_JFIF_0_255,
+ VPP_CSC_RGB2YUV_SMPTE170M,
+ VPP_CSC_RGB2YUV_SMPTE240M,
+ VPP_CSC_MAX,
+ VPP_CSC_BYPASS
+} vpp_csc_t;
+
+typedef enum {
+ VPP_REG_LEVEL_1,
+ VPP_REG_LEVEL_2,
+ VPP_REG_LEVEL_MAX,
+} vpp_reglevel_t;
+
+typedef enum {
+ VPP_BPP_1,
+ VPP_BPP_2,
+ VPP_BPP_4,
+ VPP_BPP_8,
+ VPP_BPP_12,
+ VPP_BPP_16_565,
+ VPP_BPP_16_555 = VPP_BPP_16_565,
+ VPP_BPP_18,
+ VPP_BPP_24,
+ VPP_BPP_MAX,
+} vpp_bpp_t;
+
+typedef enum {
+ VPP_GAMMA_DISABLE,
+ VPP_GAMMA_22,
+ VPP_GAMMA_24,
+ VPP_GAMMA_28,
+ VPP_GAMMA_MAX,
+} vpp_gamma_t;
+
+typedef enum {
+ VPP_DATAWIDHT_12,
+ VPP_DATAWIDHT_24,
+ VPP_DATAWIDHT_MAX,
+} vpp_datawidht_t;
+
+typedef enum {
+ VPP_ADDR_NORMAL,
+ VPP_ADDR_BURST,
+ VPP_ADDR_MAX,
+} vpp_addr_mode_t;
+
+typedef enum {
+ VPP_DEI_WEAVE,
+ VPP_DEI_BOB,
+ VPP_DEI_ADAPTIVE_ONE,
+ VPP_DEI_ADAPTIVE_THREE,
+ VPP_DEI_FIELD,
+ VPP_DEI_MOTION_VECTOR,
+ VPP_DEI_MAX,
+ VPP_DEI_DYNAMIC,
+} vpp_deinterlace_t;
+
+typedef enum {
+ VPP_DIR_THERE,
+ VPP_DIR_TWO,
+ VPP_DIR_FOUR,
+ VPP_DIR_FIVE,
+ VPP_DIR_MAX,
+} vpp_direction_t;
+
+typedef enum {
+ VPP_FIELD_TOP,
+ VPP_FIELD_BOTTOM,
+ VPP_FIELD_MAX,
+} vpp_field_t;
+
+typedef enum {
+ VPP_TVSYS_NTSC,
+ VPP_TVSYS_NTSCJ,
+ VPP_TVSYS_NTSC443,
+ VPP_TVSYS_PAL,
+ VPP_TVSYS_PALM,
+ VPP_TVSYS_PAL60,
+ VPP_TVSYS_PALN,
+ VPP_TVSYS_PALNC,
+ VPP_TVSYS_720P,
+ VPP_TVSYS_1080I,
+ VPP_TVSYS_1080P,
+ VPP_TVSYS_MAX
+} vpp_tvsys_t;
+
+typedef enum {
+ VPP_TVCONN_YCBCR,
+ VPP_TVCONN_SCART,
+ VPP_TVCONN_YPBPR,
+ VPP_TVCONN_VGA,
+ VPP_TVCONN_SVIDEO,
+ VPP_TVCONN_CVBS,
+ VPP_TVCONN_MAX
+} vpp_tvconn_t;
+
+#define VPP_OPT_INTERLACE 0x01
+#define VPP_DVO_SYNC_POLAR_HI 0x08
+#define VPP_DVO_VSYNC_POLAR_HI 0x10
+
+typedef struct {
+ unsigned int pixel_clock;
+ unsigned int option;
+
+ unsigned int hsync;
+ unsigned int hbp;
+ unsigned int hpixel;
+ unsigned int hfp;
+
+ unsigned int vsync;
+ unsigned int vbp;
+ unsigned int vpixel;
+ unsigned int vfp;
+} vpp_timing_t;
+
+typedef struct {
+ int read_cycle;
+
+ unsigned int total_pixel_of_line;
+ unsigned int begin_pixel_of_active;
+ unsigned int end_pixel_of_active;
+
+ unsigned int total_line_of_frame;
+ unsigned int begin_line_of_active;
+ unsigned int end_line_of_active;
+
+ unsigned int hsync;
+ unsigned int vsync;
+ unsigned int line_number_between_VBIS_VBIE;
+ unsigned int line_number_between_PVBI_VBIS;
+} vpp_clock_t;
+
+typedef struct _vpp_image_t {
+ U32 y_addr;
+ U32 c_addr;
+ vdo_color_fmt src_col_fmt;
+} vpp_image_t;
+
+typedef struct {
+ vdo_framebuf_t src_fb;
+ vdo_framebuf_t dst_fb;
+} vpp_scale_t;
+
+typedef struct {
+ int mode;
+ vdo_framebuf_t src_fb;
+ vdo_framebuf_t src2_fb;
+ vdo_framebuf_t dst_fb;
+} vpp_scale_overlap_t;
+
+typedef struct {
+ unsigned int alpha_src_type:2; /* 0-mif1,1-mif2,2-fix */
+ unsigned int alpha_src:8;
+ unsigned int alpha_dst_type:2; /* 0-mif1,1-mif2,2-fix */
+ unsigned int alpha_dst:8;
+ unsigned int alpha_swap:1; /* 0-(alpha,1-alpha),1:(1-alpha,alpha) */
+ unsigned int color_key_from:2; /* 0-RMIF1,1-RMIF2 */
+ unsigned int color_key_comp:2; /* 0-888,1-777,2-666,3-555 */
+ unsigned int color_key_mode:3; /* (Non-Hit,Hit):0/1-(alpha,alpha),
+ 2-(alpha,pix1), 3-(pix1,alpha),4-(alpha,pix2),
+ 5-(pix2,alpha),6-(pix1,pix2),7-(pix2,pix1) */
+ unsigned int reserved:4;
+ unsigned int color_key; /* ARGB */
+} vpp_overlap_t;
+
+#define VPP_VOUT_STS_REGISTER 0x01
+#define VPP_VOUT_STS_ACTIVE 0x02
+#define VPP_VOUT_STS_PLUGIN 0x04
+#define VPP_VOUT_STS_EDID 0x08
+#define VPP_VOUT_STS_BLANK 0x10
+#define VPP_VOUT_STS_POWERDN 0x20
+#define VPP_VOUT_STS_CONTENT_PROTECT 0x40
+typedef struct {
+ int num;
+ unsigned int status;
+ char name[10];
+} vpp_vout_info_t;
+
+typedef struct {
+ int num;
+ int arg;
+} vpp_vout_parm_t;
+
+/* =============================================================================
+ vout option
+ =============================================================================
+ vout id option1 option2 - b8(1:blank)
+ SDA 0 tvsys tvconn
+ SDD 1 tvsys tvconn
+ LCD 2 lcd id bit per pixel(bit0-7),
+ b8(1:blank),b9-11(igs mode),
+ bit12-13(swap)
+ DVI 3 colfmt b0(0:12bit,1:24bit),
+ b1(1:interlace),b8(1:blank),
+ b9-11(igs mode),bit12-13(swap)
+ HDMI 4 colfmt b0(0:12bit,1:24bit),
+ b1(1:interlace),b8(1:blank)
+ DVO2HDMI 5 colfmt b0(0:12bit,1:24bit),
+ b1(1:interlace),b8(1:blank)
+ LVDS 6 lcd id bit per pixel(bit0-7),
+ b8(1:blank),b9-11(igs mode),
+ bit12-13(swap)
+ VGA 7
+ FBDEV 8
+ =============================================================================
+ tvsys : 0-NTSC,3-PAL,8-720p,9-1080i,10-1080p
+ tvconn: 0-YCbCr,1-SCART,2-YPbPr,3-VGA,4-SVideo,5-CVBS
+ colfmt: 0-420,1-422H,3-444,6-ARGB,8-RGB24,9-RGB18,10-RGB16
+ lcdid : 0-OEM(VGA 1024x768),1-Chilin(7"800x480),
+ 2-Innolux(7"800x480),3-AUO(800x600),
+ 4-Eking(800x600),5-Hannstar(10"1024x600)
+ ========================================================================== */
+#define VOUT_OPT_DWIDTH24 BIT(0)
+#define VOUT_OPT_INTERLACE BIT(1)
+#define VOUT_OPT_BLANK BIT(8)
+
+typedef struct {
+ int num;
+ unsigned int option[3];
+} vpp_vout_option_t;
+
+typedef struct {
+ int num;
+ vpp_timing_t tmr;
+} vpp_vout_tmr_t;
+
+typedef struct {
+ unsigned int yaddr;
+ unsigned int caddr;
+} vpp_fbaddr_t;
+
+#define VPP_CAP_DUAL_DISPLAY BIT(0)
+typedef struct {
+ unsigned int chip_id;
+ unsigned int version;
+ unsigned int resx_max;
+ unsigned int resy_max;
+ unsigned int pixel_clk;
+ unsigned int module;
+ unsigned int option;
+} vpp_cap_t;
+
+typedef struct {
+ vpp_flag_t enable;
+ vpp_alpha_t mode;
+ int A;
+ int B;
+} vpp_alpha_parm_t;
+
+typedef struct {
+ vpp_path_t src_path;
+ vpp_flag_t enable;
+} vpp_src_path_t;
+
+typedef struct {
+ vpp_flag_t enable;
+ vpp_flag_t sync_polar;
+ vpp_flag_t vsync_polar;
+ vdo_color_fmt color_fmt;
+ vpp_datawidht_t data_w;
+ int clk_inv;
+ int clk_delay;
+} vdo_dvo_parm_t;
+
+typedef struct {
+ unsigned int addr;
+ unsigned int index;
+ unsigned int val;
+} vpp_i2c_t;
+
+#define VPP_FLAG_DISPFB_ADDR BIT(0)
+#define VPP_FLAG_DISPFB_INFO BIT(1)
+#define VPP_FLAG_DISPFB_VIEW BIT(2)
+#define VPP_FLAG_DISPFB_PIP BIT(3)
+#define VPP_FLAG_DISPFB_MB_ONE BIT(4)
+#define VPP_FLAG_DISPFB_MB_NO BIT(6)
+typedef struct {
+ unsigned int yaddr;
+ unsigned int caddr;
+ vdo_framebuf_t info;
+ vdo_view_t view;
+
+ unsigned int flag;
+} vpp_dispfb_t;
+
+typedef struct {
+ vpp_mod_t mod;
+ int read;
+ unsigned int arg1;
+ unsigned int arg2;
+} vpp_mod_arg_t;
+
+typedef struct {
+ vpp_mod_t mod;
+ int read;
+ vpp_timing_t tmr;
+} vpp_mod_timing_t;
+
+typedef struct {
+ vpp_mod_t mod;
+ int read;
+ vdo_framebuf_t fb;
+} vpp_mod_fbinfo_t;
+
+typedef struct {
+ vpp_mod_t mod;
+ int read;
+ vdo_view_t view;
+} vpp_mod_view_t;
+
+typedef struct {
+ unsigned int resx;
+ unsigned int resy;
+ unsigned int fps;
+ unsigned int option;
+} vpp_vmode_parm_t;
+
+#define VPP_VOUT_VMODE_NUM 20
+typedef struct {
+ vpp_vout_t mode;
+ int num;
+ vpp_vmode_parm_t parm[VPP_VOUT_VMODE_NUM];
+} vpp_vout_vmode_t;
+
+typedef struct {
+ int queue_cnt;
+ int cur_cnt;
+ int isr_cnt;
+ int disp_cnt;
+ int skip_cnt;
+ int full_cnt;
+ int reserved[2];
+} vpp_dispfb_info_t;
+
+typedef struct {
+ vpp_vout_t mode;
+ int size;
+ char *buf;
+} vpp_vout_edid_t;
+
+typedef struct {
+ int num;
+ unsigned int bksv[2];
+} vpp_vout_cp_info_t;
+
+#define VPP_VOUT_CP_NUM 336
+typedef struct {
+ char key[VPP_VOUT_CP_NUM];
+} vpp_vout_cp_key_t;
+
+typedef struct {
+ vpp_deinterlace_t mode;
+ int level;
+} vpp_deinterlace_parm_t;
+
+#define VPPIO_MAGIC 'f'
+
+/* VPP common ioctl command */
+#define VPPIO_VPP_BASE 0x0
+#define VPPIO_VPPGET_INFO _IOR(VPPIO_MAGIC, VPPIO_VPP_BASE + 0, vpp_cap_t)
+#define VPPIO_VPPSET_INFO _IOW(VPPIO_MAGIC, VPPIO_VPP_BASE + 0, vpp_cap_t)
+#define VPPIO_I2CSET_BYTE _IOW(VPPIO_MAGIC, VPPIO_VPP_BASE + 1, vpp_i2c_t)
+#define VPPIO_I2CGET_BYTE _IOR(VPPIO_MAGIC, VPPIO_VPP_BASE + 1, vpp_i2c_t)
+#define VPPIO_MODULE_FRAMERATE \
+ _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 2, vpp_mod_arg_t)
+#define VPPIO_MODULE_ENABLE \
+ _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 3, vpp_mod_arg_t)
+#define VPPIO_MODULE_TIMING \
+ _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 4, vpp_mod_timing_t)
+#define VPPIO_MODULE_FBADDR \
+ _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 5, vpp_mod_arg_t)
+#define VPPIO_MODULE_FBINFO \
+ _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 6, vpp_mod_fbinfo_t)
+/* #define VPPIO_VPPSET_DIRECTPATH _IO(VPPIO_MAGIC,VPPIO_VPP_BASE + 7) */
+#define VPPIO_VPPSET_FBDISP _IOW(VPPIO_MAGIC, VPPIO_VPP_BASE + 8, vpp_dispfb_t)
+#define VPPIO_VPPGET_FBDISP \
+ _IOW(VPPIO_MAGIC, VPPIO_VPP_BASE + 8, vpp_dispfb_info_t)
+#define VPPIO_WAIT_FRAME _IO(VPPIO_MAGIC, VPPIO_VPP_BASE + 9)
+#define VPPIO_MODULE_VIEW \
+ _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 10, vpp_mod_view_t)
+#define VPPIO_STREAM_ENABLE _IO(VPPIO_MAGIC, VPPIO_VPP_BASE + 12)
+#define VPPIO_STREAM_GETFB \
+ _IOR(VPPIO_MAGIC, VPPIO_VPP_BASE + 13, vdo_framebuf_t)
+#define VPPIO_STREAM_PUTFB \
+ _IOW(VPPIO_MAGIC, VPPIO_VPP_BASE + 13, vdo_framebuf_t)
+#define VPPIO_MODULE_CSC _IOWR(VPPIO_MAGIC, VPPIO_VPP_BASE + 14, vpp_mod_arg_t)
+#define VPPIO_MULTIVD_ENABLE _IO(VPPIO_MAGIC, VPPIO_VPP_BASE + 15)
+
+/* VOUT ioctl command */
+#define VPPIO_VOUT_BASE 0x10
+#define VPPIO_VOGET_INFO _IOR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 0, vpp_vout_info_t)
+#define VPPIO_VOSET_MODE \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 1, vpp_vout_parm_t)
+#define VPPIO_VOSET_BLANK \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 2, vpp_vout_parm_t)
+/* #define VPPIO_VOSET_DACSENSE \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 3, vpp_vout_parm_t) */
+#define VPPIO_VOSET_BRIGHTNESS \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 4, vpp_vout_parm_t)
+#define VPPIO_VOGET_BRIGHTNESS \
+ _IOR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 4, vpp_vout_parm_t)
+#define VPPIO_VOSET_CONTRAST \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 5, vpp_vout_parm_t)
+#define VPPIO_VOGET_CONTRAST \
+ _IOR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 5, vpp_vout_parm_t)
+#define VPPIO_VOSET_OPTION \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 6, vpp_vout_option_t)
+#define VPPIO_VOGET_OPTION \
+ _IOR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 6, vpp_vout_option_t)
+#define VPPIO_VOUT_VMODE \
+ _IOWR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 7, vpp_vout_vmode_t)
+#define VPPIO_VOGET_EDID _IOR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 8, vpp_vout_edid_t)
+#define VPPIO_VOGET_CP_INFO \
+ _IOR(VPPIO_MAGIC, VPPIO_VOUT_BASE + 9, vpp_vout_cp_info_t)
+#define VPPIO_VOSET_CP_KEY \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 10, vpp_vout_cp_key_t)
+#define VPPIO_VOSET_AUDIO_PASSTHRU _IO(VPPIO_MAGIC, VPPIO_VOUT_BASE + 11)
+/* #define VPPIO_VOSET_OVERSCAN_OFFSET \
+ _IOW(VPPIO_MAGIC, VPPIO_VOUT_BASE + 12, vpp_vout_overscan_offset_t) */
+#define VPPIO_VOSET_VIRTUAL_FBDEV _IO(VPPIO_MAGIC, VPPIO_VOUT_BASE + 13)
+
+/* GOVR ioctl command */
+#define VPPIO_GOVR_BASE 0x20
+#define VPPIO_GOVRSET_DVO _IOW(VPPIO_MAGIC, VPPIO_GOVR_BASE + 0, vdo_dvo_parm_t)
+#define VPPIO_GOVRSET_CUR_COLKEY \
+ _IOW(VPPIO_MAGIC, VPPIO_GOVR_BASE + 1, vpp_mod_arg_t)
+#define VPPIO_GOVRSET_CUR_HOTSPOT \
+ _IOW(VPPIO_MAGIC, VPPIO_GOVR_BASE + 2, vpp_mod_arg_t)
+
+/* SCL ioctl command */
+#define VPPIO_SCL_BASE 0x60
+#define VPPIO_SCL_SCALE _IOWR(VPPIO_MAGIC, VPPIO_SCL_BASE + 0, vpp_scale_t)
+#define VPPIO_SCL_DROP_LINE_ENABLE _IO(VPPIO_MAGIC, VPPIO_SCL_BASE + 1)
+#define VPPIO_SCL_SCALE_ASYNC \
+ _IOWR(VPPIO_MAGIC, VPPIO_SCL_BASE + 2, vpp_scale_t)
+#define VPPIO_SCL_SCALE_FINISH _IO(VPPIO_MAGIC, VPPIO_SCL_BASE + 3)
+#define VPPIO_SCLSET_OVERLAP \
+ _IOW(VPPIO_MAGIC, VPPIO_SCL_BASE + 4, vpp_overlap_t)
+#define VPPIO_SCL_SCALE_OVERLAP \
+ _IOWR(VPPIO_MAGIC, VPPIO_SCL_BASE + 5, vpp_scale_overlap_t)
+
+#define VPPIO_MAX 0x70
+#endif /* COM_VPP_H */
diff --git a/drivers/video/wmt/dev-cec.c b/drivers/video/wmt/dev-cec.c
new file mode 100644
index 00000000..f03b202b
--- /dev/null
+++ b/drivers/video/wmt/dev-cec.c
@@ -0,0 +1,488 @@
+/*++
+ * linux/drivers/video/wmt/dev-cec.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define DEV_CEC_C
+
+/* #define DEBUG */
+#include <linux/platform_device.h>
+#include <linux/major.h>
+#include <linux/cdev.h>
+#include <linux/netlink.h>
+#include <net/sock.h>
+#include "cec.h"
+
+#define DRIVER_NAME "wmt-cec"
+
+#define TXNORSP 0x8000 /* Tx no response interrupt */
+#define TXLSARB 0x2000 /* Tx lose arbitration */
+#define TXDONE 0x0100 /* Tx done */
+#define RXFAIL 0x0002 /* Rx fail, details in Rx10 */
+#define RXDONE 0x0001 /* Rx done */
+
+#define MAX_RETRY 3
+#define MAX_TIMEOUT 5000
+
+static int tx_state;
+
+static struct cdev *wmt_cec_cdev;
+static DEFINE_SEMAPHORE(wmt_cec_sem);
+
+struct wmt_cec_msg recv_msg;
+static void wmt_cec_do_rx_notify(struct work_struct *ptr)
+{
+ vpp_netlink_notify(USER_PID, DEVICE_RX_DATA, (int)&recv_msg);
+}
+DECLARE_DELAYED_WORK(wmt_cec_rx_work, wmt_cec_do_rx_notify);
+
+
+/* receive message interrupt handling */
+static void wmt_cec_do_recv(void)
+{
+ memset(&recv_msg, 0, sizeof(recv_msg));
+ /* get received data */
+ recv_msg.msglen = wmt_cec_rx_data(recv_msg.msgdata);
+ if (recv_msg.msglen > MAX_MSG_BYTE)
+ recv_msg.msglen = MAX_MSG_BYTE;
+
+ DBGMSG("read a received byte! msglen: %d\n", recv_msg.msglen);
+
+ /* clear receive blockage */
+
+ /* notify AP the received message, let AP decide what to response */
+ schedule_delayed_work(&wmt_cec_rx_work, HZ / 10);
+}
+
+/* check if logic address is valid */
+static int bvalidaddr(char addr)
+{
+ if (addr > 15)
+ return 0;
+ return 1;
+}
+
+/* make sure cec line is not busy */
+static int tx_get_cecline(void)
+{
+ int timeout = 400;
+
+ while (timeout-- > 0) {
+ /* if not busy */
+ if (1)
+ return 0;
+ msleep(1);
+ }
+ return -ETIMEDOUT;
+}
+
+/* transfer a time*/
+DECLARE_COMPLETION(txcomp);
+static int wmt_cec_do_xfer_one(char *msgdata, int msglen)
+{
+ int ret;
+ unsigned long jiffies;
+
+ ret = tx_get_cecline();
+ if (ret != 0)
+ ret = -EAGAIN;
+
+ wmt_cec_tx_data(msgdata, msglen);
+
+ jiffies = msecs_to_jiffies((unsigned int)MAX_TIMEOUT);
+ jiffies = wait_for_completion_timeout(&txcomp, jiffies);
+ if (jiffies == 0)
+ return -EAGAIN;
+
+ if (tx_state == TXLSARB)
+ return -EAGAIN;
+
+ return tx_state;
+}
+
+/* transfer a message, including retransmission if needed*/
+static int wmt_cec_do_xfer(char *msgdata, int msglen)
+{
+ int retry;
+ int ret;
+ char srcaddr, tgtaddr;
+
+ if (msglen < 1) {
+ DPRINT("[CEC] xfer: invalid message, msglen is less than 1.\n");
+ return -1;
+ }
+
+ srcaddr = (msgdata[0] & 0xf0) >> 4;
+ tgtaddr = (msgdata[0] & 0x0f);
+ if (!bvalidaddr(srcaddr) || !bvalidaddr(tgtaddr)) {
+ DPRINT("[CEC] xfer: invalid logic address in msg data.\n");
+ return -1;
+ }
+
+ for (retry = 0; retry < MAX_RETRY; retry++) {
+ ret = wmt_cec_do_xfer_one(msgdata, msglen);
+ if (ret != -EAGAIN)
+ goto out;
+ DPRINT("[CEC] Retrying transmission (%d)\n", retry);
+ udelay(100);
+ }
+ return -EAGAIN;
+out:
+ /* if polling message ret is no-response, set logical address register,
+ and enable slave mode */
+ if (srcaddr == tgtaddr && msglen == 1) {
+ if (ret == TXNORSP) {
+ DBGMSG("[CEC] logic addr register is 0x%x\n", tgtaddr);
+ return 0;
+ } else
+ return -1;
+ } else if (ret == TXDONE)
+ return 0;
+ else
+ return -1;
+}
+
+/* irq handling function */
+irqreturn_t wmt_cec_do_irq(int irq, void *dev_id)
+{
+ int sts;
+
+ sts = wmt_cec_get_int();
+ wmt_cec_clr_int(sts);
+ if (sts & BIT0) { /* tx done */
+ DBGMSG("[CEC] write ok int\n");
+ complete(&txcomp);
+ tx_state = TXDONE;
+ }
+ if (sts & BIT1) { /* rx done */
+ DBGMSG("[CEC] read ok int\n");
+ wmt_cec_do_recv();
+ }
+ if (sts & BIT2) { /* rx error */
+ DBGMSG("[CEC] read error int\n");
+ }
+ if (sts & BIT3) { /* tx arb fail */
+ DBGMSG("[CEC] wr arb fail int\n");
+ complete(&txcomp);
+ tx_state = TXLSARB;
+ }
+ if (sts & BIT4) { /* tx no ack */
+/* DBGMSG("[CEC] write no ack int(addr not match)\n"); */
+ complete(&txcomp);
+ tx_state = TXNORSP;
+ }
+ return IRQ_HANDLED;
+}
+
+static int wmt_cec_open(
+ struct inode *inode,
+ struct file *filp
+)
+{
+ int ret = 0;
+
+ DBGMSG("[CEC] wmt_cec_open\n");
+
+ down(&wmt_cec_sem);
+ wmt_cec_rx_enable(1);
+ up(&wmt_cec_sem);
+ return ret;
+} /* End of videodecoder_open() */
+
+static int wmt_cec_release(
+ struct inode *inode,
+ struct file *filp
+)
+{
+ int ret = 0;
+
+ DBGMSG("[CEC] wmt_cec_release\n");
+
+ down(&wmt_cec_sem);
+ wmt_cec_rx_enable(0);
+ up(&wmt_cec_sem);
+ return ret;
+} /* End of videodecoder_release() */
+
+static long wmt_cec_ioctl(
+ struct file *filp,
+ unsigned int cmd,
+ unsigned long arg
+)
+{
+ int ret = -EINVAL;
+
+ DBGMSG("[CEC] wmt_cec_ioctl 0x%x,0x%x\n", cmd, arg);
+
+ /* check type and number, if fail return ENOTTY */
+ if (_IOC_TYPE(cmd) != WMT_CEC_IOC_MAGIC)
+ return -ENOTTY;
+ if (_IOC_NR(cmd) > WMT_CEC_IOC_MAXNR)
+ return -ENOTTY;
+
+ /* check argument area */
+ if (_IOC_DIR(cmd) & _IOC_READ)
+ ret = !access_ok(VERIFY_WRITE,
+ (void __user *) arg, _IOC_SIZE(cmd));
+ else if (_IOC_DIR(cmd) & _IOC_WRITE)
+ ret = !access_ok(VERIFY_READ,
+ (void __user *) arg, _IOC_SIZE(cmd));
+ else
+ ret = 0;
+
+ if (ret)
+ return -EFAULT;
+
+ down(&wmt_cec_sem);
+ switch (cmd) {
+ case CECIO_TX_DATA:
+ {
+ struct wmt_cec_msg msg;
+
+ ret = copy_from_user((void *) &msg, (const void *)arg,
+ sizeof(struct wmt_cec_msg));
+ wmt_cec_do_xfer(msg.msgdata, msg.msglen);
+ }
+ break;
+ case CECIO_TX_LOGADDR:
+ wmt_cec_set_logical_addr((arg & 0xFF00) >> 8, arg & 0xFF, 1);
+ break;
+ case CECIO_RX_PHYADDR:
+ {
+ wmt_phy_addr_t parm;
+
+ parm.phy_addr = edid_get_hdmi_phy_addr();
+ ret = copy_to_user((void *)arg, (void *)&parm,
+ sizeof(wmt_phy_addr_t));
+ }
+ break;
+ default:
+ DBGMSG("[CEC] *W* wmt_cec_ioctl cmd 0x%x\n", cmd);
+ break;
+ }
+ up(&wmt_cec_sem);
+ return ret;
+} /* End of videodecoder_ioctl() */
+
+static int wmt_cec_mmap(
+ struct file *filp,
+ struct vm_area_struct *vma
+)
+{
+ int ret = -EINVAL;
+ down(&wmt_cec_sem);
+ up(&wmt_cec_sem);
+ return ret;
+}
+
+static ssize_t wmt_cec_read(
+ struct file *filp,
+ char __user *buf,
+ size_t count,
+ loff_t *f_pos
+)
+{
+ ssize_t ret = 0;
+ down(&wmt_cec_sem);
+ up(&wmt_cec_sem);
+ return ret;
+} /* videodecoder_read */
+
+static ssize_t wmt_cec_write(
+ struct file *filp,
+ const char __user *buf,
+ size_t count,
+ loff_t *f_pos
+)
+{
+ ssize_t ret = 0;
+ down(&wmt_cec_sem);
+ up(&wmt_cec_sem);
+ return ret;
+} /* End of videodecoder_write() */
+
+struct file_operations wmt_cec_fops = {
+ .owner = THIS_MODULE,
+ .open = wmt_cec_open,
+ .release = wmt_cec_release,
+ .read = wmt_cec_read,
+ .write = wmt_cec_write,
+ .unlocked_ioctl = wmt_cec_ioctl,
+ .mmap = wmt_cec_mmap,
+};
+
+static int __init wmt_cec_probe
+(
+ struct platform_device *dev /*!<; // a pointer to struct device */
+)
+{
+ dev_t dev_no;
+ int ret;
+
+ DBGMSG("[CEC] Enter wmt_cec_probe\n");
+
+ wmt_cec_init_hw();
+ wmt_cec_set_logical_addr(4, 0xf, 1); /* default set boardcast address */
+ wmt_cec_rx_enable(0);
+ /* wmt_cec_enable_loopback(1); */
+
+ dev_no = MKDEV(CEC_MAJOR, 0);
+ ret = register_chrdev_region(dev_no, 1, "wmtcec");
+ if (ret < 0) {
+ DPRINT("can't get %s dev major %d\n", DRIVER_NAME, CEC_MAJOR);
+ return ret;
+ }
+
+ /* register char device */
+ wmt_cec_cdev = cdev_alloc();
+ if (!wmt_cec_cdev) {
+ DPRINT("alloc dev error.\n");
+ return -ENOMEM;
+ }
+
+ cdev_init(wmt_cec_cdev, &wmt_cec_fops);
+ ret = cdev_add(wmt_cec_cdev, dev_no, 1);
+ if (ret) {
+ DPRINT("reg char dev error(%d).\n", ret);
+ cdev_del(wmt_cec_cdev);
+ return ret;
+ }
+
+ if (vpp_request_irq(IRQ_VPP_IRQ20, wmt_cec_do_irq,
+ SA_INTERRUPT, "cec", (void *) 0))
+ DPRINT("*E* request CEC ISR fail\n");
+ vppif_reg32_out(REG_CEC_INT_ENABLE, 0x1f);
+
+ DBGMSG("[CEC] Exit wmt_cec_probe(0x%x)\n", dev_no);
+ return 0;
+} /* End of wmt_cec_probe */
+
+static int wmt_cec_remove
+(
+ struct platform_device *dev /*!<; // a pointer point to struct device */
+)
+{
+ return 0;
+} /* End of wmt_cec_remove */
+
+#ifdef CONFIG_PM
+static int wmt_cec_suspend
+(
+ struct platform_device *pDev, /*!<; // a pointer to struct device */
+ pm_message_t state /*!<; // suspend state */
+)
+{
+ DPRINT("Enter wmt_cec_suspend\n");
+ wmt_cec_do_suspend();
+ return 0;
+} /* End of wmt_cec_suspend */
+
+static int wmt_cec_resume
+(
+ struct platform_device *pDev /*!<; // a pointer to struct device */
+)
+{
+ DPRINT("Enter wmt_cec_resume\n");
+ wmt_cec_do_resume();
+ return 0;
+} /* End of wmt_cec_resume */
+#else
+#define wmt_cec_suspend NULL
+#define wmt_cec_resume NULL
+#endif
+
+/***************************************************************************
+ device driver struct define
+****************************************************************************/
+static struct platform_driver wmt_cec_driver = {
+ .driver.name = "wmtcec", /* equal to platform device name. */
+ .driver.bus = &platform_bus_type,
+ .probe = wmt_cec_probe,
+ .remove = wmt_cec_remove,
+ .suspend = wmt_cec_suspend,
+ .resume = wmt_cec_resume,
+};
+
+/***************************************************************************
+ platform device struct define
+****************************************************************************/
+static u64 wmt_cec_dma_mask = 0xffffffffUL;
+static struct platform_device wmt_cec_device = {
+ .name = "wmtcec",
+ .dev = {
+ .dma_mask = &wmt_cec_dma_mask,
+ .coherent_dma_mask = ~0,
+ },
+#if 0
+ .id = 0,
+ .dev = {
+ .release = wmt_cec_platform_release,
+ },
+ .num_resources = 0, /* ARRAY_SIZE(wmt_cec_resources), */
+ .resource = NULL, /* wmt_cec_resources, */
+#endif
+};
+
+static int __init wmt_cec_init(void)
+{
+ int ret;
+ char buf[100];
+ int varlen = 100;
+ unsigned int cec_enable = 0;
+
+ if (wmt_getsyspara("wmt.display.cec", buf, &varlen) == 0)
+ vpp_parse_param(buf, &cec_enable, 1, 0);
+
+ if (cec_enable == 0)
+ return 0;
+
+ DBGMSG(KERN_ALERT "Enter wmt_cec_init\n");
+
+ ret = platform_driver_register(&wmt_cec_driver);
+ if (!ret) {
+ ret = platform_device_register(&wmt_cec_device);
+ if (ret)
+ platform_driver_unregister(&wmt_cec_driver);
+ }
+ return ret;
+}
+
+static void __exit wmt_cec_exit(void)
+{
+ dev_t dev_no;
+
+ DBGMSG(KERN_ALERT "Enter wmt_cec_exit\n");
+
+ platform_driver_unregister(&wmt_cec_driver);
+ platform_device_unregister(&wmt_cec_device);
+ dev_no = MKDEV(CEC_MAJOR, 0);
+ unregister_chrdev_region(dev_no, 1);
+ return;
+}
+
+module_init(wmt_cec_init);
+module_exit(wmt_cec_exit);
+
+MODULE_LICENSE("GPL");
+MODULE_DESCRIPTION("WMT CEC driver");
+MODULE_AUTHOR("WMT TECH");
+MODULE_VERSION("1.0.0");
+
diff --git a/drivers/video/wmt/devices/Makefile b/drivers/video/wmt/devices/Makefile
new file mode 100644
index 00000000..955c31ec
--- /dev/null
+++ b/drivers/video/wmt/devices/Makefile
@@ -0,0 +1,20 @@
+#
+# Makefile for the Wonder Media framebuffer driver
+#
+
+# wmt external video device
+obj-$(CONFIG_LCD_WMT) += lcd-oem.o
+obj-$(CONFIG_LCD_CHILIN_LW0700AT9003) += lcd-CHILIN-LW700at9003.o
+obj-$(CONFIG_LCD_INNOLUX_AT070TN83) += lcd-INNOLUX-AT070TN83.o
+obj-$(CONFIG_LCD_AUO_A080SN01) += lcd-AUO-A080SN01.o
+obj-$(CONFIG_LCD_EKING_EK08009) += lcd-EKING-EK08009-70135.o
+obj-$(CONFIG_LCD_HANNSTAR_HSD101PFW2) += lcd-HANNSTAR-HSD101PFW2.o
+obj-y += lcd-lvds-1024x600.o
+obj-y += cs8556.o
+obj-y += lcd-b079xan01.o
+
+obj-$(CONFIG_LCD_WMT) += lcd.o
+obj-$(CONFIG_DVI_VT1632_WMT) += vt1632.o vt1625.o
+#obj-$(CONFIG_HDMI_CAT6610_WMT) += cat6612.o cat6610/
+#obj-$(CONFIG_HDMI_AD9389_WMT) += ad9389.o
+#obj-$(CONFIG_EXTTV_ADV7393_WMT) += ad7393.o
diff --git a/drivers/video/wmt/devices/cs8556.c b/drivers/video/wmt/devices/cs8556.c
new file mode 100644
index 00000000..c44fb336
--- /dev/null
+++ b/drivers/video/wmt/devices/cs8556.c
@@ -0,0 +1,617 @@
+/*++
+ * linux/drivers/video/wmt/cs8556.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+/********************************************************************************
+* REVISON HISTORY
+*
+* VERSION | DATE | AUTHORS | DESCRIPTION
+* 1.0 | 2013/08/24 | Howay Huo | First Release
+*******************************************************************************/
+
+#define CS8556_C
+// #define DEBUG
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include <linux/i2c.h>
+#include <mach/hardware.h>
+#include <linux/irq.h>
+#include <linux/interrupt.h>
+#include <linux/gpio.h>
+#include <mach/wmt_iomux.h>
+#include "../vout.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define VT1632_XXXX xxxx *//*Example*/
+//#define CONFIG_CS8556_INTERRUPT
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define VT1632_XXXX 1 *//*Example*/
+#define CS8556_ADDR 0x3d
+#define CS8556_NAME "CS8556"
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx vt1632_xxx_t; *//*Example*/
+typedef struct {
+ unsigned int flag;
+ unsigned int gpiono;
+ unsigned int act;
+}avdetect_gpio_t;
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in vt1632.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int vt1632_xxx; *//*Example*/
+static int s_cs8556_ready;
+static int s_cs8556_init;
+static struct i2c_client *s_cs8556_client;
+static int s_irq_init;
+static avdetect_gpio_t s_avdetect_gpio = {0, WMT_PIN_GP0_GPIO5, 1};
+
+static unsigned char s_CS8556_Original_Offset0[]={
+ 0xF0,0x7F,0x00,0x00,0x00,0x00,0x80,0x80,0x80,0x00,0x00,0x02,0x01,0x00,0x00,0x01,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00
+};
+
+static unsigned char s_RGB888_To_PAL_Offset0[]={
+ 0x01,0x80,0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x02,0x02,0x00,0x00,0x00,0x00,
+ 0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x5F,0x03,0x3F,0x00,0x7D,0x00,0x53,0x03,0x00,0x00,0x00,0x10,0x00,0x00,0x00,0x00,
+ 0x70,0x02,0x04,0x00,0x2E,0x00,0x62,0x02,0x00,0x00,0x84,0x00,0x2B,0x00,0x36,0x00,
+ 0x12,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xBF,0x06,0x7F,0x00,0xFE,0x00,0xA4,0x06,0x00,0x00,0x2D,0x11,0x3C,0x01,0x3A,0x01,
+ 0x70,0x02,0x04,0x00,0x12,0x00,0x34,0x01,0x00,0x00,0x70,0x70,0x70,0x00,0x00,0x00,
+ 0x22,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x04,0x41,0x18,0x09,0x00,0x00,0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xFF,0xFF,0x00,0x00,0xFF,0xFF,0x00,0x00,0x80,0x80,0x00,0x00,0x80,0x80,0x00,0x00,
+ 0xFF,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,
+ 0xFF,0x00,0xFF,0x00,0xFF,0x00,0xFF,0x00,0x80,0x00,0x80,0x00,0x80,0x00,0x80,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x01,0x24,0x1A,0x00,0x01,0x03,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x04,0x01,0xA4,0x06,0x0B,0x00,0x07,0x01,0xF0,0x00,0x00,0x00,0x00,0x04,0x40,0x01
+};
+
+static unsigned char s_RGB888_To_NTSC_Offset0[]={
+ 0x01,0x80,0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x02,0x02,0x00,0x00,0x00,0x00,
+ 0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x59,0x03,0x3D,0x00,0x7E,0x00,0x49,0x03,0x00,0x00,0x00,0x10,0x00,0x00,0x00,0x00,
+ 0x0C,0x02,0x05,0x00,0x21,0x00,0x03,0x02,0x00,0x00,0x7A,0x00,0x23,0x00,0x16,0x00,
+ 0x12,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xB3,0x06,0x7F,0x00,0x00,0x01,0xA4,0x06,0x00,0x00,0x05,0x50,0x00,0x01,0x07,0x01,
+ 0x0C,0x02,0x02,0x00,0x12,0x00,0x07,0x01,0x00,0x00,0x70,0x70,0x70,0x00,0x00,0x00,
+ 0x22,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x04,0x41,0x18,0x09,0x00,0x00,0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xFF,0xFF,0x00,0x00,0xFF,0xFF,0x00,0x00,0x80,0x80,0x00,0x00,0x80,0x80,0x00,0x00,
+ 0xFF,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,
+ 0xFF,0x00,0xFF,0x00,0xFF,0x00,0xFF,0x00,0x80,0x00,0x80,0x00,0x80,0x00,0x80,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x01,0x24,0x1A,0x00,0x01,0x03,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x04,0x01,0xA4,0x06,0x0B,0x00,0x07,0x01,0xF0,0x00,0x00,0x00,0x00,0x04,0x00,0x00
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void vt1632_xxx(void); *//*Example*/
+static int I2CMultiPageRead(unsigned char maddr, unsigned char page, unsigned char saddr, int number, unsigned char *value)
+{
+ int ret;
+ unsigned char wbuf[2];
+ struct i2c_msg rd[2];
+
+ wbuf[0] = page;
+ wbuf[1] = saddr;
+
+ rd[0].addr = maddr;
+ rd[0].flags = 0;
+ rd[0].len = 2;
+ rd[0].buf = wbuf;
+
+ rd[1].addr = maddr;
+ rd[1].flags = I2C_M_RD;
+ rd[1].len = number;
+ rd[1].buf = value;
+
+ ret = i2c_transfer(s_cs8556_client->adapter, rd, ARRAY_SIZE(rd));
+
+ if (ret != ARRAY_SIZE(rd)) {
+ DBG_ERR("fail\n");
+ return -1;
+ }
+
+ return 0;
+}
+
+static int I2CMultiPageWrite(unsigned char maddr, unsigned char page, unsigned char saddr, int number, unsigned char *value)
+{
+ int ret;
+ unsigned char *pbuf;
+ struct i2c_msg wr[1];
+
+ pbuf = kmalloc(number + 2, GFP_KERNEL);
+ if(!pbuf) {
+ DBG_ERR("alloc memory fail\n");
+ return -1;
+ }
+
+ *pbuf = page;
+ *(pbuf + 1) = saddr;
+
+ memcpy(pbuf + 2, value, number);
+
+ wr[0].addr = maddr;
+ wr[0].flags = 0;
+ wr[0].len = number + 2;
+ wr[0].buf = pbuf;
+
+ ret = i2c_transfer(s_cs8556_client->adapter, wr, ARRAY_SIZE(wr));
+
+ if (ret != ARRAY_SIZE(wr)) {
+ DBG_ERR("fail\n");
+ kfree(pbuf);
+ return -1;
+ }
+
+ kfree(pbuf);
+ return 0 ;
+}
+
+/************************** i2c device struct definition **************************/
+static int __devinit cs8556_i2c_probe(struct i2c_client *i2c,
+ const struct i2c_device_id *id)
+{
+ DBGMSG("cs8556_i2c_probe\n");
+
+ return 0;
+}
+
+static int __devexit cs8556_i2c_remove(struct i2c_client *client)
+{
+ DBGMSG("cs8556_i2c_remove\n");
+
+ return 0;
+}
+
+
+static const struct i2c_device_id cs8556_i2c_id[] = {
+ {CS8556_NAME, 0},
+ { },
+};
+MODULE_DEVICE_TABLE(i2c, cs8556_i2c_id);
+
+static struct i2c_board_info __initdata cs8556_i2c_board_info[] = {
+ {
+ I2C_BOARD_INFO(CS8556_NAME, CS8556_ADDR),
+ },
+};
+
+static struct i2c_driver cs8556_i2c_driver = {
+ .driver = {
+ .name = CS8556_NAME,
+ .owner = THIS_MODULE,
+ },
+ .probe = cs8556_i2c_probe,
+ .remove = __devexit_p(cs8556_i2c_remove),
+ .id_table = cs8556_i2c_id,
+};
+
+/*----------------------- Function Body --------------------------------------*/
+static void avdetect_irq_enable(void)
+{
+ wmt_gpio_unmask_irq(s_avdetect_gpio.gpiono);
+}
+
+static void avdetect_irq_disable(void)
+{
+ wmt_gpio_mask_irq(s_avdetect_gpio.gpiono);
+}
+
+int avdetect_irq_hw_init(int resume)
+{
+ int ret;
+
+ if(!resume) {
+ ret = gpio_request(s_avdetect_gpio.gpiono, "avdetect irq"); //enable gpio
+ if(ret < 0) {
+ DBG_ERR("gpio(%d) request fail for avdetect irq\n", s_avdetect_gpio.gpiono);
+ return ret;
+ }
+ }else
+ gpio_re_enabled(s_avdetect_gpio.gpiono); //re-enable gpio
+
+ gpio_direction_input(s_avdetect_gpio.gpiono); //gpio input
+
+ wmt_gpio_setpull(s_avdetect_gpio.gpiono, WMT_GPIO_PULL_UP); //enable pull and pull-up
+
+ wmt_gpio_mask_irq(s_avdetect_gpio.gpiono); //disable interrupt
+
+ wmt_gpio_set_irq_type(s_avdetect_gpio.gpiono, IRQ_TYPE_EDGE_BOTH); //rise edge and clear interrupt
+
+ return 0;
+}
+
+/*
+static void avdetect_irq_hw_free(void)
+{
+ gpio_free(AVDETECT_IRQ_PIN);
+
+}
+*/
+
+static irqreturn_t avdetect_irq_handler(int irq, void *dev_id)
+{
+ //printk("avdetect_irq_handler\n");
+
+ if(!gpio_irqstatus(s_avdetect_gpio.gpiono))
+ return IRQ_NONE;
+
+ wmt_gpio_ack_irq(s_avdetect_gpio.gpiono); //clear interrupt
+
+ //printk("cvbs hotplug interrupt\n");
+ if(!is_gpio_irqenable(s_avdetect_gpio.gpiono)) {
+ //pr_err("avdetect irq is disabled\n");
+ return IRQ_HANDLED;
+ }else
+ return IRQ_WAKE_THREAD;
+}
+
+static irqreturn_t avdetect_irq_thread(int irq, void *dev)
+{
+ //printk(cvbs_hotplug_irq_thread\n");
+
+ if(s_avdetect_gpio.act == 1) {
+ if(gpio_get_value(s_avdetect_gpio.gpiono))
+ printk("av plug in\n");
+ else
+ printk("av plug out\n");
+ }
+ else {
+ if(gpio_get_value(s_avdetect_gpio.gpiono))
+ printk("av plug out\n");
+ else
+ printk("av plug in\n");
+ }
+
+ return IRQ_HANDLED;
+}
+
+void cs8556_set_tv_mode(int ntsc)
+{
+ int ret;
+ unsigned char rbuf[256] = {0};
+
+ if( !s_cs8556_ready )
+ return;
+
+ ret = I2CMultiPageRead(CS8556_ADDR, 0x00, 0x00, 256, rbuf);
+ if(ret) {
+ DBG_ERR("I2C read Offset0 fail\n");
+ return;
+ }
+
+ if(ntsc < 0) {
+ if(memcmp(rbuf, s_CS8556_Original_Offset0, 0x11) != 0) {
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 256, s_CS8556_Original_Offset0);
+ if(ret)
+ DBG_ERR("I2C write Original_Offset0 fail\n");
+ else {
+ if(s_irq_init)
+ avdetect_irq_disable();
+ }
+ }
+
+ return;
+ }
+
+ if(ntsc) {
+ if(memcmp(rbuf, s_RGB888_To_NTSC_Offset0, 0x50) !=0) {
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 256, s_RGB888_To_NTSC_Offset0);
+ if(ret)
+ DBG_ERR("I2C write NTSC_Offset0 fail\n");
+ }
+ } else {
+ if(memcmp(rbuf, s_RGB888_To_PAL_Offset0, 0x50) != 0) {
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 256, s_RGB888_To_PAL_Offset0);
+ if(ret)
+ DBG_ERR("I2C write PAL_Offset0 fail\n");
+ }
+ }
+
+}
+
+int cs8556_check_plugin(int hotplug)
+{
+ return 1;
+}
+
+int cs8556_init(struct vout_s *vo)
+{
+ int ret;
+ char buf[40] = {0};
+ int varlen = 40;
+ int no = 1; //default i2c1
+ int num;
+ unsigned char rbuf[256] = {0};
+ struct i2c_adapter *adapter = NULL;
+ vout_tvformat_t tvformat = TV_MAX;
+
+ DPRINT("cs8556_init\n");
+
+ if(wmt_getsyspara("wmt.display.tvformat", buf, &varlen) == 0) {
+ if(!strnicmp(buf, "PAL", 3))
+ tvformat = TV_PAL;
+ else if(!strnicmp(buf, "NTSC", 4))
+ tvformat = TV_NTSC;
+ else
+ tvformat = TV_UNDEFINED;
+ } else
+ tvformat = TV_UNDEFINED;
+
+
+ if(tvformat == TV_UNDEFINED)
+ goto err0;
+
+ if(!s_cs8556_init) {
+ if(wmt_getsyspara("wmt.cs8556.i2c", buf, &varlen) == 0) {
+ if(strlen(buf) > 0)
+ no = buf[0] - '0';
+ }
+
+ adapter = i2c_get_adapter(no);
+ if (adapter == NULL) {
+ DBG_ERR("Can not get i2c adapter, client address error\n");
+ goto err0;
+ }
+
+ s_cs8556_client = i2c_new_device(adapter, cs8556_i2c_board_info);
+ if (s_cs8556_client == NULL) {
+ DBG_ERR("allocate i2c client failed\n");
+ goto err0;
+ }
+
+ i2c_put_adapter(adapter);
+
+ ret = i2c_add_driver(&cs8556_i2c_driver);
+ if (ret != 0) {
+ DBG_ERR("Failed to register CS8556 I2C driver: %d\n", ret);
+ goto err1;
+ }
+
+ if(wmt_getsyspara("wmt.io.avdetect", buf, &varlen) == 0) {
+ num = sscanf(buf, "%d:%d:%d", &s_avdetect_gpio.flag, &s_avdetect_gpio.gpiono,
+ &s_avdetect_gpio.act);
+
+ if(num != 3)
+ DBG_ERR("wmt.io.avdetect is error. param num = %d\n", num);
+ else {
+ if(s_avdetect_gpio.gpiono > 19)
+ DBG_ERR("invalid avdetect gpio no: %d\n", s_avdetect_gpio.gpiono);
+ else {
+ ret = avdetect_irq_hw_init(0);
+ if(!ret) {
+ ret = request_threaded_irq(IRQ_GPIO, avdetect_irq_handler,
+ avdetect_irq_thread, IRQF_SHARED, CS8556_NAME, s_cs8556_client);
+
+ if(ret)
+ DBG_ERR("%s: irq request failed: %d\n", __func__, ret);
+ else {
+ s_irq_init = 1;
+ DPRINT("avdetect irq request success\n");
+ }
+ }
+ }
+ }
+ }
+
+ s_cs8556_init = 1;
+ }
+ else {
+ if(s_irq_init)
+ avdetect_irq_hw_init(1);
+ }
+
+ ret = I2CMultiPageRead(CS8556_ADDR, 0x00, 0x00, 256, rbuf);
+ if(ret) {
+ DBG_ERR("I2C address 0x%02X is not found\n", CS8556_ADDR);
+ goto err0;
+ }
+
+ //if(s_cs8556_ready) {
+ // DPRINT("cs8556_reinit");
+ switch(tvformat) {
+ case TV_PAL:
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 256, s_RGB888_To_PAL_Offset0);
+ if(ret) {
+ DBG_ERR("PAL init fail\n");
+ goto err0;
+ }
+ break;
+
+ case TV_NTSC:
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 256, s_RGB888_To_NTSC_Offset0);
+ if(ret) {
+ DBG_ERR("NTSC init fail\n");
+ goto err0;
+ }
+ break;
+
+ default:
+ goto err0;
+ break;
+ }
+ //}
+
+ if(s_irq_init)
+ avdetect_irq_enable();
+
+ s_cs8556_ready = 1;
+
+ return 0;
+//err3:
+// cvbs_hotplug_irq_disable();
+// free_irq(IRQ_GPIO, s_cs8556_client);
+// cvbs_hotplug_irq_hw_free();
+//err2:
+// i2c_del_driver(&cs8556_i2c_driver);
+err1:
+ i2c_unregister_device(s_cs8556_client);
+err0:
+ s_cs8556_ready = 0;
+ return -1;
+}
+
+static int cs8556_set_mode(unsigned int *option)
+{
+ if(!s_cs8556_ready)
+ return -1;
+
+ return 0;
+}
+
+static void cs8556_set_power_down(int enable)
+{
+ int ret;
+ vout_t *vo;
+ unsigned char rbuf[256] = {0};
+
+ if( !s_cs8556_ready )
+ return;
+
+ vo = vout_get_entry(VPP_VOUT_NUM_DVI);
+ if (vo->status & (VPP_VOUT_STS_BLANK + VPP_VOUT_STS_POWERDN)) {
+ enable = 1;
+ }
+
+ DPRINT("cs8556_set_power_down(%d)\n",enable);
+
+ ret = I2CMultiPageRead(CS8556_ADDR, 0x00, 0x00, 0x11, rbuf);
+ if(ret) {
+ DBG_ERR("I2C read Offset0 fail\n");
+ return;
+ }
+
+ if( enable ){
+ if(memcmp(rbuf, s_CS8556_Original_Offset0, 0x11) != 0) {
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 0x11, s_CS8556_Original_Offset0);
+ if(ret)
+ DBG_ERR("I2C write Offset0 to power down CS8556 fail\n");
+ else {
+ if(s_irq_init)
+ avdetect_irq_disable();
+ }
+ }
+ } else {
+ if(memcmp(rbuf, s_RGB888_To_PAL_Offset0, 0x11) != 0) {
+ ret = I2CMultiPageWrite(CS8556_ADDR, 0x00, 0x00, 0x11, s_RGB888_To_PAL_Offset0);
+ if(ret)
+ DBG_ERR("I2C write offset0 to power up CS8556 fail\n");
+ }
+ }
+}
+
+static int cs8556_config(vout_info_t *info)
+{
+ int ntsc;
+
+ if(!s_cs8556_ready)
+ return -1;
+
+ if(info->resx == 720 && (info->resy == 576 || info->resy == 480)) {
+ ntsc = (info->resy == 480) ? 1 : 0;
+
+ DPRINT("cs8556_config (%dx%d@%d) %s\n", info->resx, info->resy, info->fps, ntsc ? "NTSC" : "PAL");
+
+ cs8556_set_tv_mode(ntsc);
+ } else {
+
+ DPRINT("cs8556_config (%dx%d@%d)\n", info->resx, info->resy, info->fps);
+ cs8556_set_tv_mode(-1);
+ }
+
+ return 0;
+}
+
+static int cs8556_get_edid(char *buf)
+{
+ return -1;
+}
+
+#ifdef CONFIG_CS8556_INTERRUPT
+static int cs8556_interrupt(void)
+{
+ return cs8556_check_plugin(1);
+}
+#endif
+
+void cs8556_read(void)
+{
+ int i, ret;
+ unsigned char rbuf[256] = {0};
+
+ ret = I2CMultiPageRead(CS8556_ADDR, 0x00, 0x00, 256, rbuf);
+ if(!ret) {
+ printk("CS8556 Read offset0 data as follows:\n");
+ for(i = 0; i < 256;) {
+ printk("0x%02X,", rbuf[i]);
+ if((++i) % 16 == 0)
+ printk("\n");
+ }
+
+ }
+}
+
+/*----------------------- vout device plugin --------------------------------------*/
+vout_dev_t cs8556_vout_dev_ops = {
+ .name = CS8556_NAME,
+ .mode = VOUT_INF_DVI,
+
+ .init = cs8556_init,
+ .set_power_down = cs8556_set_power_down,
+ .set_mode = cs8556_set_mode,
+ .config = cs8556_config,
+ .check_plugin = cs8556_check_plugin,
+ .get_edid = cs8556_get_edid,
+#ifdef CONFIG_CS8556_INTERRUPT
+ .interrupt = cs8556_interrupt,
+#endif
+};
+
+int cs8556_module_init(void)
+{
+ vout_device_register(&cs8556_vout_dev_ops);
+ return 0;
+} /* End of cs8556_module_init */
+module_init(cs8556_module_init);
+/*--------------------End of Function Body -----------------------------------*/
+#undef CS8556_C
+
diff --git a/drivers/video/wmt/devices/lcd-AUO-A080SN01.c b/drivers/video/wmt/devices/lcd-AUO-A080SN01.c
new file mode 100644
index 00000000..113438d8
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-AUO-A080SN01.c
@@ -0,0 +1,93 @@
+/*++
+ * linux/drivers/video/wmt/lcd-AUO-A080SN01.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_AUO_A080SN01_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_A080SN01_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_A080SN01_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+static void lcd_a080sn01_initial(void);
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+lcd_parm_t lcd_a080sn01_parm = {
+ .bits_per_pixel = 24,
+ .capability = 0,
+ .vmode = {
+ .name = "AUO A080SN01",
+ .refresh = 60,
+ .xres = 800,
+ .yres = 600,
+ .pixclock = KHZ2PICOS(40000),
+ .left_margin = 88,
+ .right_margin = 40,
+ .upper_margin = 24,
+ .lower_margin = 1,
+ .hsync_len = 128,
+ .vsync_len = 3,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+ .initial = lcd_a080sn01_initial,
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+static void lcd_a080sn01_initial(void)
+{
+ DPRINT("lcd_a080sn01_initial\n");
+
+ /* TODO */
+}
+
+lcd_parm_t *lcd_a080sn01_get_parm(int arg)
+{
+ return &lcd_a080sn01_parm;
+}
+
+int lcd_a080sn01_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_AUO_A080SN01,
+ (void *) lcd_a080sn01_get_parm);
+ return ret;
+} /* End of lcd_a080sn01_init */
+module_init(lcd_a080sn01_init);
+
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_AUO_A080SN01_C
+
diff --git a/drivers/video/wmt/devices/lcd-CHILIN-LW700at9003.c b/drivers/video/wmt/devices/lcd-CHILIN-LW700at9003.c
new file mode 100644
index 00000000..a9508f0b
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-CHILIN-LW700at9003.c
@@ -0,0 +1,102 @@
+/*++
+ * linux/drivers/video/wmt/lcd-CHILIN-LW700at9003.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_CHILIN_LW700AT9003_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_LW700AT9003_XXXX xxxx *//*Example*/
+
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_LW700AT9003_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+static void lcd_lw700at9003_power_on(void);
+static void lcd_lw700at9003_power_off(void);
+
+/*----------------------- INTERNAL PRIVATE VARIABLES ------------------------*/
+/* int lcd_xxx; *//*Example*/
+lcd_parm_t lcd_lw700at9003_parm = {
+ .bits_per_pixel = 18,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "CHILIN LW700AT9003",
+ .refresh = 48,
+ .xres = 800,
+ .yres = 480,
+ .pixclock = KHZ2PICOS(33260),
+ .left_margin = 105,
+ .right_margin = 105,
+ .upper_margin = 23,
+ .lower_margin = 22,
+ .hsync_len = 40,
+ .vsync_len = 5,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+ .initial = lcd_lw700at9003_power_on,
+ .uninitial = lcd_lw700at9003_power_off,
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+static void lcd_lw700at9003_power_on(void)
+{
+ DPRINT("lcd_lw700at9003_power_on\n");
+ /* TODO */
+}
+
+static void lcd_lw700at9003_power_off(void)
+{
+ DPRINT("lcd_lw700at9003_power_off\n");
+
+ /* TODO */
+}
+
+lcd_parm_t *lcd_lw700at9003_get_parm(int arg)
+{
+ lcd_lw700at9003_parm.bits_per_pixel = arg;
+ return &lcd_lw700at9003_parm;
+}
+
+int lcd_lw700at9003_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_CHILIN_LW0700AT9003,
+ (void *) lcd_lw700at9003_get_parm);
+ return ret;
+} /* End of lcd_oem_init */
+module_init(lcd_lw700at9003_init);
+
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_CHILIN_LW700AT9003_C
diff --git a/drivers/video/wmt/devices/lcd-EKING-EK08009-70135.c b/drivers/video/wmt/devices/lcd-EKING-EK08009-70135.c
new file mode 100644
index 00000000..45ed2c2c
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-EKING-EK08009-70135.c
@@ -0,0 +1,101 @@
+/*++
+ * linux/drivers/video/wmt/lcd-EKING_EK08009-70135.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_EKING_EK08009_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_EK08009_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_EK08009_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+static void lcd_ek08009_power_on(void);
+static void lcd_ek08009_power_off(void);
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+lcd_parm_t lcd_ek08009_parm = {
+ .bits_per_pixel = 18,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "EKING EK08009",
+ .refresh = 60,
+ .xres = 800,
+ .yres = 600,
+ .pixclock = KHZ2PICOS(40000),
+ .left_margin = 46,
+ .right_margin = 210,
+ .upper_margin = 24,
+ .lower_margin = 12,
+ .hsync_len = 1,
+ .vsync_len = 1,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+ .initial = lcd_ek08009_power_on,
+ .uninitial = lcd_ek08009_power_off,
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+static void lcd_ek08009_power_on(void)
+{
+ DPRINT("lcd_ek08009_power_on\n");
+
+ /* TODO */
+}
+
+static void lcd_ek08009_power_off(void)
+{
+ DPRINT("lcd_ek08009_power_off\n");
+
+ /* TODO */
+}
+
+lcd_parm_t *lcd_ek08009_get_parm(int arg)
+{
+ return &lcd_ek08009_parm;
+}
+
+int lcd_ek08009_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_EKING_EK08009,
+ (void *) lcd_ek08009_get_parm);
+ return ret;
+} /* End of lcd_oem_init */
+module_init(lcd_ek08009_init);
+
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_EKING_EK08009_C
diff --git a/drivers/video/wmt/devices/lcd-HANNSTAR-HSD101PFW2.c b/drivers/video/wmt/devices/lcd-HANNSTAR-HSD101PFW2.c
new file mode 100644
index 00000000..6c1acfc9
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-HANNSTAR-HSD101PFW2.c
@@ -0,0 +1,101 @@
+/*++
+ * linux/drivers/video/wmt/lcd-HANNSTAR-HSD101PFW2.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_EKING_HSD101PFW2_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_HSD101PFW2_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_HSD101PFW2_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+static void lcd_HSD101PFW2_power_on(void);
+static void lcd_HSD101PFW2_power_off(void);
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+lcd_parm_t lcd_HSD101PFW2_parm = {
+ .bits_per_pixel = 18,
+ .capability = 0,
+ .vmode = {
+ .name = "HANNSTAR HSD101PFW2",
+ .refresh = 60,
+ .xres = 1024,
+ .yres = 600,
+ .pixclock = KHZ2PICOS(45000),
+ .left_margin = 44,
+ .right_margin = 88,
+ .upper_margin = 10,
+ .lower_margin = 5,
+ .hsync_len = 44,
+ .vsync_len = 10,
+ .sync = FB_SYNC_VERT_HIGH_ACT,
+ .vmode = 0,
+ .flag = 0,
+ },
+ .initial = lcd_HSD101PFW2_power_on,
+ .uninitial = lcd_HSD101PFW2_power_off,
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+static void lcd_HSD101PFW2_power_on(void)
+{
+ DPRINT("lcd_HSD101PFW2_power_on\n");
+
+ /* TODO */
+}
+
+static void lcd_HSD101PFW2_power_off(void)
+{
+ DPRINT("lcd_HSD101PFW2_power_off\n");
+
+ /* TODO */
+}
+
+lcd_parm_t *lcd_HSD101PFW2_get_parm(int arg)
+{
+ return &lcd_HSD101PFW2_parm;
+}
+
+int lcd_HSD101PFW2_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_HANNSTAR_HSD101PFW2,
+ (void *) lcd_HSD101PFW2_get_parm);
+ return ret;
+} /* End of lcd_oem_init */
+module_init(lcd_HSD101PFW2_init);
+
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_EKING_HSD101PFW2_C
diff --git a/drivers/video/wmt/devices/lcd-INNOLUX-AT070TN83.c b/drivers/video/wmt/devices/lcd-INNOLUX-AT070TN83.c
new file mode 100644
index 00000000..a6db70e1
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-INNOLUX-AT070TN83.c
@@ -0,0 +1,122 @@
+/*++
+ * linux/drivers/video/wmt/lcd-INNOLUX-AT070TN83.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_INNOLUX_AT070TN83_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_AT070TN83_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_AT070TN83_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+static void lcd_at070tn83_power_on(void);
+static void lcd_at070tn83_power_off(void);
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+lcd_parm_t lcd_at070tn83_parm = {
+ .bits_per_pixel = 18,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "INNOLUX AT707TN83",
+ .refresh = 60,
+ .xres = 800,
+ .yres = 480,
+ .pixclock = KHZ2PICOS(33333),
+ .left_margin = 45,
+ .right_margin = 210,
+ .upper_margin = 22,
+ .lower_margin = 22,
+ .hsync_len = 1,
+ .vsync_len = 1,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+ .initial = lcd_at070tn83_power_on,
+ .uninitial = lcd_at070tn83_power_off,
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+static void lcd_at070tn83_power_on(void)
+{
+/* DPRINT("lcd_at070tn83_power_on\n"); */
+
+ /* TODO */
+#if (WMT_CUR_PID == WMT_PID_8425)
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x24, 0x7, 0, 0x7); /* gpio enable */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x54, 0x7, 0, 0x7); /* output mode */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x7, 0, 0x0); /* output mode */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x640, 0x707, 0, 0x707); /*pull en*/
+
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x2, 1, 0x1); /* VGL lo */
+ mdelay(8); /* delay 5ms */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x1, 0, 0x1); /* AVDD hi */
+ mdelay(6); /* delay 5ms */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x4, 2, 0x1); /* VGH hi */
+ mdelay(10); /* delay 10ms */
+#endif
+}
+
+static void lcd_at070tn83_power_off(void)
+{
+/* DPRINT("lcd_at070tn83_power_off\n"); */
+
+ /* TODO */
+#if (WMT_CUR_PID == WMT_PID_8425)
+ mdelay(10); /* delay 10ms */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x4, 2, 0x0); /* VGH lo */
+ mdelay(6); /* delay 5ms */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x1, 0, 0x0); /* AVDD lo */
+ mdelay(8); /* delay 5ms */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x84, 0x2, 1, 0x0); /* VGL lo */
+#endif
+}
+
+lcd_parm_t *lcd_at070tn83_get_parm(int arg)
+{
+ return &lcd_at070tn83_parm;
+}
+
+int lcd_at070tn83_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_INNOLUX_AT070TN83,
+ (void *) lcd_at070tn83_get_parm);
+ return ret;
+} /* End of lcd_oem_init */
+module_init(lcd_at070tn83_init);
+
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_INNOLUX_AT070TN83_C
diff --git a/drivers/video/wmt/devices/lcd-b079xan01.c b/drivers/video/wmt/devices/lcd-b079xan01.c
new file mode 100644
index 00000000..54e5921e
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-b079xan01.c
@@ -0,0 +1,465 @@
+/*++
+ linux/drivers/video/wmt/devices/lcd-b079xan01.c
+
+ Copyright (c) 2013 WonderMedia Technologies, Inc.
+
+ 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.
+ You should have received a copy of the GNU General Public License along with
+ this program. If not, see <http://www.gnu.org/licenses/>.
+
+ WonderMedia Technologies, Inc.
+ 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C.
+--*/
+
+#include <linux/delay.h>
+#include <linux/spi/spi.h>
+#include <mach/hardware.h>
+#include <mach/wmt-spi.h>
+#include <linux/gpio.h>
+#include <mach/wmt_iomux.h>
+#include "../lcd.h"
+
+#define DRIVERNAME "ssd2828"
+#define DELAY_MASK 0xff000000
+
+#undef pr_err
+#undef pr_info
+#undef pr_warning
+#define pr_err(fmt, args...) printk("[" DRIVERNAME "] " fmt, ##args)
+#define pr_info(fmt, args...) printk("[" DRIVERNAME "] " fmt, ##args)
+#define pr_warning(fmt, args...) printk("[" DRIVERNAME "] " fmt, ##args)
+
+extern void lcd_power_on(bool on);
+
+struct ssd2828_chip {
+ struct spi_device *spi;
+ int id;
+ int gpio_reset;
+};
+
+static int wmt_lcd_panel_id(void)
+{
+ char buf[96];
+ int len = sizeof(buf);
+ int type, id = 0;
+
+ if (wmt_getsyspara("wmt.display.param", buf, &len)) {
+ return -ENODEV;
+ }
+
+ sscanf(buf, "%d:%d", &type, &id);
+ return id;
+}
+
+// B079XAN01
+
+static const uint32_t b079xan01_init_sequence[] = {
+ 0x7000B1, 0x723240, //VSA=50, HAS=64
+ 0x7000B2, 0x725078, //VBP=30+50, HBP=56+64
+ 0x7000B3, 0x72243C, //VFP=36, HFP=60
+ 0x7000B4, 0x720300, //HACT=768
+ 0x7000B5, 0x720400, //VACT=1024
+ 0x7000B6, 0x72000b, //burst mode, 24bpp loosely packed
+ 0x7000DE, 0x720003, //no of lane=4
+ 0x7000D6, 0x720005, //RGB order and packet number in blanking period
+ 0x7000B9, 0x720000, //disable PLL
+
+ //lane speed=576 (24MHz * 24 = 576)
+ //may modify according to requirement, 500Mbps to 560Mbps
+ //LP clock : 576 / 9 / 8 = 8 MHz
+ 0x7000BA, 0x728018,
+ 0x7000BB, 0x720008,
+
+ 0x7000B9, 0x720001, //enable PPL
+ 0x7000C4, 0x720001, //enable BTA
+ 0x7000B7, 0x720342, //enter LP mode
+ 0x7000B8, 0x720000, //VC
+ 0x7000BC, 0x720000, //set packet size
+
+ 0x700011, //sleep out cmd
+
+ DELAY_MASK + 200,
+
+ 0x700029, //display on
+
+ DELAY_MASK + 200,
+
+ 0x7000B7, 0x72030b, //video mode on
+};
+
+static lcd_parm_t lcd_b079xan01_parm = {
+ .bits_per_pixel = 24,
+ .capability = 0,
+ .vmode = {
+ .name = "B079XAN01",
+ .refresh = 60,
+ .xres = 768,
+ .yres = 1024,
+ .pixclock = KHZ2PICOS(64800),
+ .left_margin = 56,
+ .right_margin = 60,
+ .upper_margin = 30,
+ .lower_margin = 36,
+ .hsync_len = 64,
+ .vsync_len = 50,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+};
+
+static lcd_parm_t *lcd_b079xan01_get_parm(int arg)
+{
+ return &lcd_b079xan01_parm;
+}
+
+// BP080WX7
+
+static const uint32_t bp080wx7_init_sequence[] = {
+ 0x7000B1, 0x720202, // VSA=02 , HSA=02
+ 0x7000B2, 0x720828, // VBP+VSA=8, HBP+HSA=40
+ 0x7000B3, 0x72040A, // VFP=04 , HFP=10
+ 0x7000B4, 0x720500, // HACT=1280
+ 0x7000B5, 0x720320, // vACT=800
+ 0x7000B6, 0x720007, // Non burst mode with sync event 24bpp
+
+ //DELAY_MASK + 10,
+
+ 0x7000DE, 0x720003, // 4lanes
+ 0x7000D6, 0x720005, // BGR
+ 0x7000B9, 0x720000,
+
+ //DELAY_MASK + 10,
+
+ 0x7000BA, 0x72C012, // PLL=24*16 = 384MHz
+ 0x7000BB, 0x720008, // LP CLK=8.3MHz
+ 0x7000B9, 0x720001,
+
+ //DELAY_MASK + 200,
+
+ 0x7000B8, 0x720000, // Virtual Channel 0
+ 0x7000B7, 0x720342, // LP Mode
+
+ //DELAY_MASK + 10,
+
+ 0x7000BC, 0x720000,
+ 0x700011, //sleep out
+
+ DELAY_MASK + 200,
+
+ 0x7000BC, 0x720000,
+ 0x700029, //display on
+
+ //DELAY_MASK + 50,
+
+ 0x7000B7, 0x72034B, //Video Mode
+
+ DELAY_MASK + 200,
+
+ 0x7000BC, 0x720000,
+ 0x700029, //display on
+};
+
+//Timing parameter for 3.0" QVGA LCD
+#define VBPD (6)
+#define VFPD (4)
+#define VSPW (2)
+
+#define HBPD (38)
+#define HFPD (10)
+#define HSPW (2)
+
+// setenv wmt.display.tmr 64800:0:8:47:1280:45:4:16:800:16
+
+static lcd_parm_t lcd_bp080wx7_parm = {
+ .bits_per_pixel = 24,
+ .capability = 0,
+ .vmode = {
+ .name = "BP080WX7",
+ .refresh = 60,
+ .xres = 1280,
+ .yres = 800,
+ .pixclock = KHZ2PICOS(64800),
+ .left_margin = HBPD,
+ .right_margin = HFPD,
+ .upper_margin = VBPD,
+ .lower_margin = VFPD,
+ .hsync_len = HSPW,
+ .vsync_len = VSPW,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+};
+
+static lcd_parm_t *lcd_bp080wx7_get_parm(int arg)
+{
+ return &lcd_bp080wx7_parm;
+}
+
+// SSD2828 api
+static int ssd2828_read(struct spi_device *spi, uint8_t reg)
+{
+ int ret;
+ uint8_t buf1[3] = { 0x70, 0x00, 0x00 };
+ uint8_t buf2[3] = { 0x73, 0x00, 0x00 };
+
+ buf1[2] = reg;
+
+ ret = spi_write(spi, buf1, 3);
+ if (ret) {
+ pr_err("spi_write ret=%d\n", ret);
+ return ret;
+ }
+
+ ret = spi_w8r16(spi, buf2[0]);
+ if (ret < 0) {
+ pr_err("spi_write ret=%d\n", ret);
+ }
+
+ return ret;
+}
+
+#if 0
+static int ssd2828_write(struct spi_device *spi, uint8_t reg, uint16_t data)
+{
+ int ret;
+ uint8_t buf_reg[3] = { 0x70, 0x00, 0x00 };
+ uint8_t buf_data[3] = { 0x72, 0x00, 0x00 };
+
+ buf_reg[2] = reg;
+
+ buf_data[1] = (data >> 8) & 0xff;
+ buf_data[2] = data & 0xff;
+
+ ret = spi_write(spi, buf_reg, 3);
+ if (ret) {
+ pr_err("spi_write ret=%d,w cmd=0x%06x\n", ret, data);
+ return ret;
+ }
+
+ ret = spi_write(spi, buf_data, 3);
+ if (ret)
+ pr_err("spi_write ret=%d,w cmd=0x%06x\n", ret, data);
+
+ return ret;
+}
+#endif
+
+static inline int spi_write_24bit(struct spi_device *spi, uint32_t data)
+{
+ int ret;
+ uint8_t buf[3];
+
+ buf[0] = (data >> 16) & 0xff;
+ buf[1] = (data >> 8) & 0xff;
+ buf[2] = data & 0xff;
+
+ ret = spi_write(spi, buf, 3);
+ if (ret)
+ pr_err("spi_write ret=%d,w cmd=0x%06x\n", ret, data);
+
+ return ret;
+}
+
+static inline void ssd2828_hw_reset(struct ssd2828_chip *chip)
+{
+ lcd_power_on(1);
+ msleep(10);
+
+ gpio_direction_output(chip->gpio_reset, 1);
+ msleep(10);
+ gpio_direction_output(chip->gpio_reset, 0);
+ msleep(200);
+ gpio_direction_output(chip->gpio_reset, 1);
+ msleep(200);
+}
+
+static int ssd2828_hw_init(struct ssd2828_chip *chip)
+{
+ const uint32_t *init_sequence;
+ size_t n;
+ int i, ret = 0;
+
+ ssd2828_hw_reset(chip);
+
+ ret = ssd2828_read(chip->spi, 0xB0);
+ if (ret < 0 || ret != 0x2828) {
+ pr_err("Error: SSD2828 not found!\n");
+ return -ENODEV;
+ }
+
+ switch (chip->id) {
+ case LCD_B079XAN01:
+ init_sequence = b079xan01_init_sequence;
+ n = ARRAY_SIZE(b079xan01_init_sequence);
+ break;
+ case LCD_BP080WX7:
+ init_sequence = bp080wx7_init_sequence;
+ n = ARRAY_SIZE(bp080wx7_init_sequence);
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ for (i = 0; i < n; i++) {
+ if ((init_sequence[i] & DELAY_MASK) == DELAY_MASK) {
+ msleep(init_sequence[i] & 0xff);
+ } else {
+ ret = spi_write_24bit(chip->spi, init_sequence[i]);
+ if (ret)
+ break;
+ }
+ }
+
+ return ret;
+}
+
+static ssize_t option_port_testmode_show(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ char *s = buf;
+ int ret;
+ int reg;
+
+ struct spi_device *spi = container_of(dev, struct spi_device, dev);
+
+ s += sprintf(s, "register value\n");
+
+ for (reg = 0xb0; reg <= 0xff; reg++) {
+ ret = ssd2828_read(spi, (uint8_t)reg);
+ if (ret < 0)
+ goto out;
+ s += sprintf(s, "reg 0x%02X : 0x%04x\n", reg, ret);
+ }
+
+ s += sprintf(s, "=========\n");
+
+out:
+ return (s - buf);
+}
+
+static ssize_t option_port_testmode_store(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ return 0;
+}
+
+static DEVICE_ATTR(testmode, S_IRUGO,
+ option_port_testmode_show,
+ option_port_testmode_store);
+
+static int __devinit ssd2828_spi_probe(struct spi_device *spi)
+{
+ struct ssd2828_chip *chip;
+ int gpio = WMT_PIN_GP0_GPIO0;
+ int ret;
+
+ ret = gpio_request(gpio, "SSD2828 Reset");
+ if (ret) {
+ dev_err(&spi->dev, "can not open GPIO %d\n", gpio);
+ return ret;
+ }
+
+ ret = ssd2828_read(spi, 0xB0);
+ if (ret < 0 || ret != 0x2828) {
+ pr_err("Error: SSD2828 not found!\n");
+ return -ENODEV;
+ }
+
+ chip = kzalloc(sizeof(*chip), GFP_KERNEL);
+ if (!chip)
+ return -ENOMEM;
+
+ chip->spi = spi;
+ chip->id = wmt_lcd_panel_id();
+ chip->gpio_reset = gpio;
+ spi_set_drvdata(spi, chip);
+
+ ret = sysfs_create_file(&spi->dev.kobj, &dev_attr_testmode.attr);
+ if (unlikely(ret)) {
+ pr_err("ssd2828 sysfs_create_file failed\n");
+ }
+
+ return ret;
+}
+
+static int ssd2828_spi_resume(struct spi_device *spi)
+{
+ struct ssd2828_chip *chip = dev_get_drvdata(&spi->dev);
+ return ssd2828_hw_init(chip);
+}
+
+static struct spi_driver ssd2828_driver = {
+ .driver = {
+ .name = DRIVERNAME,
+ .owner = THIS_MODULE,
+ },
+ .probe = ssd2828_spi_probe,
+ .resume = ssd2828_spi_resume,
+};
+
+static struct spi_board_info ssd2828_spi_info[] __initdata = {
+ {
+ .modalias = DRIVERNAME,
+ .bus_num = 1,
+ .chip_select = 0,
+ .max_speed_hz = 12000000,
+ .irq = -1,
+ .mode = SPI_CLK_MODE3,
+ },
+};
+
+static int __init ssd2828_init(void)
+{
+ int ret;
+
+ switch (wmt_lcd_panel_id()) {
+ case LCD_B079XAN01:
+ case LCD_BP080WX7:
+ break;
+ default:
+ pr_warning("lcd for ssd2828 not found\n");
+ return -EINVAL;
+ }
+
+ ret = spi_register_board_info(ssd2828_spi_info,
+ ARRAY_SIZE(ssd2828_spi_info));
+ if (ret) {
+ pr_err("spi_register_board_info failed\n");
+ return ret;
+ }
+
+ ret = spi_register_driver(&ssd2828_driver);
+ if (ret) {
+ pr_err("spi_register_driver failed\n");
+ return ret;
+ }
+
+ lcd_panel_register(LCD_B079XAN01, (void *)lcd_b079xan01_get_parm);
+ lcd_panel_register(LCD_BP080WX7, (void *)lcd_bp080wx7_get_parm);
+
+ pr_info("spi %s register success\n", DRIVERNAME);
+ return 0;
+}
+
+static void ssd2828_exit(void)
+{
+ spi_unregister_driver(&ssd2828_driver);
+}
+
+module_init(ssd2828_init);
+module_exit(ssd2828_exit);
+
+MODULE_AUTHOR("Sam Mei");
+MODULE_DESCRIPTION("WonderMedia Mipi LCD Driver");
+MODULE_LICENSE("GPL");
+
diff --git a/drivers/video/wmt/devices/lcd-gl5001w.c b/drivers/video/wmt/devices/lcd-gl5001w.c
new file mode 100755
index 00000000..03e3de29
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-gl5001w.c
@@ -0,0 +1,1412 @@
+/*++
+ * linux/drivers/video/wmt/devices/lcd-gl5001w.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2012 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/kernel.h>
+#include <linux/err.h>
+#include <linux/power_supply.h>
+#include <linux/i2c.h>
+#include <linux/slab.h>
+#include <linux/interrupt.h>
+#include <linux/proc_fs.h>
+#include <linux/gpio.h>//wangaq+
+#include "../lcd.h"
+
+//#include <mach/gpio-cs.h>
+//#include <mach/wmt_env.h>
+#include <mach/wmt_iomux.h>//wangaq+
+
+
+#include "../lcd.h"//yangchen add
+
+#define DRIVER_NAME "tc358768"
+#define I2C_ADDR 0x0E
+#define I2C_ADAPTER 0
+
+#define DTYPE_DCS_SWRITE_0P 0X05
+#define DTYPE_DCS_SWRITE_1P 0X15
+#define DTYPE_DCS_LWRITE 0X39
+#define DTYPE_GEN_LWRITE 0X29
+#define DTYPE_GEN_SWRITE_2P 0X23
+#define DTYPE_GEN_SWRITE_1P 0X13
+#define DTYPE_GEN_SWRITE_0P 0X03
+
+
+#define Gl5001w_Reset_Port WMT_PIN_GP62_SUSGPIO0
+#define Tc358768_Reset_Port WMT_PIN_GP1_GPIO8
+//#define BackLight_Power_PORT WMT_PIN_GP0_GPIO0
+
+static void lcd_set_power(int status);
+static int save=1;
+//static struct work_struct cw500_resume_work;
+struct cw500_data {
+ struct work_struct cw500_resume_work;
+};
+struct cw500_data *data;
+
+#if 0
+#define lcd_debug printk
+#else
+#define lcd_debug(fmt...) do { } while (0)
+#endif
+
+#define __unused __attribute__ ((unused))
+
+static bool lcd_init_gl500 = false;
+
+
+typedef enum {
+ //LCD_BTL504880,
+ LCD_YT50F62C6,
+ LCD_BT050TN,
+ LCD_ILI9806,
+ LCD_OTM8019A,
+ LCD_MAX
+} lcd_panel_enum;
+
+struct lcd_deviceid_struct{
+ unsigned short id;
+ unsigned char addr;
+ unsigned char read_start;
+ unsigned char len;
+};
+
+static struct lcd_deviceid_struct lcd_deviceid[LCD_MAX]={
+ //{0x8009,0xa1,6,2},
+ {0x9805,0xd3,5,2},
+ {0x8009,0xa1,6,2},
+ {0x9816,0xd3,5,2},
+ {0x8019,0xa1,6,2}
+};
+
+//LCD_YT50F62C6
+static unsigned char LCD_YT50F62C6_CMD_FF_3DATA[] = {0xFF,0xFF,0x98,0x05}; //EXTC Command Set enable register
+static unsigned char LCD_YT50F62C6_CMD_FD_4DATA[] = {0xFD,0x03,0x13,0x44,0x00};
+static unsigned char LCD_YT50F62C6_CMD_F8_15DATA[] = {0xF8,0x15,0x02,0x02,0x15,0x02,0x02,0x30,0x01,0x01,0x30,0x01,0x01,0x30,0x01,0x01};
+static unsigned char LCD_YT50F62C6_CMD_B8_1DATA[] = {0xB8,0x73}; //DBI Type B Interface Setting
+static unsigned char LCD_YT50F62C6_CMD_F1_1DATA[] = {0xF1,0x00}; //Gate Modulation
+static unsigned char LCD_YT50F62C6_CMD_F2_3DATA[] = {0xF2,0x00,0x58,0x41}; //CR/EQ/PC
+static unsigned char LCD_YT50F62C6_CMD_FC_3DATA[] = {0xFC,0x04,0x0F,0x01};
+static unsigned char LCD_YT50F62C6_CMD_FE_1DATA[] = {0xFE,0x19}; //SRAM Repair
+static unsigned char LCD_YT50F62C6_CMD_EB_2DATA[] = {0xEB,0x08,0x0F}; // 3 Gamma & Dithering
+static unsigned char LCD_YT50F62C6_CMD_E0_16DATA[] = {0xE0,0x00,0x02,0x07,0x10,0x10,0x1D,0x0F,0x0B,0x00,0x03,0x02,0x0B,0x0C,0x33,0x2F,0x00}; //P-Gamma
+static unsigned char LCD_YT50F62C6_CMD_E1_16DATA[] = {0xE1,0x00,0x02,0x07,0x10,0x10,0x17,0x0B,0x0B,0x00,0x03,0x02,0x0B,0x0C,0x33,0x2F,0x00}; //N-Gamma
+static unsigned char LCD_YT50F62C6_CMD_C1_4DATA[] = {0xC1,0x13,0x26,0x06,0x26}; //Power Control 1
+static unsigned char LCD_YT50F62C6_CMD_C7_1DATA[] = {0xC7,0xC4};
+static unsigned char LCD_YT50F62C6_CMD_B1_3DATA[] = {0xB1,0x00,0x12,0x14}; //Frame Rate Control
+static unsigned char LCD_YT50F62C6_CMD_B4_1DATA[] = {0xB4,0x02}; // 2 Dot Inversion
+static unsigned char LCD_YT50F62C6_CMD_36_1DATA[] = {0x36,0x0A}; //Memory Access
+static unsigned char LCD_YT50F62C6_CMD_3A_1DATA[] = {0x3A,0x77}; //16 & 18 & 24 bits
+static unsigned char LCD_YT50F62C6_CMD_21_0DATA[] = {0x21}; //Display Inv-On
+static unsigned char LCD_YT50F62C6_CMD_B0_1DATA[] = {0xB0,0x00}; //RGB I/F Polarity
+static unsigned char LCD_YT50F62C6_CMD_B6_1DATA[] = {0xB6,0x01}; //CPU/RGB I/F Select
+static unsigned char LCD_YT50F62C6_CMD_C2_1DATA[] = {0xC2,0x11};
+static unsigned char LCD_YT50F62C6_CMD_11_0DATA[] = {0x11}; //Sleep out
+static unsigned char LCD_YT50F62C6_CMD_29_0DATA[] = {0x29}; //Display on
+static unsigned char LCD_YT50F62C6_CMD_2C_0DATA[] = {0x2C}; //Memory write
+
+//LCD_BT050TN
+static unsigned char LCD_BT050TN_CMD_FF00_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_FF00_3DATA[] = {0xFF,0x80,0x09,0x01};
+static unsigned char LCD_BT050TN_CMD_FF80_0DATA[] = {0x00,0x80};
+static unsigned char LCD_BT050TN_CMD_FF80_2DATA[] = {0xFF,0x80,0x09};
+static unsigned char LCD_BT050TN_CMD_FF03_0DATA[] = {0x00,0x03};
+static unsigned char LCD_BT050TN_CMD_FF03_1DATA[] = {0xFF,0x01};
+static unsigned char LCD_BT050TN_CMD_2100_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_2100_1DATA[] = {0x21,0x00};
+static unsigned char LCD_BT050TN_CMD_D800_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_D800_2DATA[] = {0xD8,0x6F,0x6F};
+static unsigned char LCD_BT050TN_CMD_C582_0DATA[] = {0x00,0x82};
+static unsigned char LCD_BT050TN_CMD_C582_1DATA[] = {0xC5,0xA3};
+static unsigned char LCD_BT050TN_CMD_C181_0DATA[] = {0x00,0x81};
+static unsigned char LCD_BT050TN_CMD_C181_1DATA[] = {0xC1,0x66};
+static unsigned char LCD_BT050TN_CMD_C1A1_0DATA[] = {0x00,0xA1};
+static unsigned char LCD_BT050TN_CMD_C1A1_1DATA[] = {0xC1,0x08};
+static unsigned char LCD_BT050TN_CMD_B4C0_0DATA[] = {0x00,0xB4};
+static unsigned char LCD_BT050TN_CMD_B4C0_1DATA[] = {0xC0,0x50};
+static unsigned char LCD_BT050TN_CMD_C0A3_0DATA[] = {0x00,0xA3};
+static unsigned char LCD_BT050TN_CMD_C0A3_1DATA[] = {0xC0,0x00};
+static unsigned char LCD_BT050TN_CMD_C489_0DATA[] = {0x00,0x89};
+static unsigned char LCD_BT050TN_CMD_C489_1DATA[] = {0xC4,0x08};
+static unsigned char LCD_BT050TN_CMD_C481_0DATA[] = {0x00,0x81};
+static unsigned char LCD_BT050TN_CMD_C481_1DATA[] = {0xC4,0x83};
+static unsigned char LCD_BT050TN_CMD_C590_0DATA[] = {0x00,0x90};
+static unsigned char LCD_BT050TN_CMD_C590_3DATA[] = {0xC5,0x96,0xA7,0x01};
+static unsigned char LCD_BT050TN_CMD_C5B1_0DATA[] = {0x00,0xB1};
+static unsigned char LCD_BT050TN_CMD_C5B1_1DATA[] = {0xC5,0xA9};
+static unsigned char LCD_BT050TN_CMD_D900_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_D900_1DATA[] = {0xD9,0x15};
+static unsigned char LCD_BT050TN_CMD_E100_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_E100_16DATA[] = {0xE1,0x02,0x08,0x0E,0x10,0x09,0x1D,0x0E,0x0E,0x00,0x05,0x02,0x07,0x0E,0x24,0x23,0x1D};
+static unsigned char LCD_BT050TN_CMD_E200_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_E200_16DATA[] = {0xE2,0x02,0x08,0x0E,0x0F,0x09,0x1D,0x0E,0x0D,0x00,0x04,0x02,0x07,0x0E,0x25,0x23,0x1D};
+static unsigned char LCD_BT050TN_CMD_0000_0DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_0000_1DATA[] = {0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_B3A1_0DATA[] = {0x00,0xA1};
+static unsigned char LCD_BT050TN_CMD_B3A1_1DATA[] = {0xB3,0x10};
+static unsigned char LCD_BT050TN_CMD_B3A7_0DATA[] = {0x00,0xA7};
+static unsigned char LCD_BT050TN_CMD_B3A7_1DATA[] = {0xB3,0x10};
+static unsigned char LCD_BT050TN_CMD_C090_0DATA[] = {0x00,0x90};
+static unsigned char LCD_BT050TN_CMD_C090_6DATA[] = {0xC0,0x00,0x44,0x00,0x00,0x00,0x03};
+static unsigned char LCD_BT050TN_CMD_C1A6_0DATA[] = {0x00,0xA6};
+static unsigned char LCD_BT050TN_CMD_C1A6_3DATA[] = {0xC1,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CE80_0DATA[] = {0x00,0x80};
+static unsigned char LCD_BT050TN_CMD_CE80_6DATA[] = {0xCE,0x87,0x03,0x00,0x86,0x03,0x00};
+static unsigned char LCD_BT050TN_CMD_CE90_0DATA[] = {0x00,0x90};
+static unsigned char LCD_BT050TN_CMD_CE90_6DATA[] = {0xCE,0x33,0x1E,0x00,0x33,0x1F,0x00};
+static unsigned char LCD_BT050TN_CMD_CEA0_0DATA[] = {0x00,0xA0};
+static unsigned char LCD_BT050TN_CMD_CEA0_14DATA[] = {0xCE,0x38,0x03,0x03,0x20,0x00,0x00,0x00,0x38,0x02,0x03,0x21,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CEB0_0DATA[] = {0x00,0xb0};
+static unsigned char LCD_BT050TN_CMD_CEB0_14DATA[] = {0xCE,0x38,0x01,0x03,0x22,0x00,0x00,0x00,0x38,0x00,0x03,0x23,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CEC0_0DATA[] = {0x00,0xC0};
+static unsigned char LCD_BT050TN_CMD_CEC0_14DATA[] = {0xCE,0x30,0x00,0x03,0x24,0x00,0x00,0x00,0x30,0x01,0x03,0x25,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CED0_0DATA[] = {0x00,0xD0};
+static unsigned char LCD_BT050TN_CMD_CED0_14DATA[] = {0xCE,0x30,0x02,0x03,0x26,0x00,0x00,0x00,0x30,0x03,0x03,0x27,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CFC6_0DATA[] = {0x00,0xC6};
+static unsigned char LCD_BT050TN_CMD_CFC6_2DATA[] = {0xCF,0x01,0x80};
+static unsigned char LCD_BT050TN_CMD_CFC9_0DATA[] = {0x00,0xC9};
+static unsigned char LCD_BT050TN_CMD_CFC9_1DATA[] = {0xCF,0x00};
+static unsigned char LCD_BT050TN_CMD_CBC0_0DATA[] = {0x00,0xC0};
+static unsigned char LCD_BT050TN_CMD_CBC0_15DATA[] = {0xCB,0x00,0x04,0x04,0x04,0x04,0x00,0x00,0x04,0x04,0x04,0x04,0x00,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CBD0_0DATA[] = {0x00,0xD0};
+static unsigned char LCD_BT050TN_CMD_CBD0_15DATA[] = {0xCB,0x00,0x00,0x00,0x00,0x00,0x00,0x04,0x04,0x04,0x04,0x00,0x00,0x04,0x04,0x04};
+static unsigned char LCD_BT050TN_CMD_CBE0_0DATA[] = {0x00,0xE0};
+static unsigned char LCD_BT050TN_CMD_CBE0_10DATA[] = {0xCB,0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CC80_0DATA[] = {0x00,0x80};
+static unsigned char LCD_BT050TN_CMD_CC80_10DATA[] = {0xCC,0x00,0x26,0x25,0x02,0x06,0x00,0x00,0x0A,0x0E,0x0C};
+static unsigned char LCD_BT050TN_CMD_CC90_0DATA[] = {0x00,0x90};
+static unsigned char LCD_BT050TN_CMD_CC90_15DATA[] = {0xCC,0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x26,0x25,0x01,0x05};
+static unsigned char LCD_BT050TN_CMD_CCA0_0DATA[] = {0x00,0xA0};
+static unsigned char LCD_BT050TN_CMD_CCA0_15DATA[] = {0xCC,0x00,0x00,0x09,0x0d,0x0b,0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_CCB0_0DATA[] = {0x00,0xB0};
+static unsigned char LCD_BT050TN_CMD_CCB0_10DATA[] = {0xCC,0x00,0x25,0x26,0x05,0x01,0x00,0x00,0x0F,0x0B,0x0D};
+static unsigned char LCD_BT050TN_CMD_CCC0_0DATA[] = {0x00,0xC0};
+static unsigned char LCD_BT050TN_CMD_CCC0_15DATA[] = {0xCC,0x09,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x25,0x26,0x06,0x02};
+static unsigned char LCD_BT050TN_CMD_CCD0_0DATA[] = {0x00,0xD0};
+static unsigned char LCD_BT050TN_CMD_CCD0_15DATA[] = {0xCC,0x00,0x00,0x10,0x0c,0x0e,0x0a,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
+static unsigned char LCD_BT050TN_CMD_21_0DATA[] = {0x21};
+static unsigned char LCD_BT050TN_CMD_36_0DATA[] = {0x36};
+static unsigned char LCD_BT050TN_CMD_00_0DATA[] = {0x00};
+static unsigned char LCD_BT050TN_CMD_11_0DATA[] = {0x11};
+static unsigned char LCD_BT050TN_CMD_29_0DATA[] = {0x29};
+static unsigned char LCD_BT050TN_CMD_2C_0DATA[] = {0x2C};
+
+
+//LCD_ILI9806
+static unsigned char LCD_ILI9806_CMD_FF_3DATA[] = {0xFF,0xFF,0x98,0x16};
+static unsigned char LCD_ILI9806_CMD_BA_1DATA[] = {0xBA,0x60};
+static unsigned char LCD_ILI9806_CMD_F3_1DATA[] = {0xF3,0x70};
+static unsigned char LCD_ILI9806_CMD_F9_3DATA[] = {0xF9,0x04,0xFB,0x84};
+static unsigned char LCD_ILI9806_CMD_B0_1DATA[] = {0xB0,0x01};
+static unsigned char LCD_ILI9806_CMD_BC_23DATA[] = {0xBC,0x01,0x0F,0x61,0xFE,0x01,0x01,0x0B,0x11,0x6C,0x63,0xFF,0xFF,0x01,0x01,0x00,0x00,0x55,0x53,0x01,0x00,0x00,0x43,0x0B};
+static unsigned char LCD_ILI9806_CMD_BD_8DATA[] = {0xBD,0x01,0x23,0x45,0x67,0x01,0x23,0x45,0x67};
+static unsigned char LCD_ILI9806_CMD_BE_17DATA[] = {0xBE,0x13,0x11,0x00,0x22,0x22,0xBA,0xAB,0x22,0x22,0x22,0x66,0x22,0x22,0x22,0x22,0x22,0x22};
+static unsigned char LCD_ILI9806_CMD_ED_2DATA[] = {0xED,0x7F,0x0F};
+static unsigned char LCD_ILI9806_CMD_B4_3DATA[] = {0xB4,0x02,0x02,0x02};
+static unsigned char LCD_ILI9806_CMD_B5_4DATA[] = {0xB5,0x14,0x14,0x04,0x00};
+static unsigned char LCD_ILI9806_CMD_C0_3DATA[] = {0xC0,0x7F,0x0B,0x04};
+static unsigned char LCD_ILI9806_CMD_C1_4DATA[] = {0xC1,0x17,0x78,0x78,0x20};
+static unsigned char LCD_ILI9806_CMD_D7_1DATA[] = {0xD7,0x2A};
+static unsigned char LCD_ILI9806_CMD_D8_1DATA[] = {0xD8,0x28};
+static unsigned char LCD_ILI9806_CMD_FC_1DATA[] = {0xFC,0x05};
+static unsigned char LCD_ILI9806_CMD_E0_16DATA[] = {0xE0,0x00,0x03,0x0A,0x0E,0x11,0x15,0x0A,0x08,0x04,0x09,0x07,0x0D,0x0D,0x2E,0x28,0x00};
+static unsigned char LCD_ILI9806_CMD_E1_16DATA[] = {0xE1,0x00,0x02,0x09,0x0E,0x11,0x15,0x0A,0x09,0x04,0x09,0x08,0x0C,0x0D,0x2F,0x28,0x00};
+static unsigned char LCD_ILI9806_CMD_D5_8DATA[] = {0xD5,0x09,0x0A,0x0D,0x0B,0xCB,0xA5,0x01,0x04};
+static unsigned char LCD_ILI9806_CMD_F7_1DATA[] = {0xF7,0x89};
+static unsigned char LCD_ILI9806_CMD_C7_1DATA[] = {0xC7,0x43};
+static unsigned char LCD_ILI9806_CMD_36_1DATA[] = {0x36,0x00};
+static unsigned char LCD_ILI9806_CMD_51_1DATA[] = {0x51,0xFF};
+static unsigned char LCD_ILI9806_CMD_53_1DATA[] = {0x53,0x24};
+static unsigned char LCD_ILI9806_CMD_55_1DATA[] = {0x55,0x03};
+static unsigned char LCD_ILI9806_CMD_11_0DATA[] = {0x11};
+static unsigned char LCD_ILI9806_CMD_29_0DATA[] = {0x29};
+static unsigned char LCD_ILI9806_CMD_2C_0DATA[] = {0x2C}; //Memory write
+
+//OTM8019A
+static unsigned char LCD_OTM8019A_CMD_1[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_2[]= {0xFF,0x80,0x19,0x01};
+static unsigned char LCD_OTM8019A_CMD_3[]= {0x00,0x80};
+static unsigned char LCD_OTM8019A_CMD_4[]= {0xFF,0x80,0x19};
+static unsigned char LCD_OTM8019A_CMD_5[]= {0x00,0x90};
+static unsigned char LCD_OTM8019A_CMD_6[]= {0xB3,0x02};
+static unsigned char LCD_OTM8019A_CMD_7[]= {0x00,0x92};
+static unsigned char LCD_OTM8019A_CMD_8[]= {0xB3,0x45};
+static unsigned char LCD_OTM8019A_CMD_9[]= {0x00,0xA2};
+static unsigned char LCD_OTM8019A_CMD_10[]= {0xC0,0x04,0x00,0x02};
+static unsigned char LCD_OTM8019A_CMD_11[]= {0x00,0x80};
+static unsigned char LCD_OTM8019A_CMD_12[]= {0xC0,0x00,0x58,0x00,0x14,0x16};
+static unsigned char LCD_OTM8019A_CMD_13[]= {0x00,0x90};
+static unsigned char LCD_OTM8019A_CMD_14[]= {0xC0,0x00,0x15,0x00,0x00,0x00,0x03};
+static unsigned char LCD_OTM8019A_CMD_15[]= {0x00,0xB4};
+static unsigned char LCD_OTM8019A_CMD_16[]= {0xC0,0x70};//1+2 dot inversion
+static unsigned char LCD_OTM8019A_CMD_17[]= {0x00,0x81};
+static unsigned char LCD_OTM8019A_CMD_18[]= {0xC1,0x33};
+static unsigned char LCD_OTM8019A_CMD_19[]= {0x00,0x80};
+static unsigned char LCD_OTM8019A_CMD_20[]= {0xC4,0x30,0x83};
+static unsigned char LCD_OTM8019A_CMD_21[]= {0x00,0x89};
+static unsigned char LCD_OTM8019A_CMD_22[]= {0xC4,0x08};
+static unsigned char LCD_OTM8019A_CMD_23[]= {0x00,0x82};
+static unsigned char LCD_OTM8019A_CMD_24[]= {0xC5,0xB0};
+static unsigned char LCD_OTM8019A_CMD_25[]= {0x00,0x90};
+static unsigned char LCD_OTM8019A_CMD_26[]= {0xC5,0x4E,0x79,0x01,0x03};
+static unsigned char LCD_OTM8019A_CMD_27[]= {0x00,0xB1};
+static unsigned char LCD_OTM8019A_CMD_28[]= {0xC5,0xA9};
+static unsigned char LCD_OTM8019A_CMD_29[]= {0x00,0x80};
+static unsigned char LCD_OTM8019A_CMD_30[]= {0xCE,0x87,0x03,0x00,0x85,0x03,0x00,0x86,0x03,0x00,0x84,0x03,0x00};
+static unsigned char LCD_OTM8019A_CMD_31[]= {0x00,0xA0};
+static unsigned char LCD_OTM8019A_CMD_32[]= {0xCE,0x38,0x03,0x03,0x58,0x00,0x00,0x00,0x38,0x02,0x03,0x59,0x00,0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_33[]= {0x00,0xB0};
+static unsigned char LCD_OTM8019A_CMD_34[]= {0xCE,0x38,0x01,0x03,0x5A,0x00,0x00,0x00,0x38,0x00,0x03,0x5B,0x00,0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_35[]= {0x00,0xC0};
+static unsigned char LCD_OTM8019A_CMD_36[]= {0xCE,0x30,0x00,0x03,0x5C,0x00,0x00,0x00,0x30,0x01,0x03,0x5D,0x00,0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_37[]= {0x00,0xD0};
+static unsigned char LCD_OTM8019A_CMD_38[]= {0xCE,0x30,0x02,0x03,0x5E,0x00,0x00,0x00,0x30,0x03,0x03,0x5F,0x00,0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_39[]= {0x00,0xC7};
+static unsigned char LCD_OTM8019A_CMD_40[]= {0xCF,0x00};
+static unsigned char LCD_OTM8019A_CMD_41[]= {0x00,0xC9};
+static unsigned char LCD_OTM8019A_CMD_42[]= {0xCF,0x00};
+static unsigned char LCD_OTM8019A_CMD_43[]= {0x00,0xC4};
+static unsigned char LCD_OTM8019A_CMD_44[]= {0xCB,0x01,0x01,0x01,0x01,0x01,0x01};
+static unsigned char LCD_OTM8019A_CMD_45[]= {0x00,0xD9};
+static unsigned char LCD_OTM8019A_CMD_46[]= {0xCB,0x00,0x00,0x01,0x01,0x01,0x01};
+static unsigned char LCD_OTM8019A_CMD_47[]= {0x00,0xE0};
+static unsigned char LCD_OTM8019A_CMD_48[]= {0xCB,0x01,0x01};
+static unsigned char LCD_OTM8019A_CMD_49[]= {0x00,0x84};
+static unsigned char LCD_OTM8019A_CMD_50[]= {0xCC,0x0C,0x0A,0x10,0x0E,0x03,0x04};
+static unsigned char LCD_OTM8019A_CMD_51[]= {0x00,0x9E};
+static unsigned char LCD_OTM8019A_CMD_52[]= {0xCC,0x00};
+static unsigned char LCD_OTM8019A_CMD_53[]= {0x00,0xA0};
+static unsigned char LCD_OTM8019A_CMD_54[]= {0xCC,0x00,0x02,0x01,0x0d,0x0f,0x09,0x0b};
+static unsigned char LCD_OTM8019A_CMD_55[]= {0x00,0xB4};
+static unsigned char LCD_OTM8019A_CMD_56[]= {0xCC,0x0D,0x0F,0x09,0x0B,0x02,0x01};
+static unsigned char LCD_OTM8019A_CMD_57[]= {0x00,0xCE};
+static unsigned char LCD_OTM8019A_CMD_58[]= {0xCC,0x85};
+static unsigned char LCD_OTM8019A_CMD_59[]= {0x00,0xD0};
+static unsigned char LCD_OTM8019A_CMD_60[]= {0xCC,0x05,0x03,0x04,0x0c,0x0a,0x10,0x0e};
+static unsigned char LCD_OTM8019A_CMD_61[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_62[]= {0xD8,0x85,0x85};
+static unsigned char LCD_OTM8019A_CMD_63[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_64[]= {0xD9,0x61};
+static unsigned char LCD_OTM8019A_CMD_65[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_66[]= {0xE1,0x00,0x03,0x0a,0x1d,0x33,0x49,0x54,0x89,0x7a,0x8e,0x79,0x69,0x81,0x6d,0x73,0x6d,0x66,0x5d,0x52,0x00};
+static unsigned char LCD_OTM8019A_CMD_67[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_68[]= {0xE2,0x00,0x04,0x0a,0x1d,0x33,0x49,0x54,0x89,0x7a,0x8e,0x79,0x69,0x81,0x6d,0x73,0x6d,0x66,0x5d,0x52,0x00};
+static unsigned char LCD_OTM8019A_CMD_69[]= {0x00,0x80};
+static unsigned char LCD_OTM8019A_CMD_70[]= {0xC4,0x30};
+static unsigned char LCD_OTM8019A_CMD_71[]= {0x00,0x98};
+static unsigned char LCD_OTM8019A_CMD_72[]= {0xC0,0x00};
+static unsigned char LCD_OTM8019A_CMD_73[]= {0x00,0xa9};
+static unsigned char LCD_OTM8019A_CMD_74[]= {0xC0,0x06};
+static unsigned char LCD_OTM8019A_CMD_75[]= {0x00,0xb0};
+static unsigned char LCD_OTM8019A_CMD_76[]= {0xC1,0x20,0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_77[]= {0x00,0xe1};
+static unsigned char LCD_OTM8019A_CMD_78[]= {0xC0,0x40,0x18};
+static unsigned char LCD_OTM8019A_CMD_79[]= {0x00,0x80};
+static unsigned char LCD_OTM8019A_CMD_80[]= {0xC1,0x03,0x33};
+static unsigned char LCD_OTM8019A_CMD_81[]= {0x00,0xA0};
+static unsigned char LCD_OTM8019A_CMD_82[]= {0xC1,0xe8};
+static unsigned char LCD_OTM8019A_CMD_83[]= {0x00,0x90};
+static unsigned char LCD_OTM8019A_CMD_84[]= {0xb6,0xb4};
+static unsigned char LCD_OTM8019A_CMD_85[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_86[]= {0xfb,0x01};
+static unsigned char LCD_OTM8019A_CMD_87[]= {0x00,0x00};
+static unsigned char LCD_OTM8019A_CMD_88[]= {0xFF,0xFF,0xFF,0xFF};
+static unsigned char LCD_OTM8019A_CMD_89[]= {0x11,0x00};
+static unsigned char LCD_OTM8019A_CMD_90[]= {0x29,0x00};
+//End of
+static struct i2c_client *tc358768_client;
+
+static inline int tc358768_rd_reg_32bits(uint32_t reg_addr)
+{
+ int ret;
+ struct i2c_msg msgs[2];
+ uint8_t buf[4];
+ struct i2c_client *client = tc358768_client;
+
+ buf[0] = (reg_addr >> 8) & 0xff;
+ buf[1] = reg_addr & 0xff;
+
+ msgs[0].addr = client->addr;
+ msgs[0].flags = 0;
+ msgs[0].len = 2;
+ msgs[0].buf = buf;
+
+ msgs[1].addr = client->addr;
+ msgs[1].flags = I2C_M_RD;
+ msgs[1].len = 2;
+ msgs[1].buf = buf;
+
+ ret = i2c_transfer(client->adapter, msgs, 2);
+ if (ret < 0) {
+ printk("%s:i2c_transfer fail =%d\n", __func__, ret);
+ return -1;
+ }
+
+ ret = (buf[0] << 8) | buf[1];
+
+ return ret;
+}
+
+static inline int tc358768_wr_reg_32bits(uint32_t value)
+{
+ struct i2c_msg msgs;
+ int ret = -1;
+ uint8_t buf[4];
+ struct i2c_client *client = tc358768_client;
+
+ buf[0] = value>>24;
+ buf[1] = value>>16;
+ buf[2] = value>>8;
+ buf[3] = value;
+
+ msgs.addr = client->addr;
+ msgs.flags = 0;
+ msgs.len = 4;
+ msgs.buf = buf;
+
+ ret = i2c_transfer(client->adapter, &msgs, 1);
+ if(ret < 0)
+ printk("%d:i2c_transfer fail = %d\n",__LINE__, ret);
+
+ return ret;
+}
+
+static int _tc358768_wr_regs_32bits(unsigned int reg_array[], int n)
+{
+
+ int i = 0;
+ lcd_debug("%s:%d\n", __func__, n);
+ for(i = 0; i < n; i++) {
+ if(reg_array[i] < 0x00020000) {
+ if(reg_array[i] < 20000)
+ udelay(reg_array[i]);
+ else {
+ mdelay(reg_array[i]/1000);
+ }
+ } else {
+ tc358768_wr_reg_32bits(reg_array[i]);
+ }
+ }
+ return 0;
+}
+
+#define tc358768_wr_regs_32bits(reg_array) \
+ _tc358768_wr_regs_32bits(reg_array, ARRAY_SIZE(reg_array))
+
+static uint32_t initialize[] = {
+ // **************************************************
+ // Initizlize -> Display On after power-on
+ // **************************************************
+ // **************************************************
+ // Power on TC358768XBG according to recommended power-on sequence
+ // Relase reset (RESX="H")
+ // Start input REFCK and PCLK
+ // **************************************************
+ // **************************************************
+ // TC358768XBG Software Reset
+ // **************************************************
+ 0x00020001, //SYSctl, S/W Reset
+ 10,
+ 0x00020000, //SYSctl, S/W Reset release
+
+ // **************************************************
+ // TC358768XBG PLL,Clock Setting
+ // **************************************************
+ 0x00160063, //PLL Control Register 0 (PLL_PRD,PLL_FBD)
+ 0x00180603, //PLL_FRS,PLL_LBWS, PLL oscillation enable
+ 1000,
+ 0x00180613, //PLL_FRS,PLL_LBWS, PLL clock out enable
+
+ // **************************************************
+ // TC358768XBG DPI Input Control
+ // **************************************************
+ 0x00060064, //FIFO Control Register
+
+ // **************************************************
+ // TC358768XBG D-PHY Setting
+ // **************************************************
+ 0x01400000, //D-PHY Clock lane enable
+ 0x01420000, //
+ 0x01440000, //D-PHY Data lane0 enable
+ 0x01460000, //
+ 0x01480000, //D-PHY Data lane1 enable
+ 0x014A0000, //
+ 0x014C0001, //
+ 0x014E0001, //
+ 0x01500001, //
+ 0x01520001, //
+
+ 0x01000002, //
+ 0x01020000, //
+ 0x01040000, //
+ 0x01060002, //
+ 0x01080002, //
+ 0x010A0000, //
+ 0x010C0002, //
+ 0x010E0000, //
+ 0x01100002, //
+ 0x01120000, //
+ // **************************************************
+ // TC358768XBG DSI-TX PPI Control
+ // **************************************************
+ 0x02100A5A, //LINEINITCNT
+ 0x02120000, //
+ 0x02140002, //LPTXTIMECNT
+ 0x02160000, //
+ 0x02180E02, //TCLK_HEADERCNT
+ 0x021A0000, //
+ 0x021C0000, //TCLK_TRAILCNT
+ 0x021E0000, //
+ 0x02200002, //THS_HEADERCNT
+ 0x02220000, //
+ 0x02244650, //TWAKEUPCNT
+ 0x02260000, //
+ 0x02280000, //TCLK_POSTCNT
+ 0x022A0000, //
+ 0x022C0001, //THS_TRAILCNT
+ 0x022E0000, //
+ 0x02300005, //HSTXVREGCNT
+ 0x02320000, //
+ 0x02340007, //HSTXVREGEN enable
+ 0x02360000, //
+ 0x02380001, //DSI clock Enable/Disable during LP
+ 0x023A0000, //
+ 0x023C0002, //BTACNTRL1
+ 0x023E0002, //
+ 0x02040001, //STARTCNTRL
+ 0x02060000, //
+
+ // **************************************************
+ // TC358768XBG DSI-TX Timing Control
+ // **************************************************
+ 0x06200001, //Sync Event mode setting Event mode
+ 0x06220014, //V Control Register1 VBP
+ 0x0624000C, //V Control Register2 not used
+ 0x06260356, //V Control Register3 800
+ 0x0628005E, //H Control Register1
+ 0x062A003F, //H Control Register2
+ //0x062C0438, //H Control Register3 (480*18)/8=1080
+ 0x062C05A0, //H Control Register3 (480*24)/8=1440
+
+ 0x05180001, //DSI Start
+ 0x051A0000, //
+};
+
+static uint32_t start_dsi_hs_mode[] = {
+
+ // **************************************************
+ // Set to HS mode
+ // **************************************************
+ 0x05000083, //DSI lane setting, DSI mode=HS
+ 0x0502A300, //bit set
+ 0x05008000, //Switch to DSI mode
+ 0x0502C300, //
+
+ // **************************************************
+ // Host: RGB(DPI) input start
+ // **************************************************
+
+ //0x00080047, //DSI-TX Format setting: RGB666
+ 0x00080037, //DSI-TX Format setting//3 RGB888;4 RGB666
+ //0x0050001E, //DSI-TX Pixel stream packet Data Type setting
+ 0x0050003E, //Packed Pixel Stream, 24-bit RGB, 8-8-8 Format
+ 0x00320000, //HSYNC Polarity
+
+ 0x00040040, //Configuration Control Register
+
+};
+
+static inline void mipi_dsi_init(void)
+{
+ tc358768_wr_regs_32bits(initialize);
+}
+
+static inline void mipi_dsi_hs_start(void)
+{
+ tc358768_wr_regs_32bits(start_dsi_hs_mode);
+}
+
+static void tc_print(u32 addr)
+{
+ lcd_debug("+++addr->%04x: %04x\n", addr, tc358768_rd_reg_32bits(addr));
+}
+
+static int tc358768_command_tx_less8bytes(unsigned char type,
+ unsigned char *regs, int n)
+{
+ int i = 0;
+ unsigned int command[] = {
+ 0x06020000,
+ 0x06040000,
+ 0x06100000,
+ 0x06120000,
+ 0x06140000,
+ 0x06160000,
+ };
+
+ if(n <= 2)
+ command[0] |= 0x1000; //short packet
+ else {
+ command[0] |= 0x4000; //long packet
+ command[1] |= n; //word count byte
+ }
+ command[0] |= type; //data type
+
+ lcd_debug("*cmd:\n");
+ lcd_debug("0x%08x\n", command[0]);
+ lcd_debug("0x%08x\n", command[1]);
+
+ for(i = 0; i < (n + 1)/2; i++) {
+ command[i+2] |= regs[i*2];
+ if((i*2 + 1) < n)
+ command[i+2] |= regs[i*2 + 1] << 8;
+ lcd_debug("0x%08x\n", command[i+2]);
+ }
+
+ _tc358768_wr_regs_32bits(command, (n + 1)/2 + 2);
+ tc358768_wr_reg_32bits(0x06000001); //Packet Transfer
+ if(regs[0] == 0x29){
+ //tc358768_wr_reg_32bits(0x06000001);
+ }
+ //wait until packet is out
+ i = 100;
+ while(tc358768_rd_reg_32bits(0x0600) & 0x01) {
+ if(i-- == 0)
+ break;
+ tc_print(0x0600);
+ }
+
+ //udelay(50);
+ return 0;
+}
+
+static int __unused tc358768_command_tx_more8bytes_hs(unsigned char type,
+ unsigned char regs[], int n)
+{
+
+ int i = 0;
+ unsigned int dbg_data = 0x00E80000, temp = 0;
+ unsigned int command[] = {
+ 0x05000080, // HS data 4 lane, EOT is added
+ 0x0502A300,
+ 0x00080001,
+ 0x00500000, // Data ID setting
+ 0x00220000, // Transmission byte count= byte
+ 0x00E08000, // Enable I2C/SPI write to VB
+ 0x00E20048, // Total word count = 0x48 (max 0xFFF).
+ // This value should be adjusted considering
+ // trade off between transmission time and
+ // transmission start/stop time delay
+ 0x00E4007F, // Vertical blank line = 0x7F
+ };
+
+
+ command[3] |= type; //data type
+ command[4] |= n & 0xffff; //Transmission byte count
+
+ tc358768_wr_regs_32bits(command);
+
+ for(i = 0; i < (n + 1)/2; i++) {
+ temp = dbg_data | regs[i*2];
+ if((i*2 + 1) < n)
+ temp |= (regs[i*2 + 1] << 8);
+ lcd_debug("0x%08x\n", temp);
+ tc358768_wr_reg_32bits(temp);
+ }
+ if((n % 4 == 1) || (n % 4 == 2)) //4 bytes align
+ tc358768_wr_reg_32bits(dbg_data);
+
+ tc358768_wr_reg_32bits(0x00E0C000); //Start command transmisison
+ tc358768_wr_reg_32bits(0x00E00000); //Stop command transmission. This setting should be done just after above setting to prevent multiple output
+ udelay(200);
+ //Re-Initialize
+ //tc358768_wr_regs_32bits(re_initialize);
+ return 0;
+}
+
+static int tc358768_command_tx_more8bytes_lp(unsigned char type,
+ unsigned char regs[], int n)
+{
+
+ int i = 0;
+ unsigned int dbg_data = 0x00E80000, temp = 0;
+ unsigned int command[] = {
+ 0x00080001,
+ 0x00500000, //Data ID setting
+ 0x00220000, //Transmission byte count= byte
+ 0x00E08000, //Enable I2C/SPI write to VB
+ };
+
+ command[1] |= type; //data type
+ command[2] |= n & 0xffff; //Transmission byte count
+
+ tc358768_wr_regs_32bits(command);
+
+ for(i = 0; i < (n + 1)/2; i++) {
+ temp = dbg_data | regs[i*2];
+ if((i*2 + 1) < n)
+ temp |= (regs[i*2 + 1] << 8);
+ lcd_debug("0x%08x\n", temp);
+ tc358768_wr_reg_32bits(temp);
+
+ }
+ if((n % 4 == 1) || (n % 4 == 2)) //4 bytes align
+ tc358768_wr_reg_32bits(dbg_data);
+
+ tc358768_wr_reg_32bits(0x00E0E000); //Start command transmisison
+ udelay(1000);
+ tc358768_wr_reg_32bits(0x00E02000); //Keep Mask High to prevent short packets send out
+ tc358768_wr_reg_32bits(0x00E00000); //Stop command transmission. This setting should be done just after above setting to prevent multiple output
+ udelay(10);
+ return 0;
+}
+
+int _tc358768_send_packet(unsigned char type, unsigned char regs[], int n) {
+
+ if(n <= 8) {
+ tc358768_command_tx_less8bytes(type, regs, n);
+ } else {
+ //tc358768_command_tx_more8bytes_hs(type, regs, n);
+ tc358768_command_tx_more8bytes_lp(type, regs, n);
+ }
+ return 0;
+}
+
+/*
+ * The DCS is separated into two functional areas:
+ * the User Command Set and the Manufacturer Command Set.
+ * Each command is an eight-bit code with 00h to AFh assigned to
+ * the User Command Set and all other codes assigned to
+ * the Manufacturer Command Set.
+ */
+int _mipi_dsi_send_dcs_packet(unsigned char regs[], int n) {
+
+ unsigned char type = 0;
+ if(n == 1) {
+ type = DTYPE_DCS_SWRITE_0P;
+ } else if (n == 2) {
+ type = DTYPE_DCS_SWRITE_1P;
+ } else if (n > 2) {
+ type = DTYPE_DCS_LWRITE;
+ }
+ _tc358768_send_packet(type, regs, n);
+ return 0;
+}
+
+static int tc358768_command_read_bytes(unsigned char addr,unsigned char *regs, int n)
+{
+ unsigned short *data = (unsigned short *)regs,i;
+ unsigned int command[] = {
+ 0x06021037,
+ 0x06040000,
+ 0x06100000,
+ 0x06000001,
+ 0x06021000,
+ 0x06040000,
+ 0x06100000,
+ 0x06000001,
+ };
+
+ lcd_debug("%s start addr=0x%x\n",__FUNCTION__,addr);
+
+ command[2] |= 32;
+ command[4] |= 0x14;
+ command[6] |= addr;
+
+ _tc358768_wr_regs_32bits(command,4);
+ udelay(100);
+ tc358768_wr_reg_32bits(0x05040010);
+ tc358768_wr_reg_32bits(0x05060000);
+ udelay(100);
+ _tc358768_wr_regs_32bits(&command[4],4);
+
+ while(n-- > 0){
+ *data = (unsigned short)tc358768_rd_reg_32bits(0x0430);
+ //printf("%s *data=0x%x\n",__FUNCTION__,*data);
+ data++;
+ *data = (unsigned short)tc358768_rd_reg_32bits(0x0432);
+ //printf("%s *data=0x%x\n",__FUNCTION__,*data);
+ data++;
+ udelay(100);
+ }
+
+ return 0;
+}
+
+static lcd_panel_enum tc358768_check_lcd_type(void)
+{
+ unsigned char data[50]={0},i=0;
+ int lcd_id;
+ for(i=0;i<LCD_MAX;i++)
+ {
+ tc358768_command_read_bytes(lcd_deviceid[i].addr,data,9);
+ lcd_id = (data[lcd_deviceid[i].read_start]<<8) + data[lcd_deviceid[i].read_start+1];
+ printk("%s lcd_id=0x%x,0x%x\n",__FUNCTION__,lcd_id,lcd_deviceid[i].id);
+ if(lcd_id == lcd_deviceid[i].id)
+ break;
+ }
+ return i;
+}
+
+#define mipi_dsi_send_packet(type, regs) \
+ _tc358768_send_packet(type, regs, ARRAY_SIZE(regs))
+
+#define mipi_dsi_send_dcs_packet(regs) \
+ _mipi_dsi_send_dcs_packet(regs, ARRAY_SIZE(regs))
+
+static int gl5001w_reset(void)
+{
+ printk("wangaq:gl5001w_reset\n");
+
+ gpio_direction_output(Gl5001w_Reset_Port,1);
+ mdelay(2);
+ gpio_direction_output(Gl5001w_Reset_Port,0);
+ mdelay(15);
+ gpio_direction_output(Gl5001w_Reset_Port,1);
+ mdelay(5);
+
+ return 0;
+}
+
+static int tc358768_reset(void)
+{
+ printk("wangaq:tc358768_reset\n");
+
+
+
+ gpio_direction_output(Tc358768_Reset_Port,0);
+ mdelay(20);
+ gpio_direction_output(Tc358768_Reset_Port,1);
+ mdelay(20);
+
+
+ return 0;
+}
+
+static void set_backlight_power(int on)
+{
+ printk("set_backlight_power value=%d\n",on);
+ //gpio_direction_output(BackLight_Power_PORT,on ? 1 : 0);
+
+ printk("set_backlight_power oniff=%d\n",on ? 1 : 0);
+}
+
+static void inline backlight_on(void)
+{
+ set_backlight_power(1);
+}
+
+static void inline backlight_off(void)
+{
+ set_backlight_power(0);
+}
+
+static int gl5001w_hw_init(void)
+{
+ lcd_panel_enum lcd_id;
+
+ mipi_dsi_init();
+ lcd_id = tc358768_check_lcd_type();
+ //lcd init
+ //Start of xiaogang.zhang modified on 2013-4-22 14:22 1.0
+ if(lcd_id==LCD_YT50F62C6)
+ {
+ printk("gl5001w_hw_init LCD_YT50F62C6\n");
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_FF_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_FD_4DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_F8_15DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_B8_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_F1_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_F2_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_FC_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_FE_1DATA);
+ msleep(360);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_EB_2DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_E0_16DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_E1_16DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_C1_4DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_C7_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_B1_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_B4_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_36_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_3A_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_21_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_B0_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_B6_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_C2_1DATA);
+
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_11_0DATA);
+ msleep(120);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_29_0DATA);
+
+ msleep(20);
+ mipi_dsi_send_dcs_packet(LCD_YT50F62C6_CMD_2C_0DATA);
+ }
+ else if(lcd_id==LCD_BT050TN)
+ {
+ printk("gl5001w_hw_init LCD_BT050TN\n");
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_FF00_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_FF00_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_FF80_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_FF80_2DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_FF03_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_FF03_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_2100_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_2100_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_D800_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_D800_2DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C582_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C582_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C181_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C181_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C1A1_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C1A1_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_B4C0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_B4C0_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C0A3_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C0A3_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C489_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C489_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C481_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C481_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C590_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C590_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C5B1_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C5B1_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_D900_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_D900_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_E100_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_E100_16DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_E200_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_E200_16DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_0000_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_0000_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_B3A1_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_B3A1_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_B3A7_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_B3A7_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C090_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C090_6DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C1A6_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_C1A6_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CE80_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CE80_6DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CE90_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CE90_6DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CEA0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CEA0_14DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CEB0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CEB0_14DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CEC0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CEC0_14DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CED0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CED0_14DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CFC6_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CFC6_2DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CFC9_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CFC9_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CBC0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CBC0_15DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CBD0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CBD0_15DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CBE0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CBE0_10DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CC80_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CC80_10DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CC90_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CC90_15DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCA0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCA0_15DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCB0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCB0_10DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCC0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCC0_15DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCD0_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_CCD0_15DATA);
+
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_21_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_36_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_00_0DATA);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_11_0DATA);
+ msleep(120);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_29_0DATA);
+ msleep(20);
+ mipi_dsi_send_dcs_packet(LCD_BT050TN_CMD_2C_0DATA);
+ }
+ else if(lcd_id==LCD_ILI9806)
+ {
+ printk("gl5001w_hw_init LCD_ILI9806\n");
+
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_FF_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_BA_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_F3_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_F9_3DATA);
+ msleep(10);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_B0_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_BC_23DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_BD_8DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_BE_17DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_ED_2DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_B4_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_B5_4DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_C0_3DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_C1_4DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_D7_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_D8_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_FC_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_E0_16DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_E1_16DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_D5_8DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_F7_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_C7_1DATA);
+ msleep(10);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_36_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_51_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_53_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_55_1DATA);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_11_0DATA);
+ msleep(120);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_29_0DATA);
+ msleep(20);
+ mipi_dsi_send_dcs_packet(LCD_ILI9806_CMD_2C_0DATA);
+ }
+ else if(lcd_id==LCD_OTM8019A)
+ {
+ printk("gl5001w_hw_init LCD_OTM8019A\n");
+
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_1);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_2);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_3);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_4);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_5);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_6);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_7);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_8);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_9);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_10);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_11);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_12);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_13);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_14);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_15);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_16);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_17);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_18);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_19);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_20);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_21);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_22);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_23);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_24);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_25);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_26);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_27);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_28);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_29);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_30);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_31);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_32);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_33);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_34);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_35);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_36);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_37);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_38);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_39);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_40);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_41);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_42);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_43);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_44);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_45);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_46);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_47);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_48);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_49);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_50);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_51);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_52);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_53);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_54);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_55);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_56);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_57);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_58);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_59);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_60);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_61);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_62);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_63);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_64);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_65);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_66);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_67);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_68);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_69);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_70);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_71);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_72);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_73);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_74);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_76);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_77);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_78);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_79);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_80);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_81);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_82);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_83);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_84);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_85);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_86);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_87);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_88);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_89);
+ msleep(120);
+ mipi_dsi_send_dcs_packet(LCD_OTM8019A_CMD_90);
+ msleep(120);
+
+ }
+ else
+ {
+ printk("gl5001w_hw_init can't find lcd-\n");
+ return 0;
+ }
+ msleep(1);
+ mipi_dsi_hs_start();
+ msleep(10);
+
+ return 0;
+}
+
+
+static void lcd_gl5001w_power_on(void)
+{
+ int ret;
+
+ printk("lcd_gl5001w_power_on\n");
+ gl5001w_reset();
+ tc358768_reset();
+
+ ret = tc358768_rd_reg_32bits(0);
+ if(ret == 0x4401) {
+ printk("+TC358768AXBG works ok\n");
+ } else {
+ printk("+TC358768AXBG error , read:0x%0x\n", ret);
+ return;
+ }
+
+ gl5001w_hw_init();
+}
+
+static void lcd_gl5001w_power_off(void)
+{
+ printk("lcd_gl5001w_power_off\n");
+
+ gpio_direction_output(Gl5001w_Reset_Port,0);
+
+ lcd_init_gl500 = true;
+}
+
+static int tc358768_probe(struct i2c_client *client,
+ const struct i2c_device_id *did)
+{
+ struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
+ int ret = 0;
+
+ if (!i2c_check_functionality(adapter, I2C_FUNC_I2C)) {
+ dev_warn(&adapter->dev,
+ "I2C-Adapter doesn't support I2C_FUNC_I2C\n");
+ return -EIO;
+ }
+
+ tc358768_client = client;
+ return ret;
+}
+
+static int tc358768_remove(struct i2c_client *client)
+{
+ return 0;
+}
+
+static const struct i2c_device_id tc358768_id[] = {
+ { DRIVER_NAME, 0 },
+ {},
+};
+MODULE_DEVICE_TABLE(i2c, tc358768_id);
+
+
+
+
+
+
+static void cw500_resume_work_func(struct work_struct *work)
+{
+ save=0;
+ //printk("-cw500_resume_work_func-\n");
+ lcd_set_enable(1);
+ lcd_gl5001w_power_on();
+ return NULL;
+}
+
+
+static int get_gpio_lcd_gl5001w(void)
+{
+ int ret = 0;
+ ret = gpio_request(Gl5001w_Reset_Port,"Gl5001w_Reset_Port");
+ if (ret){
+ printk("gpio_request Gl5001w_Reset_Port for GL5001W failed\n");
+ goto err_get;
+ }
+
+ ret = gpio_request(Tc358768_Reset_Port,"Tc358768_Reset_Port");
+ if (ret){
+ printk("gpio_request Tc358768_Reset_Port for Tc358768 failed\n");
+ goto err_get;
+ }
+ //ret = gpio_request(BackLight_Power_PORT,"BackLight_Power_PORT");
+ //if (ret){
+ // printk("gpio_request BackLight_Power_PORT for backlight failed\n");
+ // goto err_get;
+ //}
+ return 0;
+err_get:
+ return -1;
+}
+static void put_gpio_lcd_gl5001w(void)
+{
+ gpio_free(Gl5001w_Reset_Port);
+ gpio_free(Tc358768_Reset_Port);
+ //gpio_free(BackLight_Power_PORT);
+ return;
+}
+
+
+static int cw500_gl5001_suspend(struct i2c_client *c, pm_message_t state)
+{
+ save=1;
+ printk("-cw500_gl5001_suspend-\n");
+ put_gpio_lcd_gl5001w();
+ return 0;
+}
+
+
+static int cw500_gl5001_resume(struct i2c_client *c)
+{
+
+ printk("-cw500_gl5001_resume\n");
+ get_gpio_lcd_gl5001w();
+ schedule_work(&data->cw500_resume_work);
+ return 0;
+}
+
+static struct i2c_driver tc358768_driver = {
+ .driver = {
+ .name = DRIVER_NAME,
+ },
+ .probe = tc358768_probe,
+ .remove = __devexit_p(tc358768_remove),
+ .id_table = tc358768_id,
+ .resume= cw500_gl5001_resume,
+ .suspend= cw500_gl5001_suspend,
+};
+
+static void lcd_set_power(int status)
+{
+
+
+ if (!lcd_init_gl500) {
+ return;
+ }
+
+ if (save==0) {
+ return;
+ }
+
+ if (status == LCD_POWER_ON)
+ {
+ lcd_gl5001w_power_on();
+ lcd_init_gl500 = false;
+ }
+}
+
+static struct lcd_parm_t lcd_gl5001w_parm = {
+ //.name = "GL5001W",
+ //.fps = 60, /* frame per second */
+ .bits_per_pixel = 24,
+ .capability = 0,
+ .width = 480,
+ .height = 854,
+ .vmode = {
+ .name = "GL5001W", // mipi_480x800
+ .refresh = 60,
+ .xres = 480,
+ .yres = 854,
+ .pixclock = KHZ2PICOS(26000), //pixel_clock
+ .left_margin = 20, //hbp
+ .right_margin = 30, //hfp
+ .upper_margin = 12, //vbp
+ .lower_margin = 8, //vfp
+ .hsync_len = 10, //hsync
+ .vsync_len = 8, //vsync
+ .sync = 0, //?
+ .vmode = 0,
+ .flag = 0,
+ },
+ //.initial = dummy_call_path,
+ //.uninitial = lcd_gl5001w_power_off,
+ //.set_power = lcd_set_power,
+};
+
+static struct lcd_parm_t *lcd_gl5001w_get_parm(int arg)
+{
+ return &lcd_gl5001w_parm;
+}
+
+static int wmt_check_devices(void)
+{
+ int ret = 0;
+ int param[7];
+ char buf[96] = {0};
+ int len = sizeof(buf);
+
+ ret = wmt_getsyspara("wmt.display.fb0", buf, &len);
+ if (ret) {
+ pr_err("Read wmt.display.param Failed.\n");
+ return -ENODEV;
+ }
+
+ ret = vpp_parse_param(buf, param, 6, 0);
+ if (ret < 2)
+ return -ENODEV;
+
+ if (param[3] != LCD_GL5001W)
+ return -ENODEV;
+
+ return 0;
+}
+
+static struct i2c_board_info i2c_board_info = {
+ I2C_BOARD_INFO(DRIVER_NAME, I2C_ADDR),
+};
+
+#if 0
+static int tc358768_proc(char *page, char **start, off_t off,
+ int count, int *eof, void *data)
+{
+ lcd_gl5001w_power_on();
+ return 0;
+}
+#endif
+
+static int __init gl5001w_init(void)
+{
+ int ret;
+ struct i2c_client *client;
+ struct i2c_adapter *adap;
+ unsigned char buf[40];
+ int buflen = 40;
+ unsigned int value;
+ char *endp;
+
+ if(wmt_getsyspara("wmt.support.lcd.gl5001w", buf, &buflen) == 0) {
+ value = simple_strtoul(buf, &endp, 0);
+ if(value == 0)
+ return -1;
+ } else
+ return -1;
+
+ ret = gpio_request(Gl5001w_Reset_Port,"Gl5001w_Reset_Port");
+ if (ret){
+ printk("gpio_request Gl5001w_Reset_Port for GL5001W failed\n");
+ goto err_get;
+ }
+
+ ret = gpio_request(Tc358768_Reset_Port,"Tc358768_Reset_Port");
+ if (ret){
+ printk("gpio_request Tc358768_Reset_Port for Tc358768 failed\n");
+ goto err_get;
+ }
+ //ret = gpio_request(BackLight_Power_PORT,"BackLight_Power_PORT");
+ //if (ret){
+ // printk("gpio_request BackLight_Power_PORT for backlight failed\n");
+ // goto err_get;
+ //}
+
+ data = kzalloc(sizeof(struct cw500_data), GFP_KERNEL);
+
+ INIT_WORK(&data->cw500_resume_work, cw500_resume_work_func);
+
+ ret = wmt_check_devices();
+ if (ret) {
+ pr_info("LCD GL5001W not found\n");
+ return -ENODEV;
+ }
+
+ adap = i2c_get_adapter(I2C_ADAPTER);
+ if (!adap)
+ return -ENODEV;
+ client = i2c_new_device(adap, &i2c_board_info);
+ i2c_put_adapter(adap);
+ if (!client) {
+ printk("i2c_new_device error\n");
+ return -ENODEV;
+ }
+
+ ret = i2c_add_driver(&tc358768_driver);
+ if (ret) {
+ return -EIO;
+ }
+
+ ret = lcd_panel_register(LCD_GL5001W,(void *) lcd_gl5001w_get_parm);
+ if (ret) {
+ i2c_del_driver(&tc358768_driver);
+ return -ENODEV;
+ }
+
+ /* for debug */
+ //create_proc_read_entry(DRIVER_NAME, 0666, NULL, tc358768_proc, NULL);
+ return 0;
+err_get:
+ return -1;
+}
+
+static void __exit gl5001w_exit(void)
+{
+ struct i2c_client *client = tc358768_client;
+ gpio_free(Gl5001w_Reset_Port);
+ gpio_free(Tc358768_Reset_Port);
+ //gpio_free(BackLight_Power_PORT);
+ i2c_unregister_device(client);
+ return i2c_del_driver(&tc358768_driver);
+}
+
+module_init(gl5001w_init);
+module_exit(gl5001w_exit);
+
+MODULE_DESCRIPTION("WonderMedia GL5001W LCD Driver");
+MODULE_LICENSE("GPL");
+MODULE_ALIAS("i2c:gl5001w");
+
diff --git a/drivers/video/wmt/devices/lcd-lvds-1024x600.c b/drivers/video/wmt/devices/lcd-lvds-1024x600.c
new file mode 100644
index 00000000..f491cce4
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-lvds-1024x600.c
@@ -0,0 +1,92 @@
+/*++
+ * linux/drivers/video/wmt/lcd-lvds-1024x600.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_LVDS_1024x600_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_LVDS_1024x600_XXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_LVDS_1024x600_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+static void lcd_LVDS_1024x600_initial(void);
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+lcd_parm_t lcd_LVDS_1024x600_parm = {
+ .bits_per_pixel = 24,
+ .capability = LCD_CAP_VSYNC_HI,
+ .vmode = {
+ .name = "ePAD 1024x600", /* LVDS_1024x600 */
+ .refresh = 60,
+ .xres = 1024,
+ .yres = 600,
+ .pixclock = KHZ2PICOS(45000),
+ .left_margin = 50,
+ .right_margin = 50,
+ .upper_margin = 10,
+ .lower_margin = 10,
+ .hsync_len = 4,
+ .vsync_len = 4,
+ .sync = FB_SYNC_VERT_HIGH_ACT,
+ .vmode = 0,
+ .flag = 0,
+ },
+ .initial = lcd_LVDS_1024x600_initial,
+};
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+static void lcd_LVDS_1024x600_initial(void)
+{
+ DPRINT("lcd_LVDS_1024x600_initial\n");
+
+ /* TODO */
+}
+
+lcd_parm_t *lcd_LVDS_1024x600_get_parm(int arg)
+{
+ return &lcd_LVDS_1024x600_parm;
+}
+
+int lcd_LVDS_1024x600_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_LVDS_1024x600,
+ (void *) lcd_LVDS_1024x600_get_parm);
+ return ret;
+} /* End of lcd_oem_init */
+module_init(lcd_LVDS_1024x600_init);
+
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_1024x600_C
diff --git a/drivers/video/wmt/devices/lcd-oem.c b/drivers/video/wmt/devices/lcd-oem.c
new file mode 100644
index 00000000..ed5f3407
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-oem.c
@@ -0,0 +1,323 @@
+/*++
+ * linux/drivers/video/wmt/lcd-oem.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_OEM_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../lcd.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_OEM_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_OEM_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+
+static void lcd_oem_initial(void)
+{
+ REG32_VAL(GPIO_BASE_ADDR + 0x80) |= 0x801;
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x801;
+ lcd_enable_signal(1);
+}
+
+static void lcd_oem_uninitial(void)
+{
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x801;
+ lcd_enable_signal(0);
+}
+
+lcd_parm_t lcd_oem_parm = {
+ .bits_per_pixel = 24,
+ .capability = LCD_CAP_VSYNC_HI,
+ .vmode = {
+ .name = "WonderMedia OEM LCD (VGA 1024x768)",
+ .refresh = 60,
+ .xres = 1024,
+ .yres = 768,
+ .pixclock = KHZ2PICOS(63500),
+ .left_margin = 152,
+ .right_margin = 48,
+ .upper_margin = 23,
+ .lower_margin = 3,
+ .hsync_len = 104,
+ .vsync_len = 4,
+ .sync = FB_SYNC_VERT_HIGH_ACT,
+ .vmode = 0,
+ .flag = 0,
+ },
+// .initial = lcd_oem_initial,
+// .uninitial = lcd_oem_uninitial
+};
+
+static void lcd_oem_1024x600_initial(void)
+{
+ REG32_VAL(GPIO_BASE_ADDR + 0x80) |= 0x801; /* GPIO enable */
+
+ /* DVDD */
+ /* T2 > 0ms */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x01; /* AVDD/VCOM( bit 0 ) */
+ /* T4 > 0ms */
+ /* VGH */
+ /* 0 < T6 <= 10ms */
+ lcd_enable_signal(1); /* singal, DVO enable */
+ mdelay(200); /* T12 > 200ms */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x800; /* BL( bit 11 ) */
+}
+
+static void lcd_oem_1024x600_uninitial(void)
+{
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x800; /* BL( bit 11 ) */
+ mdelay(200); /* T12 > 200ms */
+ lcd_enable_signal(0); /* singal, DVO enable */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x01; /* AVDD/VCOM( bit 0 ) */
+}
+
+lcd_parm_t lcd_oem_parm_1024x600 = {
+ .bits_per_pixel = 24,
+ .capability = LCD_CAP_VSYNC_HI,
+ .vmode = {
+#if 1 /* 7" HHX070ML208CP21A */
+ .name = "HHX070ML208CP21A",
+ .refresh = 60,
+ .xres = 1024,
+ .yres = 600,
+ .pixclock = KHZ2PICOS(51200),
+ .left_margin = 140,
+ .right_margin = 160,
+ .upper_margin = 20,
+ .lower_margin = 12,
+ .hsync_len = 20,
+ .vsync_len = 3,
+ .sync = FB_SYNC_VERT_HIGH_ACT,
+ .vmode = 0,
+ .flag = 0,
+#else
+ .name = "ePAD 1024x600", /* HannStar HSD070PFW3 */
+ .refresh = 60,
+ .xres = 1024,
+ .yres = 600,
+ .pixclock = KHZ2PICOS(45000),
+ .left_margin = 50,
+ .right_margin = 50,
+ .upper_margin = 10,
+ .lower_margin = 10,
+ .hsync_len = 4,
+ .vsync_len = 4,
+ .sync = FB_SYNC_VERT_HIGH_ACT,
+ .vmode = 0,
+ .flag = 0,
+#endif
+ },
+// .initial = lcd_oem_1024x600_initial,
+// .uninitial = lcd_oem_1024x600_uninitial,
+};
+
+lcd_parm_t lcd_oem_parm_1024x768 = {
+ .bits_per_pixel = 24,
+ .capability = LCD_CAP_VSYNC_HI,
+ .vmode = {
+ .name = "OEM 1024x768", /* VGA 1024x768 */
+ .refresh = 60,
+ .xres = 1024,
+ .yres = 768,
+ .pixclock = KHZ2PICOS(63500),
+ .left_margin = 152,
+ .right_margin = 48,
+ .upper_margin = 23,
+ .lower_margin = 3,
+ .hsync_len = 104,
+ .vsync_len = 4,
+ .sync = FB_SYNC_VERT_HIGH_ACT,
+ .vmode = 0,
+ .flag = 0,
+ },
+// .initial = lcd_oem_initial,
+// .uninitial = lcd_oem_uninitial
+};
+
+lcd_parm_t lcd_oem_parm_1366x768 = {
+ .bits_per_pixel = 18,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "OEM 1366X768",
+ .refresh = 60,
+ .xres = 1366,
+ .yres = 768,
+ .pixclock = KHZ2PICOS(75440),
+ .left_margin = 98,
+ .right_margin = 31,
+ .upper_margin = 22,
+ .lower_margin = 4,
+ .hsync_len = 65,
+ .vsync_len = 12,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+// .initial = lcd_oem_initial,
+// .uninitial = lcd_oem_uninitial
+};
+
+lcd_parm_t lcd_oem_parm_800x480 = {
+ .bits_per_pixel = 18,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "OEM 800x480",
+ .refresh = 48,
+ .xres = 800,
+ .yres = 480,
+ .pixclock = KHZ2PICOS(27000),
+ .left_margin = 50,
+ .right_margin = 50,
+ .upper_margin = 17,
+ .lower_margin = 16,
+ .hsync_len = 10,
+ .vsync_len = 5,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+// .initial = lcd_oem_initial,
+// .uninitial = lcd_oem_uninitial
+};
+
+static void lcd_oem_1280x800_initial(void)
+{
+ DBG_MSG("lcd 10 power sequence\n");
+ REG32_VAL(GPIO_BASE_ADDR + 0x80) |= 0x801; /* GPIO enable */
+
+ /* VDD on */
+ /* 0 < T < 50ms */
+ lcd_enable_signal(1); /* singal on */
+ /* VGH,VGL low */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x01; /* AVDD/VCOM( bit 0 ) */
+ mdelay(150); /* T5 > 120ms */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x800; /* BL( bit 11 ) */
+}
+
+static void lcd_oem_1280x800_uninitial(void)
+{
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x800; /* turn off backlight */
+ mdelay(150);
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x01; /* turn off LCD */
+ lcd_enable_signal(0); /* turn off singal */
+}
+
+lcd_parm_t lcd_oem_parm_800x1280 = {
+ .bits_per_pixel = 24,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "WY101ML369IN30A",
+ .refresh = 60,
+ .xres = 800,
+ .yres = 1280,
+ .pixclock = KHZ2PICOS(71100),
+ .left_margin = 70,
+ .right_margin = 80,
+ .upper_margin = 10,
+ .lower_margin = 10,
+ .hsync_len = 10,
+ .vsync_len = 3,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+// .initial = lcd_oem_1280x800_initial,
+// .uninitial = lcd_oem_1280x800_uninitial,
+};
+
+lcd_parm_t lcd_oem_parm_1280x800 = {
+ .bits_per_pixel = 24,
+ .capability = LCD_CAP_CLK_HI,
+ .vmode = {
+ .name = "WY101ML369IN30A",
+ .refresh = 60,
+ .xres = 1280,
+ .yres = 800,
+ .pixclock = KHZ2PICOS(71100),
+ .left_margin = 70,
+ .right_margin = 80,
+ .upper_margin = 10,
+ .lower_margin = 10,
+ .hsync_len = 10,
+ .vsync_len = 3,
+ .sync = 0,
+ .vmode = 0,
+ .flag = 0,
+ },
+// .initial = lcd_oem_1280x800_initial,
+// .uninitial = lcd_oem_1280x800_uninitial,
+};
+
+/*----------------------- Function Body --------------------------------------*/
+lcd_parm_t *lcd_oem_get_parm(int arg)
+{
+ return &lcd_oem_parm;
+}
+
+int lcd_oem_init(void)
+{
+ int ret;
+
+ ret = lcd_panel_register(LCD_WMT_OEM, (void *)lcd_oem_get_parm);
+ return ret;
+} /* End of lcd_oem_init */
+module_init(lcd_oem_init);
+
+lcd_parm_t *lcd_get_oem_parm(int resx, int resy)
+{
+ lcd_parm_t *oem_parm[] = {
+ &lcd_oem_parm_1024x600,
+ &lcd_oem_parm_1024x768,
+ &lcd_oem_parm_1366x768,
+ &lcd_oem_parm_800x480,
+ &lcd_oem_parm_800x1280,
+ &lcd_oem_parm_1280x800,
+ 0
+ };
+ lcd_parm_t *p;
+ int i;
+
+ for (i = 0; ; i++) {
+ p = oem_parm[i];
+ if (p == 0) {
+ p = oem_parm[0];
+ break;
+ }
+ if ((resx == p->vmode.xres) && (resy == p->vmode.yres))
+ break;
+ }
+ return p;
+}
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_OEM_C
diff --git a/drivers/video/wmt/devices/lcd-setup.c b/drivers/video/wmt/devices/lcd-setup.c
new file mode 100755
index 00000000..798f020e
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-setup.c
@@ -0,0 +1,311 @@
+/*
+ * ==========================================================================
+ *
+ * Filename: lcd-setup.c
+ *
+ * Description:
+ *
+ * Version: 0.01
+ * Created: 2014.6.6
+ *
+ * Author: SamMei
+ * Company:
+ *
+ * ==========================================================================
+ */
+
+#include <linux/delay.h>
+#include <linux/spi/spi.h>
+#include <linux/i2c.h>
+#include <mach/hardware.h>
+#include <mach/wmt-spi.h>
+#include <linux/gpio.h>
+#include <mach/wmt_iomux.h>
+#include "../lcd.h"
+
+#define DRIVERNAME "lcd-setup"
+
+#undef pr_err
+#undef pr_info
+#undef pr_warning
+#define pr_err(fmt, args...) printk("[" DRIVERNAME "] " fmt, ##args)
+#define pr_info(fmt, args...) printk("[" DRIVERNAME "] " fmt, ##args)
+#define pr_warning(fmt, args...) printk("[" DRIVERNAME "] " fmt, ##args)
+
+enum {
+ LCD_SETUP_TS8224B, /* spi init */
+ LCD_SETUP_CHUNGHWA, /* i2c init */
+ LCD_SETUP_MAX,
+};
+
+static int lcd_setup_id = -1;
+struct spi_device *g_spi_device;
+
+static int wmt_lcd_setup_id(void)
+{
+ char buf[32];
+ int len = sizeof(buf);
+ int id;
+
+ if (wmt_getsyspara("wmt.lcd.setup", buf, &len))
+ id = -ENODEV;
+ sscanf(buf, "%d", &id);
+ return id;
+}
+
+static inline void spi_ctrl_9bit_tx(u8 val, int cmd_data)
+{
+ uint8_t buf[2];
+
+ if (cmd_data)
+ buf[0] = (val >> 1) | BIT7;
+ else
+ buf[0] = (val >> 1) & 0x7f;
+
+ buf[1] = (val << 7);
+
+ spi_write(g_spi_device, buf, sizeof(buf));
+}
+
+static inline void spi_9bit_tx(u8 val, int cmd_data)
+{
+ spi_ctrl_9bit_tx(val, cmd_data);
+}
+
+static inline int ts8224b_cmd(u8 cmd)
+{
+ spi_9bit_tx(cmd, 0);
+ return 0;
+}
+
+static inline int ts8224b_data(u8 data)
+{
+ spi_9bit_tx(data, 1);
+ return 0;
+}
+
+static int ts8224b_init(void)
+{
+ static uint16_t settings[] = {
+ #include "ts8224b.h"
+ };
+ int i;
+
+ printk(" ## %s, %d\n", __func__, __LINE__);
+
+ for (i = 0; i < ARRAY_SIZE(settings); i += 2) {
+ ts8224b_cmd(settings[i] >> 8);
+ ts8224b_data(settings[i+1]);
+ }
+
+ ts8224b_cmd(0x11);
+ msleep(120);
+
+ ts8224b_cmd(0x29);
+ msleep(50);
+ ts8224b_cmd(0x2c);
+ return 0;
+}
+
+extern void lcd_power_on(bool on);
+
+//int lcd_spi_resume(struct spi_device *spi)
+int lcd_spi_resume(void)
+{
+ //printk(" ## %s, %d\n", __func__, __LINE__);
+ switch (lcd_setup_id) {
+ case LCD_SETUP_TS8224B:
+ printk(" ## %s, %d\n", __func__, __LINE__);
+ lcd_power_on(1);
+ mdelay(5);
+ return ts8224b_init();
+ default:
+ return 0;
+ }
+}
+
+static int __devinit lcd_spi_probe(struct spi_device *spi)
+{
+ g_spi_device = spi;
+
+ //lcd_spi_resume();
+
+ return 0;
+}
+
+static struct spi_driver lcd_spi_driver = {
+ .driver = {
+ .name = DRIVERNAME,
+ .owner = THIS_MODULE,
+ },
+ .probe = lcd_spi_probe,
+ //.resume = lcd_spi_resume,
+};
+
+static struct spi_board_info lcd_spi_info[] __initdata = {
+ {
+ .modalias = DRIVERNAME,
+ .bus_num = 0,
+ .chip_select = 0,
+ .max_speed_hz = 12000000,
+ .irq = -1,
+ .mode = SPI_CLK_MODE1,
+ },
+};
+
+
+// I2C
+
+static int lcd_i2c_resume(struct i2c_client *client)
+{
+ static uint8_t init_data[] = {
+ 0x4b, 0x01,
+ 0x0c, 0x01,
+ 0x05, 0x03,
+ 0x41, 0x03,
+ 0x10, 0x06,
+ 0x11, 0xE0,
+ 0x12, 0x00,
+ 0x13, 0x3C,
+ 0x14, 0x06,
+ 0x15, 0x40,
+ 0x16, 0x03,
+ 0x17, 0x9E,
+ 0x18, 0x00,
+ 0x19, 0x10,
+ 0x1a, 0x03,
+ 0x1b, 0x84,
+ 0x1c, 0x80,
+ 0x1d, 0x0A,
+ 0x1e, 0x80,
+ 0x1f, 0x06,
+ 0x3c, 0x17,
+ 0x3e, 0x16,
+ 0x36, 0x00,
+ 0x31, 0x00,
+ 0x35, 0x41,
+ 0x30, 0xB0,
+ 0x30, 0xB1,
+ 0x00, 0x0B,
+ };
+
+ int i, ret;
+
+ printk(" ## %s, %d\n", __FUNCTION__, __LINE__);
+ for (i = 0; i < ARRAY_SIZE(init_data); i += 2) {
+ ret = i2c_master_send(client, &init_data[i], 2);
+ if (ret < 0)
+ return ret;
+ }
+ return 0;
+}
+
+static int __devinit lcd_i2c_probe(struct i2c_client *client,
+ const struct i2c_device_id *id)
+{
+ printk(" ## %s, %d\n", __FUNCTION__, __LINE__);
+ return 0;
+}
+
+static const struct i2c_device_id lcd_i2c_id[] = {
+ { "lcd-i2c", 0 },
+ { },
+};
+
+static struct i2c_driver lcd_i2c_driver = {
+ .driver = {
+ .name = "lcd-i2c",
+ .owner = THIS_MODULE,
+ },
+ .probe = lcd_i2c_probe,
+// .remove = __devexit_p(lcd_i2c_remove),
+// .suspend = lcd_i2c_suspend,
+ .resume = lcd_i2c_resume,
+// .shutdown = lcd_i2c_shutdown,
+ .id_table = lcd_i2c_id,
+};
+
+static struct i2c_board_info lcd_i2c_board_info = {
+ .type = "lcd-i2c",
+ .flags = 0x00,
+ .addr = 0xe0 >> 1,
+ .platform_data = NULL,
+ .archdata = NULL,
+ .irq = -1,
+};
+
+static struct i2c_client *i2c_client;
+static struct i2c_adapter *i2c_adap;
+
+static int __init lcd_setup_init(void)
+{
+ int ret;
+
+ lcd_setup_id = wmt_lcd_setup_id();
+ switch (lcd_setup_id) {
+
+ // SPI
+ case LCD_SETUP_TS8224B:
+ ret = spi_register_board_info(lcd_spi_info, ARRAY_SIZE(lcd_spi_info));
+ if (ret) {
+ pr_err("spi_register_board_info failed\n");
+ return ret;
+ }
+
+ ret = spi_register_driver(&lcd_spi_driver);
+ if (ret) {
+ pr_err("spi_register_driver failed\n");
+ return ret;
+ }
+
+ pr_info("spi %s register success\n", DRIVERNAME);
+ return 0;
+
+ // I2C
+ case LCD_SETUP_CHUNGHWA:
+ printk(" ## %s, %d\n", __FUNCTION__, __LINE__);
+ i2c_adap = i2c_get_adapter(1);
+ if (!i2c_adap) {
+ pr_err("Cannot get i2c adapter 1\n");
+ return -ENODEV;
+ }
+
+ i2c_client = i2c_new_device(i2c_adap, &lcd_i2c_board_info);
+ if (!i2c_client) {
+ pr_err("Unable to add I2C device for 0x%x\n",
+ lcd_i2c_board_info.addr);
+ return -ENODEV;
+ }
+
+ return i2c_add_driver(&lcd_i2c_driver);
+
+ // INVALID
+ default:
+ return -EINVAL;
+ }
+}
+
+static void lcd_setup_exit(void)
+{
+ switch (lcd_setup_id) {
+
+ // SPI
+ case LCD_SETUP_TS8224B:
+ spi_unregister_driver(&lcd_spi_driver);
+ break;
+
+ // I2C
+ case LCD_SETUP_CHUNGHWA:
+ i2c_put_adapter(i2c_adap);
+ i2c_del_driver(&lcd_i2c_driver);
+ i2c_unregister_device(i2c_client);
+ break;
+
+ // INVALID
+ default:
+ return;
+ }
+}
+
+module_init(lcd_setup_init);
+module_exit(lcd_setup_exit);
diff --git a/drivers/video/wmt/devices/lcd-spi.c b/drivers/video/wmt/devices/lcd-spi.c
new file mode 100755
index 00000000..2782806a
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd-spi.c
@@ -0,0 +1,786 @@
+#include <linux/wait.h>
+#include <linux/delay.h>
+#include <linux/gpio.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/fs.h>
+#include <linux/slab.h>
+#include <linux/init.h>
+#include <linux/list.h>
+#include <linux/irq.h>
+#include <linux/jiffies.h>
+#include <linux/uaccess.h>
+#include <linux/delay.h>
+#include <linux/interrupt.h>
+#include <linux/io.h>
+#include <linux/platform_device.h>
+#include <linux/gpio.h>
+#include <linux/miscdevice.h>
+#include <linux/spinlock.h>
+
+#include <mach/hardware.h>
+
+#include <linux/spi/spi.h>
+#include <mach/wmt_iomux.h>
+
+#define wmt_spi_lcd_err(fmt, args...) printk("[%s-%d] %s: " fmt, __FILE__,__LINE__,__FUNCTION__, ## args)
+#define wmt_spi_lcd_printk(fmt, ...) wmt_spi_lcd_err(fmt, ##__VA_ARGS__)
+
+#define WMT_SPI_READ_BUF_MAX_LEN 200
+#define WMT_SPI_LCD_SPI_MODE SPI_MODE_0
+
+#define WMT_SPI_LCD_A0_PIN WMT_PIN_GP1_GPIO10
+
+#define WMT_LCD_POWER_PIN WMT_PIN_GP0_GPIO0
+#define WMT_WIFI_POWER_PIN WMT_PIN_GP1_GPIO14
+#define WMT_AIT_POWER_PIN WMT_PIN_GP1_GPIO12
+#define WMT_PRINT_POWER_PIN WMT_PIN_GP62_WAKEUP4
+#define WMT_OTG_PRINT_SWITCH_PIN WMT_PIN_GP21_HDMIDCSDA
+
+#if 0
+typedef enum{
+ WMT_SPI_LCD_STATE_NULL,
+ WMT_SPI_LCD_STATE_HAPPY,
+ WMT_SPI_LCD_STATE_SUN,
+ WMT_SPI_LCD_STATE_CLOUDY,
+ WMT_SPI_LCD_STATE_SAD,
+ WMT_SPI_LCD_STATE_PRINT,
+ WMT_SPI_LCD_STATE_MESSAGE,
+ WMT_SPI_LCD_STATE_MAX
+}wmt_sub_lcd_state;
+#endif
+#define WMT_SPI_LCD_IOCTL_MAGIC 0x11
+#define WMT_SPI_LCD_STATE_HAPPY _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 0, int)
+#define WMT_SPI_LCD_STATE_SUN _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 1, int)
+#define WMT_SPI_LCD_STATE_CLOUDY _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 2, int)
+#define WMT_SPI_LCD_STATE_SAD _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 3, int)
+#define WMT_SPI_LCD_STATE_PRINT _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 4, int)
+#define WMT_SPI_LCD_STATE_MESSAGE _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 5, int)
+
+#define WMT_WIFI_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 20, int)
+#define WMT_AIT_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 21, int)
+#define WMT_PRINT_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 22, int)
+#define WMT_SWITCH_OTG_PRINT _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 23, int)
+
+#define WMT_GET_LCD_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 30, int)
+#define WMT_GET_WIFI_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 31, int)
+#define WMT_GET_AIT_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 32, int)
+#define WMT_GET_PRINT_POWER _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 33, int)
+#define WMT_GET_OTG_PRINT_STATE _IOW(WMT_SPI_LCD_IOCTL_MAGIC, 34, int)
+
+static int wmt_sub_lcd_display_state=WMT_SPI_LCD_STATE_HAPPY;
+
+struct wmt_spi_lcd_dev {
+ wait_queue_head_t read_wq;
+ struct mutex read_mutex;
+ struct spi_device *spi;
+ struct miscdevice sub_lcd_misc_device;
+ struct kobject *kobj;
+ bool irq_enabled;
+ spinlock_t irq_enabled_lock;
+ int lcd_val;
+ int wifi_val;
+ int ait_val;
+ int print_val;
+ int otg_print_state;
+};
+
+struct wmt_spi_lcd_dev *wmt_spi_lcd_pContext;
+
+const char wmt_spi_lcd_display_happy[][64]={
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xC0,0xE0,
+ 0xF0,0xF0,0xF8,0xF8,0xF8,0x7C,0x7C,0x7C,0x3E,0x3E,0x3E,0x3E,0x1E,0x1F,0x1F,0x1F,
+ 0x1F,0x1F,0x1F,0x1E,0x3E,0x3E,0x3E,0x3E,0x7E,0x7C,0x7C,0xFC,0xF8,0xF8,0xF0,0xF0,
+ 0xE0,0xC0,0xC0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x80,0xE0,0xF0,0xF8,0xFC,0xFE,0x7F,0x3F,0x1F,0x0F,0x07,
+ 0x03,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x03,
+ 0x07,0x0F,0x0F,0x1F,0x3F,0xFE,0xFC,0xF8,0xF0,0xE0,0xC0,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x80,0xF0,0xFC,0xFF,0xFF,0xFF,0x1F,0x07,0x01,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xC0,0xF0,0xF0,0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,0xF0,0xE0,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0xC0,0xF0,0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,0xF0,0xE0,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x07,0x1F,0x7F,0xFF,0xFF,0xFC,0xF0,0x80,0x00,
+ 0xF0,0xFF,0xFF,0xFF,0xFF,0x07,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x07,0x1F,0x1F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x1F,0x0F,0x01,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x0F,0x1F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x1F,0x0F,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xFF,0xFF,0xFF,0xFF,0xF8,
+ 0x0F,0xFF,0xFF,0xFF,0xFF,0xE0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xC0,0xFF,0xFF,0xFF,0xFF,0x3F,
+ 0x00,0x01,0x0F,0x3F,0xFF,0xFF,0xFE,0xF8,0xE0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x07,0x0F,0x3F,0x7F,0x7F,0xFC,0xF8,0xF0,0xF0,0xE0,0xE0,0xE0,0xE0,
+ 0xE0,0xE0,0xE0,0xF0,0xF0,0xF8,0xFC,0x7F,0x7F,0x3F,0x0F,0x07,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xF0,0xFE,0xFF,0xFF,0x7F,0x1F,0x03,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x03,0x07,0x0F,0x3F,0x7F,0xFF,0xFC,0xF8,0xF0,0xE0,0xC0,
+ 0xC0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x03,0x03,0x03,0x03,0x03,
+ 0x03,0x03,0x03,0x03,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,
+ 0xC0,0xE0,0xF0,0xF8,0xFC,0xFE,0x7F,0x3F,0x1F,0x07,0x03,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x03,0x07,0x07,
+ 0x0F,0x1F,0x1F,0x3F,0x3E,0x3E,0x7E,0x7C,0x7C,0x7C,0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,
+ 0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,0x7C,0x7C,0x7C,0x7C,0x3E,0x3E,0x3F,0x1F,0x1F,0x0F,
+ 0x0F,0x07,0x03,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00
+};
+
+char wmt_spi_lcd_display_sad[][64]={
+ /* ͼÏñ C:\Users\admin\Desktop\androd camera\yuhao\sad.bmp 64x64 */
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xE0,0xE0,
+ 0xF0,0xF0,0xF8,0xF8,0x7C,0x7C,0x7E,0x3E,0x3E,0x3E,0x1E,0x1F,0x1F,0x1F,0x1F,0x1F,
+ 0x1F,0x1F,0x1F,0x1F,0x1F,0x1E,0x3E,0x3E,0x3E,0x7E,0x7C,0x7C,0xF8,0xF8,0xF0,0xF0,
+ 0xE0,0xE0,0xC0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0xC0,0xE0,0xF0,0xFC,0xFE,0xFF,0x3F,0x1F,0x0F,0x07,0x07,
+ 0x03,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x03,
+ 0x07,0x07,0x0F,0x1F,0x3F,0xFF,0xFE,0xFC,0xF0,0xE0,0xC0,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0xC0,0xF0,0xFC,0xFF,0xFF,0x7F,0x0F,0x07,0x01,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xE0,0xF0,0xF8,0xF8,0xFC,0xFC,0xFC,0xF8,0xF8,0xF0,0xE0,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0xE0,0xF0,0xF8,0xF8,0xFC,0xFC,0xFC,0xF8,0xF8,0xF0,0xE0,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x07,0x0F,0x7F,0xFF,0xFF,0xFC,0xF0,0xC0,0x00,
+ 0xF8,0xFF,0xFF,0xFF,0xFF,0x07,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x07,0x1F,0x1F,0x3F,0x3F,0x3F,0x3F,0x3F,0x1F,0x1F,0x07,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x07,0x1F,0x1F,0x3F,0x3F,0x3F,0x3F,0x3F,0x1F,0x1F,0x07,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xFF,0xFF,0xFF,0xFF,0xF8,
+ 0x1F,0xFF,0xFF,0xFF,0xFF,0xE0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xC0,0xC0,0xE0,0xE0,0xE0,0xE0,
+ 0xE0,0xE0,0xC0,0xC0,0xC0,0x80,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xC0,0xFF,0xFF,0xFF,0xFF,0x1F,
+ 0x00,0x03,0x0F,0x3F,0xFF,0xFF,0xFE,0xF0,0xE0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x40,0xF0,0xFC,0xFE,0xFF,0x7F,0x1F,0x0F,0x07,0x07,0x03,0x03,0x03,0x03,
+ 0x03,0x03,0x07,0x07,0x0F,0x1F,0x3F,0xFF,0xFE,0xFC,0xF8,0x60,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xE0,0xF0,0xFE,0xFF,0xFF,0x3F,0x0F,0x03,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x03,0x07,0x0F,0x3F,0x7F,0xFE,0xFC,0xF8,0xF0,0xE0,0xC0,
+ 0xC0,0x80,0x00,0x00,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x00,0x00,0x00,0x00,0x80,0xC0,
+ 0xC0,0xE0,0xF0,0xF8,0xFC,0xFE,0x7F,0x3F,0x1F,0x07,0x03,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x03,0x07,0x07,
+ 0x0F,0x1F,0x1F,0x1F,0x3E,0x3E,0x7E,0x7C,0x7C,0x7C,0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,
+ 0xF8,0xF8,0xF8,0xF8,0xF8,0xF8,0x7C,0x7C,0x7C,0x7E,0x3E,0x3E,0x1F,0x1F,0x1F,0x0F,
+ 0x07,0x07,0x03,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+};
+
+char wmt_spi_lcd_display_print[][64]={
+ /* ͼÏñ C:\Users\admin\Desktop\androd camera\yuhao\print.bmp 64x64 */
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFE,0xFF,0xFF,0xFF,0xFF,0x1F,
+ 0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,0x1F,
+ 0x1F,0x1F,0x1F,0x1F,0x1F,0xFF,0xFF,0xFF,0x1E,0x3C,0x78,0xF0,0xF0,0xE0,0xC0,0x80,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xE0,0xE0,0xE0,0xE0,0xE1,0xE3,0xE7,0xEF,
+ 0xFF,0xFE,0xFC,0xF8,0xF0,0xC0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xF8,0xFE,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFE,0xF8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xF0,0xFC,0xFE,0xFE,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFE,0xFE,0xFC,0xF8,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0x03,0x03,0x03,0x03,0x03,0x03,0x83,0x83,0x83,0x83,0x83,
+ 0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,
+ 0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,0x83,
+ 0x83,0x83,0x83,0x83,0x83,0x03,0x03,0x03,0x03,0x03,0x03,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x00,0x7F,0xFF,0xFF,0xFF,0xFF,0x0F,
+ 0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,
+ 0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,0x0F,
+ 0x0F,0xFF,0xFF,0xFF,0xFF,0x7F,0x00,0x00,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0x01,0x07,0x0F,0x1F,0x1F,0x3F,0x3F,0x3F,0x3F,0x3F,0xBF,0xFF,0xFF,0xFF,0xFF,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0xFF,0xFF,0xFF,0xFF,0xBF,0x3F,0x3F,0x3F,0x3F,0x3F,0x1F,0x1F,0x0F,0x07,0x03,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7F,0xFF,0xFF,0xFF,0xFF,0xF0,
+ 0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,
+ 0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,0xF0,
+ 0xF0,0xFF,0xFF,0xFF,0xFF,0x7F,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+};
+
+char wmt_spi_lcd_display_message[][64]={
+ /* ͼÏñ C:\Users\admin\Desktop\androd camera\yuhao\message.bmp 64x64 */
+ 0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,
+ 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,
+ 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,
+ 0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,
+ 0xFE,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xBF,0xBF,0x3F,
+ 0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,
+ 0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,0x3F,
+ 0x3F,0xBF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFE,0xFC,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x03,0x07,0x0F,0x1F,0x1F,0x3F,0x7F,0xFF,0xFF,
+ 0xFE,0xFC,0xFC,0xF8,0xF0,0xF0,0xE0,0xC0,0x80,0x80,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xC0,0xE0,0xF0,0xF8,0xF8,0xFC,0xFE,0xFF,
+ 0xFF,0x7F,0x3F,0x3F,0x1F,0x0F,0x07,0x07,0x03,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x01,0x03,0x03,0x07,0x0F,0x1F,0x1F,0x3F,0x7F,0x7F,0xFF,0xFE,0xFC,0xFC,0xF8,0xF0,
+ 0xF8,0xF8,0xFC,0xFE,0xFF,0xFF,0x7F,0x3F,0x3F,0x1F,0x0F,0x07,0x07,0x03,0x01,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x03,0x03,0x07,0x07,
+ 0x07,0x07,0x03,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,
+ 0x3F,0x7F,0xFF,0xFF,0xFF,0xFF,0xFF,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,
+ 0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,
+ 0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,
+ 0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFC,0xFF,0xFF,0xFF,0xFF,0xFF,0x7F,0x3F,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+};
+
+
+const char wmt_spi_lcd_display_sun[][64]={
+ /* ͼÏñ C:\Users\admin\Desktop\androd camera\yuhao\sun.bmp 64x64 */
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xE0,
+ 0xF0,0xF0,0xF0,0xE0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFC,0xFE,0xFF,
+ 0xFF,0xFE,0xFC,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xE0,0xF0,0xF0,0xF0,
+ 0xE0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x01,
+ 0x03,0x0F,0x3F,0x7F,0x7F,0x7E,0x78,0x00,0x00,0x00,0x00,0x80,0x80,0x87,0x8F,0x8F,
+ 0x8F,0x8F,0x87,0x80,0x80,0x00,0x00,0x00,0x30,0x7C,0x7F,0x7F,0x7F,0x3F,0x0F,0x03,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x07,0x0F,0x0F,0x1F,0x1F,0x3F,0x7E,0x7E,0x7C,0x7C,0x78,0x00,
+ 0x00,0xC0,0xE0,0xF0,0xF8,0xFC,0xFE,0x7E,0x3F,0x3F,0x1F,0x1F,0x0F,0x0F,0x0F,0x0F,
+ 0x0F,0x0F,0x0F,0x0F,0x1F,0x1F,0x3F,0x7F,0x7E,0xFC,0xFC,0xF8,0xF0,0xE0,0x80,0x00,
+ 0x30,0x78,0x7C,0x7C,0x7E,0x3F,0x3F,0x1F,0x1F,0x0F,0x0F,0x07,0x00,0x00,0x00,0x00,
+ 0x80,0xC0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xC0,0x00,0x00,0x00,0xF8,
+ 0xFF,0xFF,0xFF,0xFF,0x0F,0x03,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x03,0x1F,0xFF,0xFF,0xFF,0xFE,
+ 0xF8,0x00,0x00,0x80,0xC0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xC0,0x80,
+ 0x01,0x03,0x07,0x07,0x07,0x07,0x07,0x07,0x07,0x07,0x03,0x03,0x00,0x00,0x00,0x1F,
+ 0xFF,0xFF,0xFF,0xFF,0xF0,0xC0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xF8,0xFF,0xFF,0xFF,0x7F,
+ 0x1F,0x00,0x00,0x01,0x03,0x03,0x07,0x07,0x07,0x07,0x07,0x07,0x07,0x03,0x03,0x01,
+ 0x00,0x00,0x00,0x00,0xE0,0xF0,0xF0,0xF8,0xFC,0xFC,0x7E,0x7E,0x3F,0x3F,0x1E,0x00,
+ 0x00,0x01,0x07,0x0F,0x1F,0x3F,0x3F,0x7E,0x7E,0xFC,0xF8,0xF8,0xF0,0xF0,0xF0,0xF0,
+ 0xF0,0xF0,0xF0,0xF8,0xF8,0xF8,0xFC,0x7E,0x7F,0x3F,0x1F,0x1F,0x0F,0x03,0x01,0x00,
+ 0x0C,0x1E,0x3F,0x3F,0x7E,0x7E,0xFC,0xF8,0xF8,0xF0,0xF0,0xE0,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,
+ 0xE0,0xF0,0xFC,0xFE,0xFF,0x7F,0x1E,0x00,0x00,0x00,0x00,0x01,0x01,0xE1,0xF1,0xF9,
+ 0xF9,0xF1,0xE1,0x01,0x01,0x00,0x00,0x00,0x0C,0x3E,0x7F,0xFF,0xFE,0xFC,0xF0,0xC0,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x01,0x01,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,
+ 0x0F,0x0F,0x0F,0x07,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3F,0x7F,0xFF,
+ 0xFF,0x7F,0x3F,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x07,0x0F,0x0F,0x0F,
+ 0x07,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+};
+
+const char wmt_spi_lcd_display_cloudy[][64]={
+ /* ͼÏñ C:\Users\admin\Desktop\androd camera\yuhao\cloudy.bmp 64x64 */
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x40,0xE0,0xE0,0xE0,0xC0,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x87,0x8F,0x87,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0xC0,0xE0,0xE0,0xE0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x80,0x80,0x80,0xC0,0xC0,0xC0,0xC0,0xC0,0xC0,0xC1,0xC1,0x81,0x80,0x80,0x18,0x38,
+ 0x3C,0x1E,0x1E,0x0F,0x0F,0x07,0x07,0x07,0x07,0x07,0x07,0x07,0x0F,0x0F,0x1E,0x3E,
+ 0x7C,0xF8,0xF0,0xE0,0xC0,0x01,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xF0,0xF8,0x7C,0x3C,0x1E,0x0F,0x0F,
+ 0x07,0x07,0x03,0x03,0x03,0x03,0x01,0x01,0x03,0x03,0x03,0x03,0x07,0x07,0x0F,0x0F,
+ 0x1E,0x3C,0x7C,0xF8,0xF0,0xC0,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x03,0xFF,0xFF,0xFF,0xF0,0x00,0x00,0x00,0x20,0x70,0x70,0x70,0x70,0x20,
+ 0x00,0x00,0x00,0x80,0xC0,0xE0,0xFC,0xFF,0xFF,0x0F,0x01,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x01,0x0F,0x1F,0x1F,0x0E,0x0F,0x0F,0x07,0x07,0x0F,0x0F,0x0F,0x1E,
+ 0x1E,0x3C,0x78,0xF9,0xF3,0xC7,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0xF8,0xFE,0xFF,0x1F,0x07,0x01,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x81,0xFF,0xFF,0xFF,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x03,0x0F,0x1F,0x3F,0x7C,0xF0,0xF0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,
+ 0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,
+ 0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xE0,0xF0,
+ 0xF0,0x78,0x3E,0x3F,0x0F,0x07,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,
+ 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,
+ 0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x00,0x00,
+ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
+};
+
+extern int wmt_getsyspara(char *varname, unsigned char *varval, int *varlen);
+
+void wmt_spi_lcd_write_n_data(const char *wbuf,int wbuf_len)
+{
+ if(wbuf_len>=WMT_SPI_READ_BUF_MAX_LEN)
+ {
+ wmt_spi_lcd_err("error!!");
+ return;
+ }
+ gpio_direction_output(WMT_SPI_LCD_A0_PIN,1);
+ spi_write_then_read(wmt_spi_lcd_pContext->spi, wbuf, wbuf_len, NULL, 0);
+}
+
+void wmt_spi_lcd_write_data(unsigned char c)
+{
+ wmt_spi_lcd_write_n_data(&c,1);
+}
+
+
+//-------------------------------------------------------------------------------
+//×Ó³ÌÐòÃû³Æ:lcdwc(unsigned char c).
+//¹¦ÄÜ:ÏòÒº¾§ÏÔʾ¿ØÖÆÆ÷ËÍÖ¸Áî.
+//-------------------------------------------------------------------------------
+void wmt_spi_lcd_write_command(unsigned char c)
+{
+ gpio_direction_output(WMT_SPI_LCD_A0_PIN,0);
+ spi_write_then_read(wmt_spi_lcd_pContext->spi, &c, 1, NULL, 0);
+}
+
+void wmt_spi_lcd_reset(void) //Òº¾§ÏÔʾ¿ØÖÆÆ÷³õʼ»¯×Ó³ÌÐò
+{
+ #if 0
+ gpio_direction_output(WMT_SPI_LCD_RESET_PIN,0);
+ mdelay(200);
+ gpio_direction_output(WMT_SPI_LCD_RESET_PIN,1);
+ mdelay(200);
+ #endif
+ wmt_spi_lcd_write_command(0xe2);
+ mdelay(100);
+
+ wmt_spi_lcd_write_command(0x2c);/////set booster on
+ wmt_spi_lcd_write_command(0x2e); ////voltage regulator circuit on
+ wmt_spi_lcd_write_command(0x2f);//// voltage follower circuit on
+
+ wmt_spi_lcd_write_command(0xa0);//
+ wmt_spi_lcd_write_command(0xa6);//
+ mdelay(3);
+ wmt_spi_lcd_write_command(0xc8);//
+ mdelay(3);
+ wmt_spi_lcd_write_command(0xa2);//
+ mdelay(3);
+
+ wmt_spi_lcd_write_command(0xf8); //set booster ratio
+ mdelay(2);
+ wmt_spi_lcd_write_command(0x00); //set X2 X3 X4
+ mdelay(2);
+
+ wmt_spi_lcd_write_command(0x24);
+ mdelay(2);
+ wmt_spi_lcd_write_command(0x81);
+ mdelay(2);
+ wmt_spi_lcd_write_command(0x28);
+ mdelay(2);
+ wmt_spi_lcd_write_command(0xac);
+ mdelay(2);
+ wmt_spi_lcd_write_command(0x00);
+ mdelay(2);
+
+ wmt_spi_lcd_write_command(0xaf);
+ wmt_spi_lcd_write_command(0xA4);
+ mdelay(2);
+}
+
+
+void wmt_spi_lcd_display_picture(const char *picture)
+{
+ int page,k;
+ int skip=0;
+ for(page=0xB0;page<0xB8;page++)
+ {
+ wmt_spi_lcd_write_command(page);
+ wmt_spi_lcd_write_command(0x10);
+ wmt_spi_lcd_write_command(0x00);
+
+ for(k=0;k<8;k++)
+ wmt_spi_lcd_write_n_data(picture+skip+k*8,8);
+
+ skip+=64;
+ }
+}
+
+void wmt_spi_lcd_display_state_picture(void)
+{
+ switch(wmt_sub_lcd_display_state){
+ case WMT_SPI_LCD_STATE_HAPPY:
+ wmt_spi_lcd_display_picture(wmt_spi_lcd_display_happy[0]);
+ break;
+ case WMT_SPI_LCD_STATE_SUN:
+ wmt_spi_lcd_display_picture(wmt_spi_lcd_display_sun[0]);
+ break;
+ case WMT_SPI_LCD_STATE_CLOUDY:
+ wmt_spi_lcd_display_picture(wmt_spi_lcd_display_cloudy[0]);
+ break;
+ case WMT_SPI_LCD_STATE_SAD:
+ wmt_spi_lcd_display_picture(wmt_spi_lcd_display_sad[0]);
+ break;
+ case WMT_SPI_LCD_STATE_PRINT:
+ wmt_spi_lcd_display_picture(wmt_spi_lcd_display_print[0]);
+ break;
+ case WMT_SPI_LCD_STATE_MESSAGE:
+ wmt_spi_lcd_display_picture(wmt_spi_lcd_display_message[0]);
+ break;
+
+ default:
+ wmt_spi_lcd_printk("%s default wmt_sub_lcd_display_state=%d\n", __func__,wmt_sub_lcd_display_state);
+ }
+}
+
+
+static int wmt_spi_lcd_open(struct inode *inode, struct file *filp)
+{
+ struct wmt_spi_lcd_dev *spi_dev = container_of(filp->private_data,
+ struct wmt_spi_lcd_dev,
+ sub_lcd_misc_device);
+ filp->private_data = spi_dev;
+ wmt_spi_lcd_display_state_picture();
+ // mdelay(3000);
+
+ return 0;
+}
+
+static long wmt_spi_lcd_ioctl(struct file *file,
+ unsigned int cmd, unsigned long arg)
+{
+ void __user *argp = (void __user *)arg;
+ int interval, val;
+ struct wmt_spi_lcd_dev *spi_dev = file->private_data;
+
+ wmt_sub_lcd_display_state = cmd;
+ wmt_spi_lcd_display_state_picture();
+
+ switch(cmd){
+ case WMT_WIFI_POWER:
+ if(copy_from_user(&val, argp, sizeof(val)))
+ return -EFAULT;
+ if(val > 1)
+ return -EINVAL;
+ if(val)
+ {
+ gpio_direction_output(WMT_WIFI_POWER_PIN,1);
+ }
+ else
+ {
+ gpio_direction_output(WMT_WIFI_POWER_PIN,0);
+ }
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_WIFI_POWER_PIN val=%d\n",val);
+ break;
+ case WMT_AIT_POWER:
+ if(copy_from_user(&val, argp, sizeof(val)))
+ return -EFAULT;
+ if(val > 1)
+ return -EINVAL;
+ if(val)
+ {
+ gpio_direction_output(WMT_AIT_POWER_PIN,1);
+ }
+ else
+ {
+ gpio_direction_output(WMT_AIT_POWER_PIN,0);
+ }
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_AIT_POWER_PIN val=%d\n",val);
+ break;
+ case WMT_PRINT_POWER:
+ if(copy_from_user(&val, argp, sizeof(val)))
+ return -EFAULT;
+ if(val > 1)
+ return -EINVAL;
+ if(val)
+ {
+ gpio_direction_output(WMT_PRINT_POWER_PIN,1);
+ }
+ else
+ {
+ gpio_direction_output(WMT_PRINT_POWER_PIN,0);
+ }
+
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_PRINT_POWER_PIN val=%d\n",val);
+ break;
+ case WMT_SWITCH_OTG_PRINT:
+ if(copy_from_user(&val, argp, sizeof(val)))
+ return -EFAULT;
+ if(val > 1)
+ return -EINVAL;
+ if(val)
+ {
+ gpio_direction_output(WMT_OTG_PRINT_SWITCH_PIN,1);
+ }
+ else
+ {
+ gpio_direction_output(WMT_OTG_PRINT_SWITCH_PIN,0);
+ }
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_OTG_PRINT_SWITCH_PIN val=%d\n",val);
+ break;
+ case WMT_GET_LCD_POWER:
+ spi_dev->lcd_val = wmt_gpio_getpull(WMT_LCD_POWER_PIN);
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_LCD_POWER=%d\n", spi_dev->lcd_val);
+ interval = spi_dev->lcd_val;
+ if(copy_to_user(argp, &interval, sizeof(interval)))
+ {
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_LCD_POWER fail\n");
+ return -EFAULT;
+ }
+ break;
+
+ case WMT_GET_WIFI_POWER:
+ spi_dev->wifi_val = wmt_gpio_getpull(WMT_WIFI_POWER_PIN);
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_WIFI_POWER=%d\n", spi_dev->wifi_val);
+ interval = spi_dev->wifi_val;
+ if(copy_to_user(argp, &interval, sizeof(interval)))
+ {
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_WIFI_POWER fail\n");
+ return -EFAULT;
+ }
+ break;
+
+ case WMT_GET_AIT_POWER:
+ spi_dev->ait_val = wmt_gpio_getpull(WMT_AIT_POWER_PIN);
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_AIT_POWER=%d\n", spi_dev->ait_val);
+ interval = spi_dev->ait_val;
+ if(copy_to_user(argp, &interval, sizeof(interval)))
+ {
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_AIT_POWER fail\n");
+ return -EFAULT;
+ }
+ break;
+
+ case WMT_GET_PRINT_POWER:
+ spi_dev->print_val = wmt_gpio_getpull(WMT_PRINT_POWER_PIN);
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_PRINT_POWER=%d\n", spi_dev->print_val);
+ interval = spi_dev->print_val;
+ if(copy_to_user(argp, &interval, sizeof(interval)))
+ {
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_GET_PRINT_POWER fail\n");
+ return -EFAULT;
+ }
+ break;
+ case WMT_GET_OTG_PRINT_STATE:
+ spi_dev->otg_print_state = wmt_gpio_getpull(WMT_OTG_PRINT_SWITCH_PIN);
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_OTG_PRINT_SWITCH_PIN=%d\n", spi_dev->otg_print_state);
+ interval = spi_dev->otg_print_state;
+ if(copy_to_user(argp, &interval, sizeof(interval)))
+ {
+ wmt_spi_lcd_printk("wmt_spi_lcd_ioctl WMT_OTG_PRINT_SWITCH_PIN fail\n");
+ return -EFAULT;
+ }
+ break;
+
+ default:
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+
+static const struct file_operations wmt_spi_lcd_fops = {
+ .owner = THIS_MODULE,
+ .llseek = no_llseek,
+ .open = wmt_spi_lcd_open,
+ .unlocked_ioctl = wmt_spi_lcd_ioctl,
+};
+
+
+static ssize_t wmt_spi_lcd_cat_dbg(struct device *dev, struct device_attribute *attr, char *buf)
+{
+ return sprintf(buf, "%d\n", wmt_sub_lcd_display_state);
+}
+
+static ssize_t wmt_spi_lcd_echo_dbg(struct device *dev, struct device_attribute *attr, const char *buf, size_t count)
+{
+ sscanf(buf,"%d", &wmt_sub_lcd_display_state);
+ wmt_sub_lcd_display_state += WMT_SPI_LCD_STATE_HAPPY;
+ wmt_spi_lcd_display_state_picture();
+ return count;
+}
+
+static DEVICE_ATTR(dbg, S_IRUGO | S_IWUSR, wmt_spi_lcd_cat_dbg, wmt_spi_lcd_echo_dbg);
+
+static struct attribute *sub_lcd_attributes[] = {
+ &dev_attr_dbg.attr,
+ NULL
+};
+
+static const struct attribute_group wmt_spi_lcd_group = {
+ .attrs = sub_lcd_attributes,
+};
+
+static int wmt_spi_lcd_sysfs_create_group(struct wmt_spi_lcd_dev *spi_dev, const struct attribute_group *group)
+{
+ int err;
+
+ spi_dev->kobj = kobject_create_and_add("wmt_spi_lcd", NULL) ;
+ if(!spi_dev->kobj){
+ wmt_spi_lcd_printk("kobj create failed.\n");
+ return -ENOMEM;
+ }
+
+ /* Register sysfs hooks */
+ err = sysfs_create_group(spi_dev->kobj, group);
+ if (err < 0){
+ kobject_del(spi_dev->kobj);
+ wmt_spi_lcd_printk("Create sysfs group failed!\n");
+ return -ENOMEM;
+ }
+
+ return 0;
+}
+
+static void wmt_spi_lcd_sysfs_remove_group(struct wmt_spi_lcd_dev *spi_dev, const struct attribute_group *group)
+{
+ sysfs_remove_group(spi_dev->kobj, group);
+ kobject_del(spi_dev->kobj);
+ return;
+}
+static int wmt_spi_lcd_probe(struct spi_device *spi)
+{
+ struct wmt_spi_lcd_dev *spi_dev = NULL;
+ int ret = 0;
+
+ ret = gpio_request(WMT_SPI_LCD_A0_PIN, "sub_lcd_control");
+ if (ret)
+ goto err_exit;
+
+ spi_dev = kzalloc(sizeof(*spi_dev), GFP_KERNEL);
+ if (spi_dev == NULL) {
+ wmt_spi_lcd_printk("failed to allocate memory for module data\n");
+ return -ENOMEM;
+ }
+
+ spi_dev->spi = spi;
+ spi_set_drvdata(spi, spi_dev);
+
+ ret = wmt_spi_lcd_sysfs_create_group(spi_dev, &wmt_spi_lcd_group);
+ if(ret < 0){
+ wmt_spi_lcd_printk("create sysfs group failed.\n");
+ goto exit_create_group;
+ }
+
+ /* init mutex and queues */
+ init_waitqueue_head(&spi_dev->read_wq);
+ mutex_init(&spi_dev->read_mutex);
+ spin_lock_init(&spi_dev->irq_enabled_lock);
+
+ spi_dev->sub_lcd_misc_device.minor = MISC_DYNAMIC_MINOR;
+ spi_dev->sub_lcd_misc_device.name = "wmt_spi_lcd";
+ spi_dev->sub_lcd_misc_device.fops = &wmt_spi_lcd_fops;
+
+ ret = misc_register(&spi_dev->sub_lcd_misc_device);
+ if (ret) {
+ wmt_spi_lcd_printk("%s : misc_register failed\n", __FILE__);
+ goto err_misc_register;
+ }
+
+ wmt_spi_lcd_pContext = spi_dev;
+
+ return 0;
+
+err_misc_register:
+ mutex_destroy(&spi_dev->read_mutex);
+ kfree(spi_dev);
+err_exit:
+ gpio_free(WMT_SPI_LCD_A0_PIN);
+exit_create_group:
+ destroy_workqueue(&spi_dev->read_wq);
+
+ return ret;
+}
+
+static int wmt_spi_lcd_remove(struct spi_device *spi)
+{
+ struct wmt_spi_lcd_dev *spi_dev = wmt_spi_lcd_pContext;
+ misc_deregister(&spi_dev->sub_lcd_misc_device);
+ mutex_destroy(&spi_dev->read_mutex);
+ gpio_free(WMT_SPI_LCD_A0_PIN);
+
+ wmt_spi_lcd_sysfs_remove_group(spi_dev, &wmt_spi_lcd_group);
+ kfree(spi_dev);
+
+ return 0;
+}
+
+static int wmt_spi_lcd_suspend(struct spi_device *spi, pm_message_t mesg)
+{
+ printk("%s\n",__func__);
+
+ return 0;
+}
+
+static int wmt_spi_lcd_resume(struct spi_device *spi)
+{
+ printk("%s\n",__func__);
+ wmt_spi_lcd_reset();
+ wmt_spi_lcd_display_state_picture();
+ return 0;
+}
+
+
+static struct spi_driver wmt_spi_lcd_driver = {
+ .driver = {
+ .name = "wmt_spi_lcd",
+ .bus = &spi_bus_type,
+ .owner = THIS_MODULE,
+ },
+ .probe = wmt_spi_lcd_probe,
+ .remove = wmt_spi_lcd_remove,
+ .suspend = wmt_spi_lcd_suspend,
+ .resume = wmt_spi_lcd_resume,
+};
+
+static struct spi_board_info wmt_spi_lcd_board_info[] __initdata = {
+ {
+ .modalias = "wmt_spi_lcd",
+ .bus_num = 0,
+ .chip_select = 0,
+ .max_speed_hz = 12000000,
+ .irq = -1,
+ .mode = SPI_MODE_0,
+ .controller_data = NULL,
+ },
+};
+
+static int wmt_spi_lcd_register_device (void)
+{
+ int ret = 0;
+ ret = spi_register_board_info(wmt_spi_lcd_board_info, ARRAY_SIZE(wmt_spi_lcd_board_info));
+ return ret;
+}
+
+static int wmt_spi_lcd_init(void)
+{
+ int ret = 0;
+ unsigned char buf[40];
+ int buflen = 40;
+ unsigned int value;
+ char *endp;
+
+ if(wmt_getsyspara("wmt.support.lcd.spi", buf, &buflen) == 0) {
+ value = simple_strtoul(buf, &endp, 0);
+ if(value == 0)
+ return -1;
+ } else
+ return -1;
+
+ if (wmt_spi_lcd_register_device()<0)
+ {
+ wmt_spi_lcd_printk(" Error to run wmt_spi_lcd_register_device()!\n");
+ return -1;
+ }
+
+ ret = spi_register_driver(&wmt_spi_lcd_driver);
+ if (ret != 0)
+ {
+ wmt_spi_lcd_printk("failed to register wmt_spi_lcd SPI driver: %d\n", ret);
+ goto out;
+ }
+
+out:
+ return ret;
+}
+
+static void wmt_spi_lcd_exit(void)
+{
+ wmt_spi_lcd_printk("Unloading wmt_spi_lcd driver\n");
+ spi_unregister_driver(&wmt_spi_lcd_driver);
+}
+
+module_init(wmt_spi_lcd_init);
+module_exit(wmt_spi_lcd_exit);
+
+MODULE_LICENSE("GPL");
+MODULE_DESCRIPTION(DRIVER_DESC);
diff --git a/drivers/video/wmt/devices/lcd.c b/drivers/video/wmt/devices/lcd.c
new file mode 100644
index 00000000..195d3483
--- /dev/null
+++ b/drivers/video/wmt/devices/lcd.c
@@ -0,0 +1,422 @@
+/*++
+ * linux/drivers/video/wmt/lcd.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LCD_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include <linux/gpio.h>
+#include "../lcd.h"
+#include "../vout.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define LCD_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LCD_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lcd_xxx_t; *//*Example*/
+struct lcd_device_t {
+ lcd_parm_t* (*get_parm)(int arg);
+};
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lcd.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lcd_xxx; *//*Example*/
+struct lcd_device_t lcd_device_array[LCD_PANEL_MAX];
+int lcd_panel_on = 1;
+int lcd_pwm_enable;
+lcd_panel_t lcd_panel_id = LCD_PANEL_MAX;
+int lcd_panel_bpp = 24;
+
+vout_dev_t lcd_vout_dev_ops;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lcd_xxx(void); *//*Example*/
+lcd_panel_t lcd_lvds_id = LCD_PANEL_MAX;
+int lcd_type;
+
+/*----------------------- Function Body --------------------------------------*/
+/*----------------------- Backlight --------------------------------------*/
+void lcd_set_type(int type)
+{
+ lcd_type = type; /* 0-LCD, 1-LVDS */
+}
+
+int lcd_get_type(void)
+{
+ return lcd_type;
+}
+
+static struct {
+ int gpio;
+ int active;
+} lcd_power;
+
+/*----------------------- Function Body --------------------------------------*/
+static int parse_uboot_param(void)
+{
+ char buf[64];
+ size_t l = sizeof(buf);
+ int ret;
+
+ if (wmt_getsyspara("wmt.lcd.power", buf, &l)) {
+ pr_err("please set wmt.lcd.power\n");
+ return -EINVAL;
+ }
+
+ ret = sscanf(buf, "%d:%d", &lcd_power.gpio, &lcd_power.active);
+ if (ret < 2) {
+ pr_err("Batt: bat env param: %s\n", buf);
+ return -EINVAL;
+ }
+
+ printk("lcd power: gpio-%d, active %d\n", lcd_power.gpio, lcd_power.active);
+
+ ret = gpio_request(lcd_power.gpio, "lcd power");
+ if (ret) {
+ pr_err("request gpio %d failed\n", lcd_power.gpio);
+ return ret;
+ }
+
+ return 0;
+}
+
+void lcd_power_on(bool on)
+{
+ if (lcd_power.gpio < 0)
+ return;
+
+ gpio_direction_output(lcd_power.gpio, on ? lcd_power.active : !lcd_power.active);
+}
+
+/*----------------------- Backlight --------------------------------------*/
+void lcd_set_lvds_id(int id)
+{
+ lcd_lvds_id = id;
+}
+
+int lcd_get_lvds_id(void)
+{
+ return lcd_lvds_id;
+}
+
+void lcd_set_parm(int id, int bpp)
+{
+ lcd_panel_id = id;
+ lcd_panel_bpp = bpp;
+}
+
+lcd_parm_t *lcd_get_parm(lcd_panel_t id, unsigned int arg)
+{
+ struct lcd_device_t *p;
+
+ p = &lcd_device_array[id];
+ if (p && p->get_parm)
+ return p->get_parm(arg);
+ return 0;
+}
+
+vout_dev_t *lcd_get_dev(void)
+{
+ if (lcd_panel_id >= LCD_PANEL_MAX)
+ return 0;
+ return &lcd_vout_dev_ops;
+}
+
+#ifdef CONFIG_KERNEL
+static DEFINE_SEMAPHORE(lcd_sem);
+#endif
+void lcd_set_mutex(int lock)
+{
+#ifdef CONFIG_KERNEL
+ if (lock)
+ down(&lcd_sem);
+ else
+ up(&lcd_sem);
+#endif
+}
+
+
+/* #define CONFIG_PANEL_SET_CLOCK */
+void lcd_set_enable(int enable)
+{
+#ifdef CONFIG_PANEL_SET_CLOCK
+ int hdmi_off;
+#endif
+
+ DBG_MSG("%d\n", enable);
+ if (!p_lcd)
+ return;
+
+ lcd_set_mutex(1);
+#ifdef CONFIG_PANEL_SET_CLOCK
+ /* enable NA12 in Panel on */
+ hdmi_off = (govrh_get_MIF_enable(p_govrh)) ? 0 : 1;
+ if (hdmi_off && enable)
+ vpp_set_clock_enable(DEV_NA12, enable, 1);
+#endif
+
+ if (enable) {
+ if (p_lcd->initial)
+ p_lcd->initial();
+ else {
+ lcd_enable_signal(1); /* singal enable */
+ //REG32_VAL(GPIO_BASE_ADDR + 0x80) |= 0x801;
+ //REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x801;
+ }
+ } else {
+ if (p_lcd->uninitial)
+ p_lcd->uninitial();
+ else {
+ lcd_enable_signal(0); /* singal disable */
+ //REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x801;
+ }
+ }
+
+#ifdef CONFIG_PANEL_SET_CLOCK
+ /* disable VPP & NA12 clock in Panel off */
+ if (hdmi_off) {
+ vpp_set_clock_enable(DEV_VPP, enable, 1);
+ vpp_set_clock_enable(DEV_GOVRHD, enable, 1);
+ vpp_set_clock_enable(DEV_HDMILVDS, enable, 1);
+ if (enable == 0)
+ vpp_set_clock_enable(DEV_NA12, enable, 1);
+ }
+#endif
+ lcd_set_mutex(0);
+}
+
+void lcd_enable_signal(int enable)
+{
+ int hdmi_off;
+
+ DBG_MSG("%d\n", enable);
+ if (lcd_get_type()) { /* LVDS */
+ /* TODO */
+ } else { /* LCD */
+ govrh_set_dvo_enable(p_govrh2, enable);
+ }
+
+ hdmi_off = (govrh_get_MIF_enable(p_govrh)) ? 0 : 1;
+ vpp_set_clock_enable(DEV_DVO, (hdmi_off && !enable) ? 0 : 1, 1);
+}
+
+#ifdef __KERNEL__
+/*----------------------- LCD --------------------------------------*/
+static int __init lcd_arg_panel_id
+(
+ char *str /*!<; // argument string */
+)
+{
+ sscanf(str, "%d", (int *) &lcd_panel_id);
+ if (lcd_panel_id >= LCD_PANEL_MAX)
+ lcd_panel_id = LCD_PANEL_MAX;
+ DBGMSG(KERN_INFO "set lcd panel id = %d\n", lcd_panel_id);
+ return 1;
+} /* End of lcd_arg_panel_id */
+
+__setup("lcdid=", lcd_arg_panel_id);
+#endif
+int lcd_panel_register(int no, void (*get_parm)(int mode))
+{
+ struct lcd_device_t *p;
+
+ if (no >= LCD_PANEL_MAX) {
+ DBGMSG(KERN_ERR "*E* lcd device no max is %d !\n",
+ LCD_PANEL_MAX);
+ return -1;
+ }
+
+ p = &lcd_device_array[no];
+ if (p->get_parm) {
+ DBGMSG(KERN_ERR "*E* lcd device %d exist !\n", no);
+ return -1;
+ }
+ p->get_parm = (void *) get_parm;
+ return 0;
+} /* End of lcd_device_register */
+
+/*----------------------- vout device plugin --------------------------------------*/
+void lcd_set_power_down(int enable)
+{
+ static int save_state = -1;
+
+ if (save_state != enable) {
+ /* lcd enable control by user */
+ lcd_power_on(enable ? false : true);
+ lcd_set_enable(enable ? false : true);
+ save_state = enable;
+ }
+}
+
+static void wmt_config_govrh_polar(vout_t *vo)
+{
+ /* wmt.display.polar [clock polar]:[hsync polart]:[vsync polar]*/
+ char buf[64];
+ size_t l = sizeof(buf);
+ int clk_pol, hsync_pol, vsync_pol;
+
+ if (wmt_getsyspara("wmt.display.polar", buf, &l)) {
+ return;
+ }
+
+ sscanf(buf, "%d:%d:%d", &clk_pol, &hsync_pol, &vsync_pol);
+
+ printk("govrh polar: clk-pol %d, hsync %d, vsync %d\n", clk_pol, hsync_pol, vsync_pol);
+ govrh_set_dvo_clock_delay(vo->govr, clk_pol ? 0 : 1, 0);
+ govrh_set_dvo_sync_polar(vo->govr, hsync_pol ? 0 : 1, vsync_pol ? 0 : 1);
+}
+
+int lcd_set_mode(unsigned int *option)
+{
+ vout_t *vo;
+ vout_inf_mode_t inf_mode;
+
+ DBG_MSG("option %d,%d\n", option[0], option[1]);
+
+ vo = lcd_vout_dev_ops.vout;
+ inf_mode = vo->inf->mode;
+ if (option) {
+ unsigned int capability;
+
+ if (lcd_panel_id == 0)
+ p_lcd = lcd_get_oem_parm(vo->resx, vo->resy);
+ else
+ p_lcd = lcd_get_parm(lcd_panel_id, lcd_panel_bpp);
+
+ if (!p_lcd) {
+ DBG_ERR("lcd %d not support\n", lcd_panel_id);
+ return -1;
+ }
+ DBG_MSG("[%s] %s (id %d,bpp %d)\n", vout_inf_str[inf_mode],
+ p_lcd->vmode.name, lcd_panel_id, lcd_panel_bpp);
+ capability = p_lcd->capability;
+ switch (inf_mode) {
+ case VOUT_INF_LVDS:
+ lvds_set_sync_polar(
+ (capability & LCD_CAP_HSYNC_HI) ? 0 : 1,
+ (capability & LCD_CAP_VSYNC_HI) ? 0 : 1);
+ lvds_set_rgb_type(lcd_panel_bpp);
+ break;
+ case VOUT_INF_DVI:
+ govrh_set_dvo_clock_delay(vo->govr,
+ (capability & LCD_CAP_CLK_HI) ? 0 : 1, 0);
+ govrh_set_dvo_sync_polar(vo->govr,
+ (capability & LCD_CAP_HSYNC_HI) ? 0 : 1,
+ (capability & LCD_CAP_VSYNC_HI) ? 0 : 1);
+ switch (lcd_panel_bpp) {
+ case 15:
+ govrh_IGS_set_mode(vo->govr, 0, 1, 1);
+ break;
+ case 16:
+ govrh_IGS_set_mode(vo->govr, 0, 3, 1);
+ break;
+ case 18:
+ govrh_IGS_set_mode(vo->govr, 0, 2, 1);
+ break;
+ case 24:
+ govrh_IGS_set_mode(vo->govr, 0, 0, 0);
+ break;
+ default:
+ break;
+ }
+ break;
+ default:
+ break;
+ }
+ } else
+ p_lcd = 0;
+
+ wmt_config_govrh_polar(vo);
+ return 0;
+}
+
+int lcd_check_plugin(int hotplug)
+{
+ return (p_lcd) ? 1 : 0;
+}
+
+int lcd_get_edid(char *buf)
+{
+ return 1;
+}
+
+int lcd_config(vout_info_t *info)
+{
+ info->resx = p_lcd->vmode.xres;
+ info->resy = p_lcd->vmode.yres;
+ info->fps = p_lcd->vmode.refresh;
+ return 0;
+}
+
+int lcd_init(vout_t *vo)
+{
+ DBG_MSG("%d\n", lcd_panel_id);
+
+ /* vo_set_lcd_id(LCD_CHILIN_LW0700AT9003); */
+ if (lcd_panel_id >= LCD_PANEL_MAX)
+ return -1;
+
+ if (lcd_panel_id == 0)
+ p_lcd = lcd_get_oem_parm(vo->resx, vo->resy);
+ else
+ p_lcd = lcd_get_parm(lcd_panel_id, 24);
+
+ if (p_lcd == 0)
+ return -1;
+
+ /* set default parameters */
+ vo->resx = p_lcd->vmode.xres;
+ vo->resy = p_lcd->vmode.yres;
+ vo->pixclk = PICOS2KHZ(p_lcd->vmode.pixclock) * 1000;
+ return 0;
+}
+
+vout_dev_t lcd_vout_dev_ops = {
+ .name = "LCD",
+ .mode = VOUT_INF_DVI,
+ .capability = VOUT_DEV_CAP_FIX_RES + VOUT_DEV_CAP_FIX_PLUG,
+
+ .init = lcd_init,
+ .set_power_down = lcd_set_power_down,
+ .set_mode = lcd_set_mode,
+ .config = lcd_config,
+ .check_plugin = lcd_check_plugin,
+ .get_edid = lcd_get_edid,
+};
+
+int lcd_module_init(void)
+{
+ if (parse_uboot_param()) {
+ lcd_power.gpio = -1;
+ }
+
+ vout_device_register(&lcd_vout_dev_ops);
+ return 0;
+} /* End of lcd_module_init */
+module_init(lcd_module_init);
+/*--------------------End of Function Body -----------------------------------*/
+#undef LCD_C
diff --git a/drivers/video/wmt/devices/ts8224b.h b/drivers/video/wmt/devices/ts8224b.h
new file mode 100755
index 00000000..87866284
--- /dev/null
+++ b/drivers/video/wmt/devices/ts8224b.h
@@ -0,0 +1,446 @@
+#ifndef _TS8224B_H_
+#define _TS8224B_H_
+
+//Enable Page1 hsd4.0
+0xFF00, 0x55,
+0xFF01, 0xAA,
+0xFF02, 0x52,
+0xFF03, 0x08,
+0xFF04, 0x80,
+
+0xF200, 0x00,
+0xF201, 0x84,
+0xF202, 0x02,
+
+0xF40A, 0x13,
+
+0xF000, 0x55,
+0xF001, 0xAA,
+0xF002, 0x52,
+0xF003, 0x08,
+0xF004, 0x01,
+
+0xB000, 0x0D,
+0xB001, 0x0D,
+0xB002, 0x0D,
+
+0xB600, 0x34,
+0xB601, 0x34,
+0xB602, 0x34,
+
+0xB100, 0x0D,
+0xB101, 0x0D,
+0xB102, 0x0D,
+
+// AVEE: manual, -6V : -2.5xVCI)
+0xB700, 0x34,
+0xB701, 0x34,
+0xB702, 0x34,
+
+//Power Control for VCL
+0xB200, 0x00,
+0xB201, 0x00,
+0xB202, 0x00,
+
+0xB800, 0x24,
+0xB801, 0x24,
+0xB802, 0x24,
+
+// VGH: Clamp Enable, 2*AVDD-AVEE, //11V
+0xBF00, 0X01,
+
+0xB300, 0x0F,
+0xB301, 0x0F,
+0xB302, 0x0F,
+
+0xB900, 0x34,//34
+0xB901, 0x34,
+0xB902, 0x34,
+
+// VGL_REG(VGLO):-10V
+0xB500, 0x08,
+0xB501, 0x08,
+0xB502, 0x08,
+
+0xC200, 0x03,
+
+//VGL(LVGL):
+0xBA00, 0x24,
+0xBA01, 0x24,
+0xBA02, 0x24,
+
+
+// VGMP/VGSP: 4.8v
+0xBC00, 0X00,
+0xBC01, 0x78,
+0xBC02, 0X00,
+
+// VGMN/VGSN -4.8v
+0xBD00, 0x00,
+0xBD01, 0x78,
+0xBD02, 0x00,
+
+// VCOM=-0.1
+0xBE00, 0x00,
+0xBE01, 0x80, //2f
+//R+
+0xD100, 0x00,
+0xD101, 0x05,
+0xD102, 0x00,
+0xD103, 0x06,
+0xD104, 0x00,
+0xD105, 0x0E,
+0xD106, 0x00,
+0xD107, 0x19,
+0xD108, 0x00,
+0xD109, 0x29,
+0xD10A, 0x00,
+0xD10B, 0x55,
+0xD10C, 0x00,
+0xD10D, 0x80,
+0xD10E, 0x00,
+0xD10F, 0xC7,
+0xD110, 0x00,
+0xD111, 0xFC,
+0xD112, 0x01,
+0xD113, 0x48,
+0xD114, 0x01,
+0xD115, 0x7C,
+0xD116, 0x01,
+0xD117, 0xC5,
+0xD118, 0x01,
+0xD119, 0xFE,
+0xD11A, 0x02,
+0xD11B, 0x00,
+0xD11C, 0x02,
+0xD11D, 0x30,
+0xD11E, 0x02,
+0xD11F, 0x60,
+0xD120, 0x02,
+0xD121, 0x7A,
+0xD122, 0x02,
+0xD123, 0x9A,
+0xD124, 0x02,
+0xD125, 0xAC,
+0xD126, 0x02,
+0xD127, 0xC4,
+0xD128, 0x02,
+0xD129, 0xD3,
+0xD12A, 0x02,
+0xD12B, 0xE9,
+0xD12C, 0x02,
+0xD12D, 0xF8,
+0xD12E, 0x03,
+0xD12F, 0x0D,
+0xD130, 0x03,
+0xD131, 0x3B,
+0xD132, 0x05,
+0xD133, 0xB5,
+//G+
+0xD200, 0x00,
+0xD201, 0x05,
+0xD202, 0x00,
+0xD203, 0x06,
+0xD204, 0x00,
+0xD205, 0x0E,
+0xD206, 0x00,
+0xD207, 0x19,
+0xD208, 0x00,
+0xD209, 0x29,
+0xD20A, 0x00,
+0xD20B, 0x55,
+0xD20C, 0x00,
+0xD20D, 0x80,
+0xD20E, 0x00,
+0xD20F, 0xC7,
+0xD210, 0x00,
+0xD211, 0xFC,
+0xD212, 0x01,
+0xD213, 0x48,
+0xD214, 0x01,
+0xD215, 0x7C,
+0xD216, 0x01,
+0xD217, 0xC5,
+0xD218, 0x01,
+0xD219, 0xFE,
+0xD21A, 0x02,
+0xD21B, 0x00,
+0xD21C, 0x02,
+0xD21D, 0x30,
+0xD21E, 0x02,
+0xD21F, 0x60,
+0xD220, 0x02,
+0xD221, 0x7A,
+0xD222, 0x02,
+0xD223, 0x9A,
+0xD224, 0x02,
+0xD225, 0xAC,
+0xD226, 0x02,
+0xD227, 0xC4,
+0xD228, 0x02,
+0xD229, 0xD3,
+0xD22A, 0x02,
+0xD22B, 0xE9,
+0xD22C, 0x02,
+0xD22D, 0xF8,
+0xD22E, 0x03,
+0xD22F, 0x0D,
+0xD230, 0x03,
+0xD231, 0x3B,
+0xD232, 0x05,
+0xD233, 0xB5,
+//B+
+0xD300, 0x00,
+0xD301, 0x05,
+0xD302, 0x00,
+0xD303, 0x06,
+0xD304, 0x00,
+0xD305, 0x0E,
+0xD306, 0x00,
+0xD307, 0x19,
+0xD308, 0x00,
+0xD309, 0x29,
+0xD30A, 0x00,
+0xD30B, 0x55,
+0xD30C, 0x00,
+0xD30D, 0x80,
+0xD30E, 0x00,
+0xD30F, 0xC7,
+0xD310, 0x00,
+0xD311, 0xFC,
+0xD312, 0x01,
+0xD313, 0x48,
+0xD314, 0x01,
+0xD315, 0x7C,
+0xD316, 0x01,
+0xD317, 0xC5,
+0xD318, 0x01,
+0xD319, 0xFE,
+0xD31A, 0x02,
+0xD31B, 0x00,
+0xD31C, 0x02,
+0xD31D, 0x30,
+0xD31E, 0x02,
+0xD31F, 0x60,
+0xD320, 0x02,
+0xD321, 0x7A,
+0xD322, 0x02,
+0xD323, 0x9A,
+0xD324, 0x02,
+0xD325, 0xAC,
+0xD326, 0x02,
+0xD327, 0xC4,
+0xD328, 0x02,
+0xD329, 0xD3,
+0xD32A, 0x02,
+0xD32B, 0xE9,
+0xD32C, 0x02,
+0xD32D, 0xF8,
+0xD32E, 0x03,
+0xD32F, 0x0D,
+0xD330, 0x03,
+0xD331, 0x3B,
+0xD332, 0x05,
+0xD333, 0xB5,
+//R-
+0xD400, 0x00,
+0xD401, 0x05,
+0xD402, 0x00,
+0xD403, 0x06,
+0xD404, 0x00,
+0xD405, 0x0E,
+0xD406, 0x00,
+0xD407, 0x19,
+0xD408, 0x00,
+0xD409, 0x29,
+0xD40A, 0x00,
+0xD40B, 0x55,
+0xD40C, 0x00,
+0xD40D, 0x80,
+0xD40E, 0x00,
+0xD40F, 0xC7,
+0xD410, 0x00,
+0xD411, 0xFC,
+0xD412, 0x01,
+0xD413, 0x48,
+0xD414, 0x01,
+0xD415, 0x7C,
+0xD416, 0x01,
+0xD417, 0xC5,
+0xD418, 0x01,
+0xD419, 0xFE,
+0xD41A, 0x02,
+0xD41B, 0x00,
+0xD41C, 0x02,
+0xD41D, 0x30,
+0xD41E, 0x02,
+0xD41F, 0x60,
+0xD420, 0x02,
+0xD421, 0x7A,
+0xD422, 0x02,
+0xD423, 0x9A,
+0xD424, 0x02,
+0xD425, 0xAC,
+0xD426, 0x02,
+0xD427, 0xC4,
+0xD428, 0x02,
+0xD429, 0xD3,
+0xD42A, 0x02,
+0xD42B, 0xE9,
+0xD42C, 0x02,
+0xD42D, 0xF8,
+0xD42E, 0x03,
+0xD42F, 0x0D,
+0xD430, 0x03,
+0xD431, 0x3B,
+0xD432, 0x05,
+0xD433, 0xB5,
+//G-
+0xD500, 0x00,
+0xD501, 0x05,
+0xD502, 0x00,
+0xD503, 0x06,
+0xD504, 0x00,
+0xD505, 0x0E,
+0xD506, 0x00,
+0xD507, 0x19,
+0xD508, 0x00,
+0xD509, 0x29,
+0xD50A, 0x00,
+0xD50B, 0x55,
+0xD50C, 0x00,
+0xD50D, 0x80,
+0xD50E, 0x00,
+0xD50F, 0xC7,
+0xD510, 0x00,
+0xD511, 0xFC,
+0xD512, 0x01,
+0xD513, 0x48,
+0xD514, 0x01,
+0xD515, 0x7C,
+0xD516, 0x01,
+0xD517, 0xC5,
+0xD518, 0x01,
+0xD519, 0xFE,
+0xD51A, 0x02,
+0xD51B, 0x00,
+0xD51C, 0x02,
+0xD51D, 0x30,
+0xD51E, 0x02,
+0xD51F, 0x60,
+0xD520, 0x02,
+0xD521, 0x7A,
+0xD522, 0x02,
+0xD523, 0x9A,
+0xD524, 0x02,
+0xD525, 0xAC,
+0xD526, 0x02,
+0xD527, 0xC4,
+0xD528, 0x02,
+0xD529, 0xD3,
+0xD52A, 0x02,
+0xD52B, 0xE9,
+0xD52C, 0x02,
+0xD52D, 0xF8,
+0xD52E, 0x03,
+0xD52F, 0x0D,
+0xD530, 0x03,
+0xD531, 0x3B,
+0xD532, 0x05,
+0xD533, 0xB5,
+//B-
+0xD600, 0x00,
+0xD601, 0x05,
+0xD602, 0x00,
+0xD603, 0x06,
+0xD604, 0x00,
+0xD605, 0x0E,
+0xD606, 0x00,
+0xD607, 0x19,
+0xD608, 0x00,
+0xD609, 0x29,
+0xD60A, 0x00,
+0xD60B, 0x55,
+0xD60C, 0x00,
+0xD60D, 0x80,
+0xD60E, 0x00,
+0xD60F, 0xC7,
+0xD610, 0x00,
+0xD611, 0xFC,
+0xD612, 0x01,
+0xD613, 0x48,
+0xD614, 0x01,
+0xD615, 0x7C,
+0xD616, 0x01,
+0xD617, 0xC5,
+0xD618, 0x01,
+0xD619, 0xFE,
+0xD61A, 0x02,
+0xD61B, 0x00,
+0xD61C, 0x02,
+0xD61D, 0x30,
+0xD61E, 0x02,
+0xD61F, 0x60,
+0xD620, 0x02,
+0xD621, 0x7A,
+0xD622, 0x02,
+0xD623, 0x9A,
+0xD624, 0x02,
+0xD625, 0xAC,
+0xD626, 0x02,
+0xD627, 0xC4,
+0xD628, 0x02,
+0xD629, 0xD3,
+0xD62A, 0x02,
+0xD62B, 0xE9,
+0xD62C, 0x02,
+0xD62D, 0xF8,
+0xD62E, 0x03,
+0xD62F, 0x0D,
+0xD630, 0x03,
+0xD631, 0x3B,
+0xD632, 0x05,
+0xD633, 0xB5,
+
+//Enable Page0
+0xF000, 0x55,
+0xF001, 0xAA,
+0xF002, 0x52,
+0xF003, 0x08,
+0xF004, 0x00,
+
+
+0xB100, 0xcc,
+0xB101, 0x00,
+
+0xB600, 0x05,
+
+//// Gate EQ:
+0xB700, 0x70,
+0xB701, 0x70,
+
+//// Source EQ:
+0xB800, 0x01,
+0xB801, 0x03,
+0xB802, 0x03,
+0xB803, 0x03,
+
+// #Inversion mode (2-dot)
+0xBC00, 0x02,
+0xBC01, 0x00,
+0xBC02, 0x00,
+
+// Timing control 4H w/ 4-delay
+0xC900, 0xD0,
+0xC901, 0x02,
+0xC902, 0x50,
+0xC903, 0x50,
+0xC904, 0x50,
+// Display Timing:
+
+0x3600, 0x00,
+0x3500, 0x00,
+
+0x3a00, 0x66,
+
+#endif
diff --git a/drivers/video/wmt/devices/vt1625.c b/drivers/video/wmt/devices/vt1625.c
new file mode 100644
index 00000000..892f287c
--- /dev/null
+++ b/drivers/video/wmt/devices/vt1625.c
@@ -0,0 +1,454 @@
+/*++
+ * linux/drivers/video/wmt/vt1625.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define VT1625_C
+#define DEBUG
+/* #define DEBUG_DETAIL */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../vout.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define VT1625_XXXX xxxx *//*Example*/
+// #define CONFIG_VT1625_INTERRUPT
+#define CONFIG_VT1625_POWER
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define VT1625_XXXX 1 *//*Example*/
+#define VT1625_ADDR 0x40
+
+enum {
+ VT1625_INPUT_SELECTION = 0x00,
+ VT1625_SYNC_SELECTION_0 = 0x01,
+ VT1625_SYNC_SELECTION_1 = 0x02,
+ VT1625_FILTER_SELECTION = 0x03,
+ VT1625_OUTPUT_MODE = 0x04,
+ VT1625_CLOCK_CONTROL = 0x05,
+
+ /* start position & overflow */
+ VT1625_OVERFLOW = 0x06,
+ VT1625_START_ACTIVE_VIDEO = 0x07,
+ VT1625_START_HORIZONTAL_POSITION = 0x08,
+ VT1625_START_VERTICAL_POSITION = 0x09,
+
+ /* amplitude factor */
+ VT1625_CR_AMPLITUDE_FACTOR = 0x0A,
+ VT1625_BLACK_LEVEL = 0x0B,
+ VT1625_Y_AMPLITUDE_FACTOR = 0x0C,
+ VT1625_CB_AMPLITUDE_FACTOR = 0x0D,
+
+ VT1625_POWER_MANAGEMENT = 0x0E,
+ VT1625_STATUS = 0x0F,
+
+ /* Hue */
+ VT1625_HUE_ADJUSTMENT = 0x10,
+ VT1625_OVERFLOW_MISC = 0x11,
+
+ /* PLL */
+ VT1625_PLL_P2 = 0x12,
+ VT1625_PLL_D = 0x13,
+ VT1625_PLL_N = 0x14,
+ VT1625_PLL_OVERFLOW = 0x15,
+
+ /* Sub Carrier */
+ VT1625_SUBCARRIER_VALUE_0 = 0x16,
+ VT1625_SUBCARRIER_VALUE_1 = 0x17,
+ VT1625_SUBCARRIER_VALUE_2 = 0x18,
+ VT1625_SUBCARRIER_VALUE_3 = 0x19,
+
+ VT1625_VERSION_ID = 0x1B,
+ VT1625_DAC_OVERFLOW = 0x1C,
+
+ /* test */
+ VT1625_TEST_0 = 0x1D,
+ VT1625_TEST_1 = 0x1E,
+
+ VT1625_FILTER_SWITCH = 0x1F,
+ VT1625_TV_SYNC_STEP = 0x20,
+ VT1625_TV_BURST_ENVELOPE_STEP = 0x21,
+ VT1625_TV_SUB_CARRIER_PHASE_ADJUST = 0x22,
+ VT1625_TV_BLANK_LEVEL = 0x23,
+ VT1625_TV_SIGNAL_OVERFLOW = 0x24,
+
+ /* DAC & GPO */
+ VT1625_DAC_SELECTION_0 = 0x4A,
+ VT1625_DAC_SELECTION_1 = 0x4B,
+ VT1625_GPO = 0x4C,
+
+ VT1625_COLBAR_LUMA_DELAY = 0x4D,
+ VT1625_UV_DELAY = 0x4E,
+ VT1625_BURST_MAX_AMPLITUDE = 0x4F,
+
+ /* Graphic timing */
+ VT1625_GRAPHIC_H_TOTAL = 0x50,
+ VT1625_GRAPHIC_H_ACTIVE = 0x51,
+ VT1625_GRAPHIC_H_OVERFLOW = 0x52,
+ VT1625_GRAPHIC_V_TOTAL = 0x53,
+ VT1625_GRAPHIC_V_OVERFLOW = 0x54,
+
+ /* TV timing */
+ VT1625_TV_H_TOTAL = 0x55,
+ VT1625_TV_H_ACTIVE = 0x56,
+ VT1625_TV_H_SYNC_WIDTH = 0x57,
+ VT1625_TV_H_OVERFLOW = 0x58,
+ VT1625_TV_BURST_START = 0x59,
+ VT1625_TV_BURST_END = 0x5A,
+ VT1625_TV_VIDEO_START = 0x5B,
+ VT1625_TV_VIDEO_END = 0x5C,
+ VT1625_TV_VIDEO_OVERFLOW = 0x5D,
+
+ /* scale factor */
+ VT1625_V_SCALE_FACTOR = 0x5E,
+ VT1625_H_SCALE_FACTOR = 0x5F,
+ VT1625_SCALE_OVERFLOW = 0x60,
+ VT1625_H_BLUR_SCALE_OVERFLOW = 0x61,
+ VT1625_ADAPTIVE_DEFLICKER_THR = 0x62,
+ VT1625_SCALE_H_TOTAL = 0x63,
+ VT1625_SCALE_H_TOTAL_OVERFLOW = 0x64,
+
+ /* Amplitude factor */
+ VT1625_PY_AMP_FACTOR = 0x65,
+ VT1625_PB_AMP_FACTOR = 0x66,
+ VT1625_PR_AMP_FACTOR = 0x67,
+
+ VT1625_POST_ADJUST = 0x68,
+ VT1625_AUTO_CORRECT_SENSE = 0x69,
+
+ /* WSS 0x6A - 0x73 */
+ VT1625_INT_WSS_2 = 0x71,
+
+ /* Close Caption 0x74 - 0x7A */
+
+ /* Signature Value 0x7B - 0x82 */
+} vt1625_reg_t;
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx vt1625_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in vt1625.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int vt1625_xxx; *//*Example*/
+
+const char vt1625_ntsc_param[] = {
+ 0x03, 0x00, 0x10, 0x10, 0x00, 0x00, 0x00, 0x88, /* 00 - 07 */
+ 0x09, 0x05, 0x6E, 0x15, 0x52, 0x4E, 0x37, 0xB7, /* 08 - 0f */
+ 0x08, 0x80, 0x04, 0x08, 0x08, 0x90, 0xD6, 0x7B, /* 10 - 17 */
+ 0xF0, 0x21, 0x02, 0x50, 0x43, 0x80, 0x00, 0x10, /* 18 - 1f */
+ 0x16, 0x08, 0xDC, 0x7D, 0x02, 0x56, 0x33, 0x8F, /* 20 - 27 */
+ 0x58, 0x00, 0x00, 0xA6, 0x29, 0xD4, 0x81, 0x00, /* 28 - 2f */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 30 - 37 */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 38 - 3f */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 40 - 47 */
+ 0x00, 0x00, 0xC5, 0x0F, 0x00, 0x01, 0x10, 0x44, /* 48 - 4f */
+ 0x59, 0xCF, 0x23, 0x0C, 0x02, 0x59, 0xCF, 0x7F, /* 50 - 57 */
+ 0x23, 0x94, 0xD6, 0xFE, 0x78, 0x06, 0x00, 0x00, /* 58 - 5f */
+ 0x80, 0x28, 0xFF, 0x59, 0x03, 0x55, 0x56, 0x56, /* 60 - 67 */
+ 0x00, 0x90, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, /* 68 - 6f */
+ 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 70 - 77 */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 78 - 7f */
+ 0x00, 0x00, 0x00
+};
+
+const char vt1625_pal_param[] = {
+ 0x03, 0x00, 0x00, 0x10, 0x03, 0x00, 0x00, 0x8f, /* 00 - 07 */
+ 0x00, 0x01, 0x7A, 0x00, 0x55, 0x58, 0x37, 0xB7, /* 08 - 0f */
+ 0x08, 0x80, 0x04, 0x08, 0x08, 0x90, 0xCB, 0x8A, /* 10 - 17 */
+ 0x09, 0x2A, 0x06, 0x50, 0x41, 0x80, 0x00, 0x00, /* 18 - 1f */
+ 0x17, 0x0C, 0x4E, 0x85, 0x02, 0x5F, 0x34, 0x8C, /* 20 - 27 */
+ 0x4F, 0x5E, 0x15, 0xA2, 0x22, 0x80, 0xD3, 0x10, /* 28 - 2f */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 30 - 37 */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 38 - 3f */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 40 - 47 */
+ 0x00, 0x00, 0xC5, 0x0F, 0x00, 0x01, 0x10, 0x4C, /* 48 - 4f */
+ 0x5f, 0xCF, 0x23, 0x70, 0x02, 0x5F, 0xD0, 0x7F, /* 50 - 57 */
+ 0x23, 0x92, 0xCE, 0xDF, 0x86, 0x06, 0x00, 0x00, /* 58 - 5f */
+ 0x80, 0x20, 0xFF, 0x5F, 0x03, 0x5f, 0x00, 0x00, /* 60 - 67 */
+ 0x00, 0x90, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, /* 68 - 6f */
+ 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 70 - 77 */
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 78 - 7f */
+ 0x00, 0x00, 0x00
+};
+
+int vt1625_vga_mode;
+int vt1625_tv_mode;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void vt1625_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+
+/*the define and struct i2c_msg were declared int linux/i2c.h*/
+void vt1625_reg_dump(void)
+{
+ int i;
+ char buf[256];
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR, VT1625_INPUT_SELECTION,
+ buf, 128);
+ for (i = 0; i < 128; i += 8) {
+ MSG("%d : 0x%02x 0x%02x 0x%02x 0x%02x",
+ i, buf[i], buf[i + 1], buf[i + 2], buf[i + 3]);
+ MSG(" 0x%02x 0x%02x 0x%02x 0x%02x\n",
+ buf[i + 4], buf[i + 5], buf[i + 6], buf[i + 7]);
+ }
+}
+
+void vt1625_set_tv_mode(int ntsc)
+{
+ char *p;
+ char buf[10];
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_INPUT_SELECTION, buf, 5);
+ DBG_MSG("ntsc %d, 0x%x, 0x%x\n", ntsc, buf[0], buf[4]);
+ vt1625_tv_mode = (ntsc) ? 1 : 2;
+ if (buf[0] && (vt1625_vga_mode != -1)) {
+ if (ntsc && !(buf[4] & BIT0))
+ return;
+ if (!ntsc && (buf[4] & BIT0))
+ return;
+ }
+
+ DBG_MSG("tv %s,vga %d\n", (ntsc) ? "NTSC" : "PAL", vt1625_vga_mode);
+
+ p = (char *)((ntsc) ? vt1625_ntsc_param : vt1625_pal_param);
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR, VT1625_INPUT_SELECTION,
+ &p[VT1625_INPUT_SELECTION], 0x71);
+ if (vt1625_vga_mode == -1) {
+ mdelay(10);
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_STATUS, buf, 1);
+ vt1625_vga_mode = (buf[0] & 0x7) ? 0 : 1;
+ DBG_MSG("get vga mode %d, 0x%x\n", vt1625_vga_mode, buf[0]);
+ }
+
+ if (vt1625_vga_mode) {
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_SYNC_SELECTION_1, buf, 1);
+ buf[0] |= 0xA0;
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_SYNC_SELECTION_1, buf, 1);
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_DAC_OVERFLOW, buf, 1);
+ buf[0] |= 0x20;
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_DAC_OVERFLOW, buf, 1);
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_TEST_1, buf, 1);
+ buf[0] |= 0x40;
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_TEST_1, buf, 1);
+ } else {
+#ifdef CONFIG_VT1625_INTERRUPT
+ /* interrupt (VGA no work) */
+ if (!g_vpp.virtual_display) {
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_INT_WSS_2, buf, 1);
+ buf[0] |= 0xA0; /* enable sense interrupt */
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_INT_WSS_2, buf, 1);
+ }
+#endif
+ }
+#ifdef CONFIG_VT1625_POWER
+ buf[0] = (vt1625_vga_mode) ? 0x38 : 0x37;
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_POWER_MANAGEMENT, buf, 1);
+#endif
+// vt1625_reg_dump();
+}
+
+int vt1625_check_plugin(int hotplug)
+{
+ char buf[1];
+ int plugin;
+
+ if (g_vpp.virtual_display)
+ return 1;
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR, VT1625_STATUS, buf, 1);
+ plugin = ((buf[0] & 0x3F) == 0x3F) ? 0 : 1;
+ DBG_MSG("[VT1625] DAC A %d, B %d, C %d, D %d, E %d, F %d\n",
+ (buf[0] & 0x20) ? 1 : 0, (buf[0] & 0x10) ? 1 : 0,
+ (buf[0] & 0x08) ? 1 : 0, (buf[0] & 0x04) ? 1 : 0,
+ (buf[0] & 0x02) ? 1 : 0, (buf[0] & 0x01) ? 1 : 0);
+ return plugin;
+}
+
+int vt1625_init(struct vout_s *vo)
+{
+ char buf[5];
+
+ DBG_MSG("\n");
+ // vt1625_vga_mode = -1;
+ vt1625_vga_mode = 0;
+ if (vt1625_tv_mode) { /* resume reinit */
+ MSG("[VT1625] DVI reinit\n");
+ vt1625_set_tv_mode((vt1625_tv_mode == 1) ? 1 : 0);
+ return 0;
+ }
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR, VT1625_VERSION_ID, buf, 1);
+ if (buf[0] != 0x50) /* check version id */
+ return -1;
+ vo->option[0] = VDO_COL_FMT_ARGB;
+ vo->option[1] = VPP_DATAWIDHT_12;
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_INPUT_SELECTION, buf, 5);
+ if (buf[0]) {
+ vt1625_tv_mode = (buf[4]) ? 2 : 1;
+ }
+ MSG("[VT1625] DVI ext device\n");
+ return 0;
+}
+
+int vt1625_set_mode(unsigned int *option)
+{
+ DBG_MSG("\n");
+#ifdef CONFIG_VT1625_INTERRUPT
+ vout_set_int_type(1);
+#endif
+ return 0;
+}
+
+void vt1625_set_power_down(int enable)
+{
+ vout_t *vo;
+ char buf[1];
+ char cur[1];
+
+ /*
+ bit 0-2 : DAC D/E/F - VGA
+ bit 3 : DAC C - CVBS
+ bit 3-5 : DAC A/B/C - YPbPr
+ bit 6 : PLL
+ bit 7 : IO pad
+ */
+ vo = vout_get_entry(VPP_VOUT_NUM_DVI);
+ if (vo->status & (VPP_VOUT_STS_BLANK + VPP_VOUT_STS_POWERDN)) {
+ enable = 1;
+ }
+
+ if (vt1625_tv_mode == 0) /* power down for not support resolution */
+ buf[0] = 0xFF;
+ else if (vt1625_vga_mode == -1)
+ buf[0] = (enable) ? 0xFF : 0x37;
+ else
+ buf[0] = (enable) ? 0xFF : ((vt1625_vga_mode) ? 0x38 : 0x37);
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_POWER_MANAGEMENT, cur, 1);
+
+ if (cur[0] == buf[0])
+ return;
+
+ DBG_MSG("enable %d,cur 0x%x,new 0x%x\n", enable, cur[0], buf[0]);
+#if 1
+ if (enable == 0) {
+ cur[0] &= ~0x40; /* turn on PLL */
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_POWER_MANAGEMENT, cur, 1);
+ mdelay(3);
+
+ cur[0] &= ~0x80; /* turn on IO pad */
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_POWER_MANAGEMENT, cur, 1);
+ mdelay(3);
+ }
+#endif
+#ifdef CONFIG_VT1625_POWER
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR,
+ VT1625_POWER_MANAGEMENT, buf, 1);
+#endif
+}
+
+int vt1625_config(vout_info_t *info)
+{
+ int ntsc = -1;
+
+ DBG_MSG("%d,%d\n", info->resx, info->resy);
+ if (info->resx == 720) {
+ switch (info->resy) {
+ case 480:
+ ntsc = 1;
+ break;
+ case 576:
+ ntsc = 0;
+ break;
+ default:
+ break;
+ }
+ }
+
+ if (ntsc != -1)
+ vt1625_set_tv_mode(ntsc);
+ else
+ vt1625_tv_mode = 0;
+ DBG_MSG("end\n");
+ return 0;
+}
+
+int vt1625_get_edid(char *buf)
+{
+ return 0;
+}
+
+int vt1625_interrupt(void)
+{
+ char buf[1];
+
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x4c0, 0x1 << VPP_VOINT_NO,
+ VPP_VOINT_NO, 0x0); /* GPIO pull-up */
+ /* interrupt */
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1625_ADDR, VT1625_INT_WSS_2, buf, 1);
+ DBG_MSG("0x%x\n", buf[0]);
+ buf[0] &= ~0x40; /* clear interrupt */
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1625_ADDR, VT1625_INT_WSS_2, buf, 1);
+ return vt1625_check_plugin(1);
+}
+/*----------------------- vout device plugin ---------------------------------*/
+vout_dev_t vt1625_vout_dev_ops = {
+ .name = "VT1625",
+ .mode = VOUT_INF_DVI,
+
+ .init = vt1625_init,
+ .set_power_down = vt1625_set_power_down,
+ .set_mode = vt1625_set_mode,
+ .config = vt1625_config,
+ .check_plugin = vt1625_check_plugin,
+ .get_edid = vt1625_get_edid,
+#ifdef CONFIG_VT1625_INTERRUPT
+ .interrupt = vt1625_interrupt,
+#endif
+};
+
+int vt1625_module_init(void)
+{
+ vout_device_register(&vt1625_vout_dev_ops);
+ return 0;
+}
+module_init(vt1625_module_init);
+/*--------------------End of Function Body -----------------------------------*/
+#undef VT1625_C
diff --git a/drivers/video/wmt/devices/vt1632.c b/drivers/video/wmt/devices/vt1632.c
new file mode 100644
index 00000000..3ece9d9e
--- /dev/null
+++ b/drivers/video/wmt/devices/vt1632.c
@@ -0,0 +1,157 @@
+/*++
+ * linux/drivers/video/wmt/vt1632.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define VT1632_C
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "../vout.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define VT1632_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define VT1632_XXXX 1 *//*Example*/
+#define VT1632_ADDR 0x10
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx vt1632_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in vt1632.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int vt1632_xxx; *//*Example*/
+static int vt1632_not_ready;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void vt1632_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+
+/*the define and struct i2c_msg were declared int linux/i2c.h*/
+int vt1632_check_plugin(int hotplug)
+{
+ char buf[1];
+ int plugin;
+
+ if (vt1632_not_ready)
+ return 1;
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1632_ADDR, 0x9, buf, 1);
+ plugin = (buf[0] & 0x4) ? 1 : 0;
+ DPRINT("[VT1632] DVI plug%s\n", (plugin) ? "in" : "out");
+ vout_set_int_type(4);
+ return plugin;
+}
+
+int vt1632_init(struct vout_s *vo)
+{
+ char buf[16];
+
+ vt1632_not_ready = 1;
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1632_ADDR, 0x0, buf, 2);
+ if ((buf[0] != 0x06) || (buf[1] != 0x11)) /* check vender id */
+ return -1;
+ vt1632_not_ready = 0;
+
+ buf[0x0] = 0x37;
+ buf[0x1] = 0x20;
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1632_ADDR, 0x8, buf, 2);
+ DPRINT("[VT1632] DVI ext device\n");
+ return 0;
+}
+
+int vt1632_set_mode(unsigned int *option)
+{
+ char buf[1];
+ vpp_datawidht_t dwidth;
+
+ if (vt1632_not_ready)
+ return -1;
+
+ dwidth = option[1] & BIT0;
+ DBGMSG("vt1632_set_mode(%d)\n", (dwidth) ? 24 : 12);
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1632_ADDR, 0x8, buf, 1);
+ if (dwidth == VPP_DATAWIDHT_12) {
+ buf[0] &= ~BIT2;
+ buf[0] |= BIT3;
+ } else {
+ buf[0] |= BIT2;
+ buf[0] &= ~BIT3;
+ }
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1632_ADDR, 0x8, buf, 1);
+ return 0;
+}
+
+void vt1632_set_power_down(int enable)
+{
+ char buf[1];
+
+ if (vt1632_not_ready)
+ return;
+
+ DBGMSG("vt1632_set_power_down(%d)\n", enable);
+
+ vpp_i2c_read(VPP_DVI_I2C_ID, VT1632_ADDR, 0x8, buf, 1);
+ if (enable)
+ buf[0] &= ~BIT0;
+ else
+ buf[0] |= BIT0;
+ vpp_i2c_write(VPP_DVI_I2C_ID, VT1632_ADDR, 0x8, buf, 1);
+}
+
+int vt1632_config(vout_info_t *info)
+{
+ return 0;
+}
+
+int vt1632_get_edid(char *buf)
+{
+ return 0;
+}
+
+int vt1632_interrupt(void)
+{
+ return vt1632_check_plugin(1);
+}
+/*----------------------- vout device plugin ---------------------------------*/
+vout_dev_t vt1632_vout_dev_ops = {
+ .name = "VT1632",
+ .mode = VOUT_INF_DVI,
+
+ .init = vt1632_init,
+ .set_power_down = vt1632_set_power_down,
+ .set_mode = vt1632_set_mode,
+ .config = vt1632_config,
+ .check_plugin = vt1632_check_plugin,
+ .get_edid = vt1632_get_edid,
+ .interrupt = vt1632_interrupt,
+};
+
+int vt1632_module_init(void)
+{
+ vout_device_register(&vt1632_vout_dev_ops);
+ return 0;
+} /* End of vt1632_module_init */
+module_init(vt1632_module_init);
+/*--------------------End of Function Body -----------------------------------*/
+#undef VT1632_C
diff --git a/drivers/video/wmt/edid.h b/drivers/video/wmt/edid.h
new file mode 100644
index 00000000..ec85a81b
--- /dev/null
+++ b/drivers/video/wmt/edid.h
@@ -0,0 +1,155 @@
+/*++
+ * linux/drivers/video/wmt/edid.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef EDID_H
+#define EDID_H
+
+#define EDID_BLOCK_MAX 4
+
+/*------------------------------------------------------------------------------
+ Following definitions, please refer spec of EDID. You may refer it on
+ http://en.wikipedia.org/wiki/EDID#EDID_1.3_data_format
+------------------------------------------------------------------------------*/
+
+#define EDID_LENGTH 0x80
+
+/*------------------------------------------------------------------------------
+ Offset 00-19: HEADER INFORMATION
+------------------------------------------------------------------------------*/
+/* 00¡V07: Header information "00h FFh FFh FFh FFh FFh FFh 00h" */
+#define EDID_HEADER 0x00
+#define EDID_HEADER_END 0x07
+
+/* 08¡V09: Manufacturer ID. These IDs are assigned by Microsoft.
+ "00001=A¡¨; ¡§00010=B¡¨; ... ¡§11010=Z¡¨. Bit 7 (at address 08h) is 0,
+ the first character (letter) is located at bits 6 ¡÷ 2
+ (at address 08h), the second character (letter) is located at
+ bits 1 & 0 (at address 08h) and bits 7 ¡÷ 5 (at address 09h), and the
+ third character (letter) is located at bits 4 ¡÷ 0 (at address 09h).
+*/
+#define ID_MANUFACTURER_NAME 0x08
+#define ID_MANUFACTURER_NAME_END 0x09
+
+/* 10¡V11: Product ID Code (stored as LSB first). Assigned by manufacturer */
+#define ID_MODEL 0x0a
+
+/* 12¡V15: 32-bit Serial Number. No requirement for the format. Usually
+ stored as LSB first. In order to maintain compatibility with previous
+ requirements the field should set at least one byte of the field to be
+ non-zero if an ASCII serial number descriptor is
+ provided in the detailed timing section.
+*/
+#define ID_SERIAL_NUMBER 0x0c
+
+/* 16: Week of Manufacture. This varies by manufacturer. One way is to count
+ January 1-7 as week 1, January 8-15 as week 2 and so on. Some count
+ based on the week number (Sunday-Saturday). Valid range is 1-54.
+ 17: Year of Manufacture. Add 1990 to the value for actual year. */
+#define MANUFACTURE_WEEK 0x10
+#define MANUFACTURE_YEAR 0x11
+
+/* 18: EDID Version Number. "01h"
+ 19: EDID Revision Number "03h" */
+#define EDID_STRUCT_VERSION 0x12
+#define EDID_STRUCT_REVISION 0x13
+
+/*------------------------------------------------------------------------------
+ Offset 20-24: BASIC DISPLAY PARAMETERS
+------------------------------------------------------------------------------*/
+#define DPMS_FLAGS 0x18
+#define ESTABLISHED_TIMING_I 0x23
+#define ESTABLISHED_TIMING_II 0x24
+#define MANUFACTURERS_TIMINGS 0x25
+
+#define STANDARD_TIMING_IDENTIFICATION_START 0x26
+#define STANDARD_TIMING_IDENTIFICATION_SIZE 16
+
+#define DETAILED_TIMING_DESCRIPTIONS_START 0x36
+#define DETAILED_TIMING_DESCRIPTION_SIZE 18
+#define NO_DETAILED_TIMING_DESCRIPTIONS 4
+
+#define DETAILED_TIMING_DESCRIPTION_1 0x36
+#define DETAILED_TIMING_DESCRIPTION_2 0x48
+#define DETAILED_TIMING_DESCRIPTION_3 0x5a
+#define DETAILED_TIMING_DESCRIPTION_4 0x6c
+
+#define EDID_TMR_INTERLACE BIT(31)
+#define EDID_TMR_FREQ 0xFF
+typedef struct {
+ unsigned int resx;
+ unsigned int resy;
+ int freq; /* EDID_TMR_XXX */
+} edid_timing_t;
+
+/* EDID option flag */
+#define EDID_OPT_VALID 0x01
+#define EDID_OPT_YUV422 0x10
+#define EDID_OPT_YUV444 0x20
+#define EDID_OPT_AUDIO 0x40
+#define EDID_OPT_UNDERSCAN 0x80
+#define EDID_OPT_HDMI 0x100
+#define EDID_OPT_3D BIT(9)
+#define EDID_OPT_16_9 BIT(10)
+
+typedef struct {
+ unsigned int establish_timing;
+ edid_timing_t standard_timing[8];
+ struct fb_videomode detail_timing[4];
+ struct fb_videomode cea_timing[6];
+ char cea_vic[8];
+ unsigned int pixel_clock_limit;
+ unsigned int option;
+ unsigned short hdmi_phy_addr;
+} edid_info_t;
+
+#define VENDOR_NAME_LEN 4
+#define MONITOR_NAME_LEN 20
+#define AUD_SAD_NUM 32
+typedef struct {
+ char flag; /* 0: sad available, 1: sad invalid */
+ char sad_byte[3];
+} sad_t;
+
+typedef struct {
+ char vendor_name[VENDOR_NAME_LEN];
+ char monitor_name[MONITOR_NAME_LEN];
+} tv_name_t;
+
+typedef struct {
+ tv_name_t tv_name;
+ sad_t sad[AUD_SAD_NUM];
+} edid_parsed_t;
+
+extern edid_info_t edid_info;
+extern int edid_msg_enable;
+extern int edid_disable;
+extern edid_parsed_t edid_parsed;
+
+extern int edid_parse(char *edid, edid_info_t *info);
+extern int edid_find_support(edid_info_t *info, unsigned int resx,
+ unsigned int resy, int freq, struct fb_videomode **vmode);
+extern void edid_dump(char *edid);
+extern int edid_checksum(char *edid, int len);
+extern unsigned int edid_get_hdmi_phy_addr(void);
+
+#endif
diff --git a/drivers/video/wmt/ge/Makefile b/drivers/video/wmt/ge/Makefile
new file mode 100644
index 00000000..819c710a
--- /dev/null
+++ b/drivers/video/wmt/ge/Makefile
@@ -0,0 +1,5 @@
+#
+# Makefile for WonderMedia GE driver
+#
+obj-$(CONFIG_FB_WMT_GE) += ge.o
+ge-objs := ge_main.o ge_accel.o
diff --git a/drivers/video/wmt/ge/ge_accel.c b/drivers/video/wmt/ge/ge_accel.c
new file mode 100644
index 00000000..ef8ce133
--- /dev/null
+++ b/drivers/video/wmt/ge/ge_accel.c
@@ -0,0 +1,551 @@
+/*
+ * Copyright (c) 2008-2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+ */
+
+#include <asm/cacheflush.h>
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/sched.h>
+#include <linux/semaphore.h>
+#include <linux/interrupt.h>
+#include <linux/workqueue.h>
+#include <linux/spinlock.h>
+#include <linux/delay.h>
+#include <linux/moduleparam.h>
+#include <mach/hardware.h>
+#include "ge_accel.h"
+
+unsigned int fb_egl_swap; /* useless */
+
+DECLARE_WAIT_QUEUE_HEAD(ge_wq);
+
+int flipcnt;
+int flipreq;
+int vbl;
+int vsync = 1;
+int sync2 = 1;
+int sync3;
+int debug;
+
+module_param(flipcnt, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(flipcnt, "Flip count");
+
+module_param(flipreq, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(flipreq, "Flip request count");
+
+module_param(vbl, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(vbl, "Wait vsync for each frame (0)");
+
+module_param(vsync, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(vsync, "Can use vsync (1)");
+
+module_param(sync2, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(sync2, "Only wait vsync if truly necessary");
+
+module_param(sync3, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(sync3, "Only wait vsync if truly necessary");
+
+module_param(debug, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(debug, "Show debug information");
+
+/**************************
+ * Export functions *
+ **************************/
+
+#define M(x) ((x)<<20)
+
+unsigned int phy_mem_end(void)
+{
+ unsigned int memsize = (num_physpages << PAGE_SHIFT);
+
+ if (memsize > M(3072)) { /* 4096M */
+ memsize = M(4096);
+ } else if (memsize > M(2048)) { /* 3072M */
+ memsize = M(3072);
+ } else if (memsize > M(1024)) { /* 2048M */
+ memsize = M(2048);
+ } else if (memsize > M(512)) { /* 1024M */
+ memsize = M(1024);
+ } else if (memsize > M(256)) { /* 512M */
+ memsize = M(512);
+ } else if (memsize > M(128)) { /* 256M */
+ memsize = M(256);
+ } else if (memsize > M(64)) { /* 128M */
+ memsize = M(128);
+ } else if (memsize > M(32)) { /* 64M */
+ memsize = M(64);
+ } else if (memsize > M(16)) { /* 32M */
+ memsize = M(32);
+ } else {
+ memsize = M(0);
+ }
+ printk(KERN_DEBUG "Detected RAM size %d MB\n", memsize>>20);
+
+ return memsize;
+}
+
+unsigned int phy_mem_end_sub(u32 size)
+{
+ return phy_mem_end() - M(size);
+}
+EXPORT_SYMBOL(phy_mem_end_sub);
+
+/* ge_vo_functions depends vpu to work */
+
+void ge_vo_get_default_var(struct fb_var_screeninfo *var)
+{
+#ifdef HAVE_VPP
+ vpp_get_info(0, var);
+#endif
+}
+
+void ge_vo_wait_vsync(void)
+{
+#ifdef HAVE_VPP
+ if (vsync)
+ vpp_wait_vsync(0, 1);
+#endif
+}
+
+static int ge_vo_pan_display(struct fb_var_screeninfo *var,
+ struct fb_info *info)
+{
+#ifdef HAVE_VPP
+ vpp_set_mutex(0, 1);
+ vpp_pan_display(var, info);
+ vpp_set_mutex(0, 0);
+#endif
+ flipcnt++;
+
+ return 0;
+}
+
+#if 0
+static void fbiomemcpy(struct fb_info *info,
+ unsigned long dst, unsigned long src, size_t len)
+{
+ void *psrc = info->screen_base + (src - info->fix.smem_start);
+ void *pdst = info->screen_base + (dst - info->fix.smem_start);
+ void *ptmp = info->screen_base + info->fix.smem_len - len;
+ unsigned long tmp = info->fix.smem_start + info->fix.smem_len - len;
+
+ if (src < info->fix.smem_start || src > tmp)
+ psrc = ioremap(src, len);
+ if (dst < info->fix.smem_start || dst > tmp)
+ pdst = ioremap(dst, len);
+ if (psrc && pdst && ptmp) {
+ memcpy(ptmp, psrc, len);
+ memcpy(pdst, ptmp, len);
+ }
+ if (psrc && (src < info->fix.smem_start || src > tmp))
+ iounmap(psrc);
+ if (pdst && (dst < info->fix.smem_start || dst > tmp))
+ iounmap(pdst);
+}
+#endif
+
+struct ge_var {
+ struct fb_info *info;
+ struct fb_var_screeninfo var[1];
+ struct fb_var_screeninfo new_var[1];
+ struct workqueue_struct *wq;
+ struct work_struct notifier;
+ struct timeval most_recent_flip_time;
+ int dirty;
+ int force_sync;
+ int vscnt; /* vsync counter */
+ spinlock_t lock[1];
+ void (*start)(struct ge_var *ge_var);
+ void (*stop)(struct ge_var *ge_var);
+ void (*get)(struct ge_var *ge_var, struct fb_var_screeninfo *var);
+ void (*set)(struct ge_var *ge_var, struct fb_var_screeninfo *var);
+ void (*sync)(struct ge_var *ge_var);
+};
+
+static struct ge_var *ge_var_s;
+
+static void ge_var_start(struct ge_var *ge_var);
+static void ge_var_stop(struct ge_var *ge_var);
+static void ge_var_get(struct ge_var *ge_var, struct fb_var_screeninfo *var);
+static void ge_var_set(struct ge_var *ge_var, struct fb_var_screeninfo *var);
+static void ge_var_sync(struct ge_var *ge_var);
+static void ge_var_sync1(struct ge_var *ge_var);
+static void ge_var_sync2(struct ge_var *ge_var);
+static void ge_var_sync3(struct ge_var *ge_var);
+
+static void ge_var_vsync_notifier(struct work_struct *work)
+{
+ struct ge_var *ge_var = container_of(work, struct ge_var, notifier);
+
+ ge_vo_wait_vsync();
+
+ spin_lock(ge_var->lock);
+ ge_var->vscnt++;
+ spin_unlock(ge_var->lock);
+
+ if (debug)
+ printk(KERN_DEBUG "vsync!\n");
+}
+
+static struct ge_var *create_ge_var(struct fb_info *info)
+{
+ struct ge_var *ge_var;
+
+ ge_var = (struct ge_var *)
+ kcalloc(1, sizeof(struct ge_var), GFP_KERNEL);
+
+ ge_var->wq = create_singlethread_workqueue("ge_var_wq");
+ ge_var->info = info;
+ ge_var->start = &ge_var_start;
+ ge_var->stop = &ge_var_stop;
+ ge_var->get = &ge_var_get;
+ ge_var->set = &ge_var_set;
+ ge_var->sync = &ge_var_sync;
+
+ do_gettimeofday(&ge_var->most_recent_flip_time);
+
+ INIT_WORK(&ge_var->notifier, ge_var_vsync_notifier);
+
+ ge_var->start(ge_var);
+
+ return ge_var;
+}
+
+static void release_ge_var(struct ge_var *ge_var)
+{
+ if (ge_var) {
+ ge_var->stop(ge_var);
+ flush_workqueue(ge_var->wq);
+ destroy_workqueue(ge_var->wq);
+ kfree(ge_var);
+ }
+}
+
+static void ge_var_start(struct ge_var *ge_var)
+{
+ spin_lock_init(ge_var->lock);
+ queue_work(ge_var->wq, &ge_var->notifier);
+}
+
+static void ge_var_stop(struct ge_var *ge_var)
+{
+}
+
+static void ge_var_get(struct ge_var *ge_var, struct fb_var_screeninfo *var)
+{
+ spin_lock(ge_var->lock);
+ memcpy(var, ge_var->var, sizeof(struct fb_var_screeninfo));
+ spin_unlock(ge_var->lock);
+}
+
+static void ge_var_set(struct ge_var *ge_var, struct fb_var_screeninfo *var)
+{
+ spin_lock(ge_var->lock);
+ if (memcmp(ge_var->new_var, var, sizeof(struct fb_var_screeninfo))) {
+ memcpy(ge_var->new_var, var, sizeof(struct fb_var_screeninfo));
+ ge_var->dirty++;
+ }
+ spin_unlock(ge_var->lock);
+
+ if (vbl || (var->activate & FB_ACTIVATE_VBL))
+ ge_var->sync(ge_var);
+ else
+ ge_var_sync1(ge_var);
+}
+
+static void ge_var_sync(struct ge_var *ge_var)
+{
+ if (sync3)
+ return ge_var_sync3(ge_var);
+
+ if (sync2)
+ return ge_var_sync2(ge_var);
+
+ ge_var_sync1(ge_var);
+}
+
+/* flip and don't wait */
+static void ge_var_sync1(struct ge_var *ge_var)
+{
+ spin_lock(ge_var->lock);
+
+ if (ge_var->dirty == 0) {
+ spin_unlock(ge_var->lock);
+ return;
+ }
+
+ memcpy(ge_var->var, ge_var->new_var, sizeof(struct fb_var_screeninfo));
+ spin_unlock(ge_var->lock);
+
+ ge_vo_pan_display(ge_var->var, ge_var->info);
+ ge_var->dirty = 0;
+}
+
+/* for double buffer */
+static void ge_var_sync2(struct ge_var *ge_var)
+{
+ struct timeval t;
+ struct timeval *mrft;
+ unsigned long us;
+
+ spin_lock(ge_var->lock);
+
+ if (ge_var->dirty == 0) {
+ spin_unlock(ge_var->lock);
+ return;
+ }
+
+ memcpy(ge_var->var, ge_var->new_var, sizeof(struct fb_var_screeninfo));
+
+ mrft = &ge_var->most_recent_flip_time;
+ do_gettimeofday(&t);
+
+ us = (t.tv_sec - mrft->tv_sec) * 1000000 +
+ (t.tv_usec - mrft->tv_usec);
+
+ spin_unlock(ge_var->lock);
+
+ ge_vo_pan_display(ge_var->var, ge_var->info);
+ ge_var->dirty = 0;
+
+ /* 60 fps */
+ if (us < 16667) {
+ if (debug) {
+ struct timeval t1;
+ struct timeval t2;
+ int ms;
+ do_gettimeofday(&t1);
+ ge_vo_wait_vsync();
+ do_gettimeofday(&t2);
+ ms = (t2.tv_sec - t1.tv_sec) * 1000 +
+ (t2.tv_usec - t1.tv_usec) / 1000;
+ printk(KERN_DEBUG "vsync2: wait vsync for %d ms\n", ms);
+ } else
+ ge_vo_wait_vsync();
+ }
+
+ do_gettimeofday(&ge_var->most_recent_flip_time);
+}
+
+/* for triple buffer */
+static void ge_var_sync3(struct ge_var *ge_var)
+{
+ spin_lock(ge_var->lock);
+
+ if (ge_var->dirty == 0) {
+ spin_unlock(ge_var->lock);
+ return;
+ }
+
+ if (ge_var->vscnt == 0) {
+ struct timeval t1;
+ struct timeval t2;
+ int ms;
+ int tmax = 16;
+
+ if (debug)
+ do_gettimeofday(&t1);
+
+ while (tmax && !ge_var->vscnt) {
+ usleep_range(1000, 2000);
+ tmax--;
+ }
+
+ if (debug) {
+ do_gettimeofday(&t2);
+ ms = (t2.tv_sec - t1.tv_sec) * 1000 +
+ (t2.tv_usec - t1.tv_usec) / 1000;
+ printk(KERN_DEBUG "vsync3: wait vsync for %d ms\n", ms);
+ }
+ }
+
+ memcpy(ge_var->var, ge_var->new_var, sizeof(struct fb_var_screeninfo));
+ spin_unlock(ge_var->lock);
+
+ ge_vo_pan_display(ge_var->var, ge_var->info);
+ ge_var->dirty = 0;
+ ge_var->vscnt = 0;
+
+ queue_work(ge_var->wq, &ge_var->notifier);
+}
+
+#if 0
+static unsigned long fb_get_phys_addr(struct fb_info *info,
+ struct fb_var_screeninfo *var)
+{
+ unsigned long offset;
+
+ if (!var)
+ var = &info->var;
+
+ offset = (var->yoffset * var->xres_virtual + var->xoffset);
+ offset *= var->bits_per_pixel >> 3;
+
+ return info->fix.smem_start + offset;
+}
+
+static unsigned long fb_get_disp_size(struct fb_info *info,
+ struct fb_var_screeninfo *var)
+{
+ unsigned long size;
+
+ if (!var)
+ var = &info->var;
+
+ size = (var->yres * var->xres_virtual);
+ size *= var->bits_per_pixel >> 3;
+
+ return size;
+}
+#endif
+
+static int fb_var_cmp(struct fb_var_screeninfo *var1,
+ struct fb_var_screeninfo *var2)
+{
+ /* Compare from xres to bits_per_pixel should be enough */
+ return memcmp(var1, var2, 28);
+}
+
+/**
+ * ge_init - Initial and display framebuffer.
+ *
+ * Fill the framebuffer with a default color, back.
+ * Display the framebuffer using GE AMX.
+ *
+ * Although VQ is supported in design, I just can't find any benefit
+ * from VQ. It wastes extra continuous physical memory, and runs much
+ * slower than direct register access. Moreover, the source code
+ * becomes more complex and is hard to maintain. Accessing VQ from
+ * the user space is also a nightmare. In brief, the overhead of VQ makes
+ * it useless. In order to gain the maximum performance
+ * from GE and to keep the driver simple, I'm going to stop using VQ.
+ * I will use VQ only when it is necessary.
+ *
+ * @info is the fb_info provided by framebuffer driver.
+ * @return zero on success.
+ */
+int ge_init(struct fb_info *info)
+{
+ static int boot_init; /* boot_init = 0 */
+ /*
+ * Booting time initialization
+ */
+ if (!boot_init) {
+ ge_var_s = create_ge_var(info);
+ boot_init = 1;
+ }
+
+ return 0;
+}
+
+/**
+ * ge_exit - Disable GE.
+ *
+ * No memory needs to be released here.
+ * Turn off the AMX to stop displaying framebuffer.
+ * Update the index of MMU.
+ *
+ * @info is fb_info from fbdev.
+ * @return zero on success.
+ */
+int ge_exit(struct fb_info *info)
+{
+ release_ge_var(ge_var_s);
+ release_mem_region(info->fix.mmio_start, info->fix.mmio_len);
+
+ return 0;
+}
+
+int ge_release(struct fb_info *info)
+{
+ struct ge_var *ge_var = ge_var_s;
+
+ ge_var->sync(ge_var); /* apply pending changes */
+
+ return 0;
+}
+
+/**
+ * ge_pan_display - Pans the display.
+ *
+ * Pan (or wrap, depending on the `vmode' field) the display using the
+ * `xoffset' and `yoffset' fields of the `var' structure.
+ * If the values don't fit, return -EINVAL.
+ *
+ * @var: frame buffer variable screen structure
+ * @info: frame buffer structure that represents a single frame buffer
+ *
+ * Returns negative errno on error, or zero on success.
+ */
+int ge_pan_display(struct fb_var_screeninfo *var, struct fb_info *info)
+{
+ struct ge_var *ge_var;
+
+ ge_var = ge_var_s;
+
+ if (debug)
+ printk(KERN_DEBUG "pan_display\n");
+ /*
+ printk(KERN_DEBUG "%s: xoff = %d, yoff = %d, xres = %d, yres = %d\n",
+ __func__, var->xoffset, var->yoffset,
+ info->var.xres, info->var.yres);
+ */
+ flipreq++;
+
+ if ((var->xoffset + info->var.xres > info->var.xres_virtual) ||
+ (var->yoffset + info->var.yres > info->var.yres_virtual)) {
+ /* Y-pan is used in most case.
+ * So please make sure that yres_virtual is
+ * greater than (yres + yoffset).
+ */
+ printk(KERN_ERR "%s: out of range\n", __func__);
+ return -EINVAL;
+ }
+
+ if ((var->activate & FB_ACTIVATE_MASK) == FB_ACTIVATE_NOW &&
+ fb_var_cmp(ge_var->new_var, var))
+ ge_var->set(ge_var, var);
+
+ return 0;
+}
+
+int ge_sync(struct fb_info *info)
+{
+ return 0;
+}
+
+int ge_blank(int mode, struct fb_info *info)
+{
+#ifdef HAVE_VPP
+ return vpp_set_blank(info, mode);
+#else
+ return 0;
+#endif
+}
+
+int ge_suspend(struct fb_info *info)
+{
+ return 0;
+}
+
+int ge_resume(struct fb_info *info)
+{
+ return 0;
+}
diff --git a/drivers/video/wmt/ge/ge_accel.h b/drivers/video/wmt/ge/ge_accel.h
new file mode 100644
index 00000000..d47d30f7
--- /dev/null
+++ b/drivers/video/wmt/ge/ge_accel.h
@@ -0,0 +1,79 @@
+/*
+ * Copyright (c) 2008-2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+ */
+
+#ifndef GE_ACCEL_H
+#define GE_ACCEL_H
+
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/fb.h>
+
+#ifndef __KERNEL__
+#define __KERNEL__
+#endif
+
+#define GE_DEBUG 0
+#define FB_ACCEL_WMT 0x8910
+#define MAX_XRES 1920
+#define MAX_YRES 1080
+#define GE_FB_NUM 3
+
+#define HAVE_MALI
+#define HAVE_VPP
+
+extern int vbl;
+extern int vsync;
+extern int sync2;
+
+unsigned int phy_mem_end(void);
+unsigned int phy_mem_end_sub(unsigned int size);
+int ge_init(struct fb_info *info);
+int ge_exit(struct fb_info *info);
+int ge_ioctl(struct fb_info *info, unsigned int cmd, unsigned long arg);
+int ge_pan_display(struct fb_var_screeninfo *var, struct fb_info *info);
+int ge_sync(struct fb_info *info);
+int ge_release(struct fb_info *info);
+int ge_blank(int mode, struct fb_info *info);
+int ge_suspend(struct fb_info *info);
+int ge_resume(struct fb_info *info);
+
+void ge_vo_get_default_var(struct fb_var_screeninfo *var);
+void ge_vo_wait_vsync(void);
+
+#ifdef HAVE_VPP
+#include "../vpp.h"
+extern void vpp_wait_vsync(int idx, int cnt);
+extern void vpp_set_mutex(int idx, int lock);
+extern void vpp_get_info(int fbn, struct fb_var_screeninfo *var);
+extern int vpp_pan_display(struct fb_var_screeninfo *var, struct fb_info *info);
+extern int vpp_set_blank(struct fb_info *info, int blank);
+extern int vpp_set_par(struct fb_info *info);
+extern int wmtfb_check_var(struct fb_var_screeninfo *var, struct fb_info *info);
+#endif /* HAVE_VPP */
+
+#ifdef HAVE_MALI
+extern unsigned int mali_ump_secure_id;
+extern unsigned int (*mali_get_ump_secure_id)(unsigned int addr,
+ unsigned int size);
+extern void (*mali_put_ump_secure_id)(unsigned int ump_id);
+#endif /* HAVE_MALI */
+
+#endif
diff --git a/drivers/video/wmt/ge/ge_main.c b/drivers/video/wmt/ge/ge_main.c
new file mode 100644
index 00000000..1109ff5f
--- /dev/null
+++ b/drivers/video/wmt/ge/ge_main.c
@@ -0,0 +1,1003 @@
+/*
+ * Copyright (c) 2008-2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+ */
+
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/fs.h>
+#include <linux/kernel.h>
+#include <linux/platform_device.h>
+#include <linux/errno.h>
+#include <linux/uaccess.h>
+#include <linux/kdev_t.h>
+#include <linux/cdev.h>
+#include <linux/fb.h>
+#include <linux/dma-mapping.h>
+#include <asm/page.h>
+#include <linux/mm.h>
+#include <linux/sched.h>
+
+#include "ge_accel.h"
+
+#define HAVE_MALI
+#define WMT_MB
+
+#ifdef HAVE_MALI
+#include "../mali.h"
+static struct mali_device *malidev;
+#define UMP_INVALID_SECURE_ID ((unsigned int)-1)
+#define GET_UMP_SECURE_ID _IOWR('m', 310, unsigned int)
+#define GET_UMP_SECURE_ID_BUF1 _IOWR('m', 311, unsigned int)
+#define GET_UMP_SECURE_ID_BUF2 _IOWR('m', 312, unsigned int)
+#define MALI_GET_UMP_SECURE_ID _IOWR('m', 320, unsigned int)
+#define MALI_PUT_UMP_SECURE_ID _IOWR('m', 321, unsigned int)
+#endif /* HAVE_MALI */
+
+#define USE_SID_ALIAS
+/*
+#define DEBUG_SID_ALIAS
+*/
+
+#ifdef USE_SID_ALIAS
+#define SID_IDX_MAX 16
+#define SID_GET_INDEX_FROM_ALIAS _IOWR('s', 100, unsigned int)
+#define SID_SET_ALIAS _IOWR('s', 101, unsigned int)
+#define SID_GET_ALIAS _IOWR('s', 102, unsigned int)
+#define SID_GET_AND_RESET_ALIAS _IOWR('s', 103, unsigned int)
+#define SID_DUMP _IOWR('s', 104, unsigned int)
+
+struct sid_alias {
+ int sid;
+ int alias;
+};
+
+static struct sid_alias sid_alias_buf[SID_IDX_MAX];
+static spinlock_t sid_lock;
+
+int sid_get_index_from_empty(int *index)
+{
+ int i;
+
+ for (i = 0; i < SID_IDX_MAX; i++) {
+ if (!sid_alias_buf[i].sid && !sid_alias_buf[i].alias) {
+ *index = i;
+ return 0;
+ }
+ }
+ return -1;
+}
+
+int sid_get_index_from_alias(int alias, int *index)
+{
+ int i;
+
+ for (i = 0; i < SID_IDX_MAX; i++) {
+ if (sid_alias_buf[i].alias == alias) {
+ *index = i;
+ return 0;
+ }
+ }
+ return -1;
+}
+
+int sid_clear_alias(int sid)
+{
+ int idx;
+
+ for (idx = 0; idx < SID_IDX_MAX; idx++) {
+ if (sid == sid_alias_buf[idx].sid ||
+ sid == sid_alias_buf[idx].alias) {
+ sid_alias_buf[idx].sid = 0;
+ sid_alias_buf[idx].alias = 0;
+ }
+ }
+
+ return 0;
+}
+
+int sid_set_alias(int sid, int alias)
+{
+ int idx;
+
+ if (alias <= 0)
+ return sid_clear_alias(sid);
+
+ sid_clear_alias(alias);
+
+ if (sid_get_index_from_alias(sid, &idx) == 0) {
+ sid_alias_buf[idx].alias = alias;
+ if (alias <= 0)
+ sid_alias_buf[idx].sid = 0;
+ return 0;
+ }
+
+ sid_clear_alias(sid);
+
+ if (sid_get_index_from_empty(&idx) == 0) {
+ sid_alias_buf[idx].sid = sid;
+ sid_alias_buf[idx].alias = alias;
+ return 0;
+ }
+
+ return -1;
+}
+
+int sid_get_alias(int sid, int *alias)
+{
+ int i;
+ int val = -1;
+
+ for (i = 0; i < SID_IDX_MAX; i++) {
+ if (sid_alias_buf[i].sid == sid) {
+ if (sid_alias_buf[i].alias > 0) {
+ val = sid_alias_buf[i].alias;
+ if (sid != val)
+ break;
+ } else {
+ /* remove invalid alias */
+ sid_alias_buf[i].sid = 0;
+ sid_alias_buf[i].alias = 0;
+ }
+ }
+ }
+
+ if (val > 0) {
+ *alias = val;
+ return 0;
+ } else
+ return -1;
+}
+
+int sid_get_and_reset_alias(int sid, int *alias)
+{
+ int i;
+ int val = -1;
+
+ for (i = 0; i < SID_IDX_MAX; i++) {
+ if (sid_alias_buf[i].sid == sid) {
+ if (sid_alias_buf[i].alias > 0) {
+ val = sid_alias_buf[i].alias;
+ sid_alias_buf[i].sid = val;
+ if (sid != val)
+ break;
+ } else {
+ /* remove invalid alias */
+ sid_alias_buf[i].sid = 0;
+ sid_alias_buf[i].alias = 0;
+ }
+ }
+ }
+
+ if (val > 0) {
+ *alias = val;
+ return 0;
+ } else
+ return -1;
+}
+
+void sid_dump(void)
+{
+ int i;
+
+ for (i = 0; i < SID_IDX_MAX; i++)
+ printk(KERN_ERR "sid_alias_buf[%d] = { %d, %d }\n",
+ i, sid_alias_buf[i].sid, sid_alias_buf[i].alias);
+}
+#endif
+
+#ifndef FBIO_WAITFORVSYNC
+#define FBIO_WAITFORVSYNC _IOW('F', 0x20, u_int32_t)
+#endif
+
+#define GEIO_MAGIC 0x69
+
+#ifdef GEIO_MAGIC
+#define GEIOGET_CHIP_ID _IOR(GEIO_MAGIC, 5, unsigned int)
+#endif
+
+static int vtotal = 32;
+static int mbsize;
+
+module_param(vtotal, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(vtotal, "Maximum GE memory size in MiB");
+
+module_param(mbsize, int, S_IRUSR | S_IWUSR | S_IWGRP | S_IRGRP | S_IROTH);
+MODULE_PARM_DESC(mbsize, "WMT-MB size in MiB");
+
+static struct fb_fix_screeninfo gefb_fix = {
+ .id = "gefb",
+ .smem_start = 0,
+ .smem_len = 0,
+ .type = FB_TYPE_PACKED_PIXELS,
+ .type_aux = 0,
+ .visual = FB_VISUAL_TRUECOLOR,
+ .xpanstep = 1,
+ .ypanstep = 1,
+ .ywrapstep = 1,
+ .line_length = 0,
+#ifdef HAVE_MALI
+ .mmio_start = 0xd8080000,
+ .mmio_len = 0x10000,
+#else
+ .mmio_start = 0,
+ .mmio_len = 0,
+#endif
+ .accel = FB_ACCEL_WMT
+};
+
+static struct fb_var_screeninfo gefb_var = {
+ .xres = CONFIG_DEFAULT_RESX,
+ .yres = CONFIG_DEFAULT_RESY,
+ .xres_virtual = CONFIG_DEFAULT_RESX,
+ .yres_virtual = CONFIG_DEFAULT_RESY,
+ /*
+ .bits_per_pixel = 32,
+ .red = {16, 8, 0},
+ .green = {8, 8, 0},
+ .blue = {0, 8, 0},
+ .transp = {0, 0, 0},
+ */
+ .bits_per_pixel = 16,
+ .red = {11, 5, 0},
+ .green = {5, 6, 0},
+ .blue = {0, 5, 0},
+ .transp = {0, 0, 0},
+ .activate = FB_ACTIVATE_NOW | FB_ACTIVATE_FORCE,
+ .height = -1,
+ .width = -1,
+ .pixclock = 39721,
+ .left_margin = 40,
+ .right_margin = 24,
+ .upper_margin = 32,
+ .lower_margin = 11,
+ .hsync_len = 96,
+ .vsync_len = 2,
+ .vmode = FB_VMODE_NONINTERLACED
+};
+
+static int gefb_open(struct fb_info *info, int user)
+{
+ return 0;
+}
+
+static int gefb_release(struct fb_info *info, int user)
+{
+ return ge_release(info);
+}
+
+static int gefb_check_var(struct fb_var_screeninfo *var,
+ struct fb_info *info)
+{
+#ifdef HAVE_VPP
+ return wmtfb_check_var(var, info);
+#else
+ switch (var->bits_per_pixel) {
+ case 1:
+ case 8:
+ if (var->red.offset > 8) {
+ /* LUT8 */
+ var->red.offset = 0;
+ var->red.length = 8;
+ var->green.offset = 0;
+ var->green.length = 8;
+ var->blue.offset = 0;
+ var->blue.length = 8;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ }
+ break;
+ case 16:
+ if (var->transp.length) {
+ /* ARGB 1555 */
+ var->red.offset = 10;
+ var->red.length = 5;
+ var->green.offset = 5;
+ var->green.length = 5;
+ var->blue.offset = 0;
+ var->blue.length = 5;
+ var->transp.offset = 15;
+ var->transp.length = 1;
+ } else {
+ /* RGB 565 */
+ var->red.offset = 11;
+ var->red.length = 5;
+ var->green.offset = 5;
+ var->green.length = 6;
+ var->blue.offset = 0;
+ var->blue.length = 5;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ }
+ break;
+ case 24:
+ /* RGB 888 */
+ var->red.offset = 16;
+ var->red.length = 8;
+ var->green.offset = 8;
+ var->green.length = 8;
+ var->blue.offset = 0;
+ var->blue.length = 8;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ break;
+ case 32:
+ /* ARGB 8888 */
+ var->red.offset = 16;
+ var->red.length = 8;
+ var->green.offset = 8;
+ var->green.length = 8;
+ var->blue.offset = 0;
+ var->blue.length = 8;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ break;
+ }
+ return 0;
+#endif
+}
+
+static int gefb_set_par(struct fb_info *info)
+{
+ struct fb_var_screeninfo *var = &info->var;
+
+ /* init your hardware here */
+ if (var->bits_per_pixel == 8)
+ info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
+ else
+ info->fix.visual = FB_VISUAL_TRUECOLOR;
+
+ if (ge_init(info))
+ return -ENOMEM;
+
+#ifdef HAVE_VPP
+ vpp_set_par(info);
+#endif
+
+ info->fix.line_length = var->xres_virtual * var->bits_per_pixel / 8;
+
+ return 0;
+}
+
+static int gefb_setcolreg(unsigned regno, unsigned red,
+ unsigned green, unsigned blue,
+ unsigned transp, struct fb_info *info)
+{
+ if (regno >= 256) /* no. of hw registers */
+ return -EINVAL;
+
+ /* grayscale */
+
+ if (info->var.grayscale) {
+ /* grayscale = 0.30*R + 0.59*G + 0.11*B */
+ red = green = blue = (red * 77 + green * 151 + blue * 28) >> 8;
+ }
+
+ /* The following is for fbcon. */
+
+ if (info->fix.visual == FB_VISUAL_TRUECOLOR ||
+ info->fix.visual == FB_VISUAL_DIRECTCOLOR) {
+
+ if (regno >= 16)
+ return -EINVAL;
+
+ switch (info->var.bits_per_pixel) {
+ case 16:
+ ((unsigned int *)(info->pseudo_palette))[regno] =
+ (red & 0xf800) |
+ ((green & 0xfc00) >> 5) |
+ ((blue & 0xf800) >> 11);
+ break;
+ case 24:
+ case 32:
+ red >>= 8;
+ green >>= 8;
+ blue >>= 8;
+ ((unsigned int *)(info->pseudo_palette))[regno] =
+ (red << info->var.red.offset) |
+ (green << info->var.green.offset) |
+ (blue << info->var.blue.offset);
+ break;
+ }
+ }
+ return 0;
+}
+
+static int gefb_ioctl(struct fb_info *info, unsigned int cmd,
+ unsigned long arg)
+{
+ int retval = 0;
+
+ switch (cmd) {
+ case FBIO_WAITFORVSYNC:
+ ge_vo_wait_vsync();
+ break;
+#ifdef HAVE_MALI
+ case GET_UMP_SECURE_ID:
+ case GET_UMP_SECURE_ID_BUF1:
+ case GET_UMP_SECURE_ID_BUF2: {
+ unsigned int ump_id;
+ if (mali_get_ump_secure_id)
+ ump_id = (*mali_get_ump_secure_id)(info->fix.smem_start,
+ info->fix.smem_len);
+ else
+ ump_id = UMP_INVALID_SECURE_ID;
+ return put_user((unsigned int) ump_id,
+ (unsigned int __user *) arg);
+ }
+ case MALI_GET_UMP_SECURE_ID: {
+ unsigned int args[3];
+ unsigned int ump_id;
+ copy_from_user(args, (void *)arg, sizeof(unsigned int) * 3);
+
+ if (mali_get_ump_secure_id)
+ ump_id = (*mali_get_ump_secure_id)(args[0], args[1]);
+ else
+ ump_id = UMP_INVALID_SECURE_ID;
+
+ return put_user((unsigned int) ump_id,
+ (unsigned int __user *) args[2]);
+ }
+ case MALI_PUT_UMP_SECURE_ID: {
+ unsigned int ump_id = (unsigned int)arg;
+ if (mali_put_ump_secure_id)
+ (*mali_put_ump_secure_id)(ump_id);
+ break;
+ }
+#endif /* HAVE_MALI */
+#ifdef GEIO_MAGIC
+ case GEIOGET_CHIP_ID: {
+ unsigned int chip_id =
+ (*(unsigned int *)SYSTEM_CFG_CTRL_BASE_ADDR);
+ copy_to_user((void *)arg, (void *)&chip_id,
+ sizeof(unsigned int));
+ break;
+ }
+#endif /* GEIO_MAGIC */
+#ifdef USE_SID_ALIAS
+ case SID_GET_INDEX_FROM_ALIAS: {
+ unsigned int args[2];
+ unsigned int index = -1;
+ copy_from_user(args, (void *)arg, sizeof(unsigned int) * 2);
+ spin_lock(&sid_lock);
+ retval = sid_get_index_from_alias(args[0], &index);
+ spin_unlock(&sid_lock);
+ put_user(index, (unsigned int __user *)args[1]);
+ break;
+ }
+ case SID_SET_ALIAS: {
+ unsigned int args[2];
+ copy_from_user(args, (void *)arg, sizeof(unsigned int) * 2);
+ spin_lock(&sid_lock);
+ retval = sid_set_alias(args[0], args[1]);
+#ifdef DEBUG_SID_ALIAS
+ printk(KERN_DEBUG "sid_set_alias %d, %d, ret = %d\n",
+ args[0], args[1], retval);
+ sid_dump();
+#endif
+ spin_unlock(&sid_lock);
+ break;
+ }
+ case SID_GET_ALIAS: {
+ unsigned int args[2];
+ unsigned int alias = -1;
+ copy_from_user(args, (void *)arg, sizeof(unsigned int) * 2);
+ spin_lock(&sid_lock);
+ retval = sid_get_alias(args[0], &alias);
+ spin_unlock(&sid_lock);
+ put_user(alias, (unsigned int __user *)args[1]);
+ break;
+ }
+ case SID_GET_AND_RESET_ALIAS: {
+ unsigned int args[2];
+ unsigned int alias = -1;
+ copy_from_user(args, (void *)arg, sizeof(unsigned int) * 2);
+ spin_lock(&sid_lock);
+ retval = sid_get_and_reset_alias(args[0], &alias);
+#ifdef DEBUG_SID_ALIAS
+ printk(KERN_DEBUG "sid_get_and_reset_alias %d, %d, ret = %d\n",
+ args[0], alias, retval);
+ sid_dump();
+#endif
+ spin_unlock(&sid_lock);
+ put_user(alias, (unsigned int __user *)args[1]);
+ break;
+ }
+ case SID_DUMP: {
+ spin_lock(&sid_lock);
+ copy_to_user((void *)arg, sid_alias_buf,
+ sizeof(struct sid_alias) * SID_IDX_MAX);
+ retval = 0;
+ spin_unlock(&sid_lock);
+ break;
+ }
+#endif
+ default:
+ break;
+ }
+
+ return retval;
+}
+
+int gefb_hw_cursor(struct fb_info *info, struct fb_cursor *cursor)
+{
+ return 0;
+}
+
+static int gefb_mmap(struct fb_info *info, struct vm_area_struct *vma)
+{
+ unsigned long off;
+ unsigned long start;
+ u32 len;
+ int ismmio = 0;
+
+ if (!info)
+ return -ENODEV;
+ if (vma->vm_pgoff > (~0UL >> PAGE_SHIFT))
+ return -EINVAL;
+ off = vma->vm_pgoff << PAGE_SHIFT;
+
+ /* frame buffer memory */
+ start = info->fix.smem_start;
+ len = PAGE_ALIGN((start & ~PAGE_MASK) + info->fix.smem_len);
+ if (off >= len) {
+ /* memory mapped io */
+ off -= len;
+ /*
+ if (info->var.accel_flags) {
+ return -EINVAL;
+ }
+ */
+ start = info->fix.mmio_start;
+ len = PAGE_ALIGN((start & ~PAGE_MASK) + info->fix.mmio_len);
+ ismmio = 1;
+ }
+ start &= PAGE_MASK;
+ if ((vma->vm_end - vma->vm_start + off) > len)
+ return -EINVAL;
+ off += start;
+ vma->vm_pgoff = off >> PAGE_SHIFT;
+ /* This is an IO map - tell maydump to skip this VMA */
+ vma->vm_flags |= VM_IO | VM_RESERVED;
+ vma->vm_page_prot = vm_get_page_prot(vma->vm_flags);
+
+ if (ismmio)
+ vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
+ else
+ vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
+
+ if (io_remap_pfn_range(vma, vma->vm_start, off >> PAGE_SHIFT,
+ vma->vm_end - vma->vm_start, vma->vm_page_prot))
+ return -EAGAIN;
+ return 0;
+}
+
+static struct fb_ops gefb_ops = {
+ .owner = THIS_MODULE,
+ .fb_open = gefb_open,
+ .fb_release = gefb_release,
+ .fb_check_var = gefb_check_var,
+ .fb_set_par = gefb_set_par,
+ .fb_setcolreg = gefb_setcolreg,
+ .fb_pan_display = ge_pan_display,
+ .fb_fillrect = cfb_fillrect,
+ .fb_copyarea = cfb_copyarea,
+ .fb_imageblit = cfb_imageblit,
+ .fb_blank = ge_blank,
+ .fb_cursor = gefb_hw_cursor,
+ .fb_ioctl = gefb_ioctl,
+ .fb_sync = ge_sync,
+ .fb_mmap = gefb_mmap,
+};
+
+#define OPT_EQUAL(opt, name) (!strncmp(opt, name, strlen(name)))
+#define OPT_INTVAL(opt, name) kstrtoul(opt + strlen(name) + 1, 0, NULL)
+#define OPT_STRVAL(opt, name) (opt + strlen(name))
+
+static inline char *get_opt_string(const char *this_opt, const char *name)
+{
+ const char *p;
+ int i;
+ char *ret;
+
+ p = OPT_STRVAL(this_opt, name);
+ i = 0;
+ while (p[i] && p[i] != ' ' && p[i] != ',')
+ i++;
+ ret = kmalloc(i + 1, GFP_KERNEL);
+ if (ret) {
+ strncpy(ret, p, i);
+ ret[i] = '\0';
+ }
+ return ret;
+}
+
+static inline int get_opt_int(const char *this_opt, const char *name,
+ int *ret)
+{
+ if (!ret)
+ return 0;
+
+ if (!OPT_EQUAL(this_opt, name))
+ return 0;
+
+ *ret = OPT_INTVAL(this_opt, name);
+
+ return 1;
+}
+
+static inline int get_opt_bool(const char *this_opt, const char *name,
+ int *ret)
+{
+ if (!ret)
+ return 0;
+
+ if (OPT_EQUAL(this_opt, name)) {
+ if (this_opt[strlen(name)] == '=')
+ *ret = kstrtoul(this_opt + strlen(name) + 1, 0, NULL);
+ else
+ *ret = 1;
+ } else {
+ if (OPT_EQUAL(this_opt, "no") && OPT_EQUAL(this_opt + 2, name))
+ *ret = 0;
+ else
+ return 0;
+ }
+ return 1;
+}
+
+static int __init gefb_setup(char *options)
+{
+ char *this_opt;
+
+ if (!options || !*options)
+ return 0;
+
+ /* The syntax is:
+ * video=gefb:[<param>][,<param>=<val>] ...
+ * e.g.,
+ * video=gefb:vtotal=12,sync2
+ */
+
+ while ((this_opt = strsep(&options, ",")) != NULL) {
+ if (!*this_opt)
+ continue;
+ if (get_opt_int(this_opt, "vtotal", &vtotal))
+ ;
+ else if (get_opt_int(this_opt, "mbsize", &mbsize))
+ ;
+ else if (get_opt_bool(this_opt, "vbl", &vbl))
+ ;
+ else if (get_opt_bool(this_opt, "vsync", &vsync))
+ ;
+ else if (get_opt_bool(this_opt, "sync2", &sync2))
+ ;
+ }
+
+ return 0;
+}
+
+#ifdef HAVE_MALI
+static struct mali_device *add_mali_device(unsigned int *smem_start_ptr,
+ unsigned int *smem_len_ptr)
+{
+ unsigned int len;
+ struct mali_device *dev = create_mali_device();
+ if (dev) {
+ dev->get_memory_base(smem_start_ptr);
+ dev->get_memory_size(&len);
+ *smem_start_ptr += len;
+ *smem_len_ptr -= len;
+ dev->set_mem_validation_base(*smem_start_ptr);
+ dev->set_mem_validation_size(*smem_len_ptr);
+ }
+ return dev;
+}
+#endif /* HAVE_MALI */
+
+#ifdef WMT_MB
+static int get_mbsize(void)
+{
+ /* It is bad to read U-Boot partition directly.
+ * I will remove this code soon.
+ * -- Vincent
+ */
+ unsigned char buf[32];
+ int varlen = 32;
+ int val;
+
+ if (wmt_getsyspara("mbsize", buf, &varlen) == 0)
+ sscanf(buf, "%dM", &val);
+ else
+ val = 0;
+
+ return val;
+}
+
+static void add_mb_device(unsigned int *smem_start_ptr,
+ unsigned int *smem_len_ptr)
+{
+ unsigned int len = mbsize << 20;
+
+ if (*smem_len_ptr > len)
+ *smem_len_ptr -= len;
+}
+#endif /* WMT_MB */
+
+static int __devinit gefb_probe(struct platform_device *dev)
+{
+ struct fb_info *info;
+ int cmap_len, retval;
+ char mode_option[] = "1024x768@60";
+ unsigned int smem_start;
+ unsigned int smem_len;
+ unsigned int len;
+ unsigned int min_smem_len;
+
+ /* Allocate fb_info and par.*/
+ info = framebuffer_alloc(sizeof(unsigned int) * 16, &dev->dev);
+ if (!info)
+ return -ENOMEM;
+
+ /* Set default fb_info */
+ info->fbops = &gefb_ops;
+ info->fix = gefb_fix;
+
+ info->var = gefb_var;
+ ge_vo_get_default_var(&info->var);
+
+ smem_start = (num_physpages << PAGE_SHIFT);
+ smem_len = phy_mem_end() - smem_start;
+
+#ifdef HAVE_MALI
+ malidev = add_mali_device(&smem_start, &smem_len);
+#endif /* HAVE_MALI */
+
+#ifdef WMT_MB
+ add_mb_device(&smem_start, &smem_len);
+#endif /* WMT_MB */
+
+ /* Set frame buffer region */
+
+ len = info->var.xres * info->var.yres *
+ (info->var.bits_per_pixel >> 3);
+ len *= GE_FB_NUM;
+ min_smem_len = (len + PAGE_MASK) & ~PAGE_MASK;
+
+ if (smem_len < min_smem_len) {
+ printk(KERN_ERR "%s: claim region of 0x%08x-0x%08x failed!\n",
+ __func__, smem_start, smem_start + min_smem_len);
+ return -EIO;
+ }
+
+ info->fix.smem_start = smem_start;
+
+ if (smem_len > (vtotal << 20))
+ smem_len = (vtotal << 20);
+
+ info->fix.smem_len = smem_len;
+
+ if (!request_mem_region(info->fix.smem_start,
+ info->fix.smem_len, "gefb")) {
+ printk(KERN_WARNING
+ "%s: request memory region failed at 0x%08lx\n",
+ __func__, info->fix.smem_start);
+ }
+
+ info->screen_base = ioremap(info->fix.smem_start,
+ info->fix.smem_len);
+ if (!info->screen_base) {
+ printk(KERN_ERR "%s: ioremap fail %d bytes at %p\n",
+ __func__, info->fix.smem_len,
+ (void *)info->fix.smem_start);
+ return -EIO;
+ }
+
+ printk(KERN_INFO
+ "gefb: phys 0x%08lx, virt 0x%08lx, total %d KB\n",
+ info->fix.smem_start, (unsigned long)info->screen_base,
+ info->fix.smem_len >> 10);
+
+ /*
+ * The pseudopalette is an 16-member array for fbcon.
+ */
+ info->pseudo_palette = info->par;
+ info->par = NULL;
+ info->flags = FBINFO_DEFAULT; /* flag for fbcon */
+
+ /*
+ * This should give a reasonable default video mode.
+ */
+ retval = fb_find_mode(&info->var, info, mode_option,
+ NULL, 0, NULL, 8);
+
+ if (!retval || retval == 4)
+ return -EINVAL;
+
+ /*
+ * This has to been done !!!
+ */
+ cmap_len = 256; /* Be the same as VESA */
+ retval = fb_alloc_cmap(&info->cmap, cmap_len, 0);
+ if (retval < 0)
+ printk(KERN_ERR "%s: fb_alloc_cmap fail.\n", __func__);
+
+ /*
+ * The following is done in the case of
+ * having hardware with a static mode.
+ */
+ info->var = gefb_var;
+
+ /*
+ * Get setting from video output device.
+ */
+ ge_vo_get_default_var(&info->var);
+
+ /*
+ * For drivers that can...
+ */
+ gefb_check_var(&info->var, info);
+
+ /*
+ * Apply setting
+ */
+ gefb_set_par(info);
+ ge_pan_display(&info->var, info);
+
+ if (register_framebuffer(info) < 0) {
+ ge_exit(info);
+ return -EINVAL;
+ }
+ info->dev->power.async_suspend = 1; /* Add by Charles */
+ dev_set_drvdata(&dev->dev, info);
+
+#ifdef USE_SID_ALIAS
+ spin_lock_init(&sid_lock);
+ memset(sid_alias_buf, 0, sizeof(struct sid_alias) * SID_IDX_MAX);
+#endif
+
+ return 0;
+}
+
+static int gefb_remove(struct platform_device *dev)
+{
+ struct fb_info *info = dev_get_drvdata(&dev->dev);
+
+ if (info) {
+ ge_exit(info);
+ unregister_framebuffer(info);
+ fb_dealloc_cmap(&info->cmap);
+ framebuffer_release(info);
+ }
+ return 0;
+}
+
+static int gefb_suspend(struct platform_device *dev, pm_message_t state)
+{
+ struct fb_info *info = dev_get_drvdata(&dev->dev);
+
+ if (info)
+ ge_suspend(info);
+
+#ifdef HAVE_MALI
+ if (malidev)
+ malidev->suspend(1);
+#endif
+
+ return 0;
+}
+
+static int gefb_resume(struct platform_device *dev)
+{
+ struct fb_info *info = dev_get_drvdata(&dev->dev);
+
+ if (info)
+ ge_resume(info);
+
+#ifdef HAVE_MALI
+ if (malidev)
+ malidev->resume(1);
+#endif
+
+ return 0;
+}
+
+static struct platform_driver gefb_driver = {
+ .driver.name = "gefb",
+ .probe = gefb_probe,
+ .remove = gefb_remove,
+ .suspend = gefb_suspend,
+ .resume = gefb_resume,
+};
+
+static u64 gefb_dma_mask = 0xffffffffUL;
+static struct platform_device gefb_device = {
+ .name = "gefb",
+ .dev = {
+ .dma_mask = &gefb_dma_mask,
+ .coherent_dma_mask = ~0,
+ },
+};
+
+#ifdef WMT_MB
+static int __init mbsize_setup(char *options)
+{
+ char *this_opt;
+
+ if (!options || !*options)
+ return 0;
+
+ while ((this_opt = strsep(&options, ",")) != NULL) {
+ if (!*this_opt)
+ continue;
+ sscanf(this_opt, "%dM", &mbsize);
+ printk(KERN_DEBUG "gefb: detected mbsize = %d MiB\n", mbsize);
+ }
+
+ return 0;
+}
+__setup("mbsize=", mbsize_setup);
+#endif /* WMT_MB */
+
+static int __init gefb_init(void)
+{
+ int ret;
+ char *option = NULL;
+
+ fb_get_options("gefb", &option);
+ gefb_setup(option);
+
+#ifdef WMT_MB
+ /* It is bad to read U-Boot partition directly.
+ * I will remove this code soon.
+ * -- Vincent
+ */
+ if (!mbsize) {
+ mbsize = get_mbsize();
+ printk(KERN_ERR "Please add \'mbsize=%dM\' in bootargs!",
+ mbsize);
+ }
+#endif /* WMT_MB */
+
+ ret = platform_driver_register(&gefb_driver);
+ if (!ret) {
+ ret = platform_device_register(&gefb_device);
+ if (ret)
+ platform_driver_unregister(&gefb_driver);
+ }
+
+ return ret;
+}
+module_init(gefb_init);
+
+static void __exit gefb_exit(void)
+{
+ release_mali_device(malidev);
+
+ platform_driver_unregister(&gefb_driver);
+ platform_device_unregister(&gefb_device);
+ return;
+}
+
+module_exit(gefb_exit);
+
+MODULE_AUTHOR("WonderMedia Technologies, Inc.");
+MODULE_DESCRIPTION("WMT GE driver");
+MODULE_LICENSE("GPL");
+
diff --git a/drivers/video/wmt/govrh.c b/drivers/video/wmt/govrh.c
new file mode 100644
index 00000000..7f851909
--- /dev/null
+++ b/drivers/video/wmt/govrh.c
@@ -0,0 +1,1926 @@
+/*++
+ * linux/drivers/video/wmt/govrh.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define GOVRH_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+#include "govrh.h"
+
+#ifdef WMT_FTBLK_GOVRH
+void govrh_reg_dump(vpp_mod_base_t *base)
+{
+ govrh_mod_t *p_govr = (govrh_mod_t *) base;
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ int igs_mode[4] = {888, 555, 666, 565};
+
+ DPRINT("========== GOVRH register dump ==========\n");
+ vpp_reg_dump((unsigned int)base->mmio, 512);
+ DPRINT("=========================================\n");
+ DPRINT("MIF enable %d\n", regs->mif.b.enable);
+ DPRINT("color bar enable %d,mode %d,inv %d\n",
+ regs->cb_enable.b.enable, regs->cb_enable.b.mode,
+ regs->cb_enable.b.inversion);
+ DPRINT("---------- frame buffer ----------\n");
+ DPRINT("colr format %s\n",
+ vpp_colfmt_str[govrh_get_color_format(p_govr)]);
+ DPRINT("width active %d,fb %d\n", regs->pixwid, regs->bufwid);
+ DPRINT("Y addr 0x%x, C addr 0x%x\n", regs->ysa, regs->csa);
+ DPRINT("Y addr2 0x%x, C addr2 0x%x\n", regs->ysa2, regs->csa2);
+ DPRINT("H crop %d, V crop %d\n", regs->hcrop, regs->vcrop);
+ DPRINT("source format %s,H264 %d\n",
+ (regs->srcfmt) ? "field" : "frame", regs->h264_input_en);
+ DPRINT("H scaleup %d,dirpath %d\n", regs->hscale_up, regs->dirpath);
+ DPRINT("---------- GOVRH TG1 ----------\n");
+ DPRINT("TG enable %d, Twin mode %d\n",
+ regs->tg_enable.b.enable, regs->tg_enable.b.mode);
+ DPRINT("DVO clk %d,%d,Read cyc %d\n",
+ vpp_get_base_clock(p_govr->mod),
+ auto_pll_divisor((p_govr->mod == VPP_MOD_GOVRH) ?
+ DEV_HDMILVDS : DEV_DVO, GET_FREQ, 0, 0),
+ regs->read_cyc);
+ DPRINT("H total %d, Sync %d, beg %d, end %d\n",
+ regs->h_allpxl, regs->hdmi_hsynw,
+ regs->actpx_bg, regs->actpx_end);
+ DPRINT("V total %d, Sync %d, beg %d, end %d\n",
+ regs->v_allln, regs->hdmi_vbisw,
+ regs->actln_bg, regs->actln_end);
+ DPRINT("VBIE %d,PVBI %d\n", regs->vbie_line, regs->pvbi_line);
+ DPRINT("---------- GOVRH TG2 ----------\n");
+ DPRINT("H total %d, Sync %d, beg %d, end %d\n",
+ regs->h_allpxl2, regs->hdmi_hsynw2,
+ regs->actpx_bg2, regs->actpx_end2);
+ DPRINT("V total %d, Sync %d, beg %d, end %d\n",
+ regs->v_allln2, regs->hdmi_vbisw2,
+ regs->actln_bg2, regs->actln_end2);
+ DPRINT("VBIE %d,PVBI %d\n", regs->vbie_line2, regs->pvbi_line2);
+ DPRINT("---------- DVO ----------------\n");
+ DPRINT("DVO enable %d,data width %d bits\n",
+ regs->dvo_set.b.enable, (regs->dvo_set.b.outwidth) ? 12 : 24);
+ DPRINT("DVO color format %s\n",
+ vpp_colfmt_str[govrh_get_dvo_color_format(p_govr)]);
+ DPRINT("Polar H %s,V %s\n",
+ (regs->dvo_set.b.hsync_polar) ? "Low" : "High",
+ (regs->dvo_set.b.vsync_polar) ? "Low" : "High");
+ DPRINT("DVO RGB mode %d,%s\n", igs_mode[regs->igs_mode.b.mode],
+ (regs->igs_mode.b.ldi) ? "msb" : "lsb");
+ DPRINT("RGB swap %d\n", regs->dvo_set.b.rgb_swap);
+ DPRINT("---------- CSC ----------------\n");
+ DPRINT("CSC mode %s\n",
+ (regs->csc_mode.b.mode) ? "YUV2RGB" : "RGB2YUV");
+ DPRINT("CSC enable DVO %d,DISP %d,LVDS %d,HDMI %d\n",
+ regs->yuv2rgb.b.dvo, regs->yuv2rgb.b.disp,
+ regs->yuv2rgb.b.lvds, regs->yuv2rgb.b.hdmi);
+ DPRINT("---------- Misc ----------------\n");
+ DPRINT("Contrast 0x%x,Brightness 0x%x\n",
+ regs->contrast.val, regs->brightness);
+ DPRINT("LVDS RGB mode %d,%s\n", igs_mode[regs->igs_mode2.b.mode],
+ (regs->igs_mode2.b.ldi) ? "msb" : "lsb");
+ DPRINT("HDMI 3D mode %d,blank 0x%x\n",
+ regs->hdmi_3d.b.mode, regs->hdmi_3d.b.blank_value);
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ DPRINT("---------- CURSOR -------------\n");
+ DPRINT("enable %d,field %d\n",
+ regs->cur_status.b.enable, regs->cur_status.b.out_field);
+ DPRINT("width %d,fb width %d\n", regs->cur_width, regs->cur_fb_width);
+ DPRINT("crop(%d,%d)\n", regs->cur_hcrop, regs->cur_vcrop);
+ DPRINT("coord H(%d,%d),V(%d,%d)\n",
+ regs->cur_hcoord.b.start, regs->cur_hcoord.b.end,
+ regs->cur_vcoord.b.start, regs->cur_vcoord.b.end);
+ DPRINT("color key enable 0x%x,color key 0x%x,alpha enable %d\n",
+ regs->cur_color_key.b.enable, regs->cur_color_key.b.colkey,
+ regs->cur_color_key.b.alpha);
+#endif
+}
+
+
+void govrh_set_tg_enable(govrh_mod_t *base, vpp_flag_t enable)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, enable);
+
+ switch (enable) {
+ case VPP_FLAG_ENABLE:
+ regs->tg_enable.b.enable = enable;
+ break;
+ case VPP_FLAG_DISABLE:
+ if (regs->tg_enable.b.enable)
+ regs->tg_enable.b.enable = enable;
+ break;
+ default:
+ break;
+ }
+ return;
+}
+
+unsigned int govrh_set_clock(govrh_mod_t *base, unsigned int pixel_clock)
+{
+ int pmc_clk = 0;
+
+ DBG_MSG("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, pixel_clock);
+
+ base->vo_clock = pixel_clock;
+ if (base->mod == VPP_MOD_GOVRH) {
+ pmc_clk = auto_pll_divisor(DEV_HDMILVDS, SET_PLLDIV,
+ 0, pixel_clock);
+ DBG_MSG("set %d,get %d\n", pixel_clock,
+ auto_pll_divisor(DEV_HDMILVDS, GET_FREQ, 0, 0));
+ g_vpp.hdmi_pixel_clock = pmc_clk;
+ } else
+ pmc_clk = auto_pll_divisor(DEV_DVO, SET_PLLDIV, 0, pixel_clock);
+ return 0;
+}
+
+void govrh_set_tg1(govrh_mod_t *base, vpp_clock_t *timing)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ regs->tg_enable.b.mode = 0;
+ regs->dstfmt = 0;
+ regs->read_cyc = timing->read_cycle;
+
+ regs->actpx_bg = timing->begin_pixel_of_active;
+ regs->actpx_end = timing->end_pixel_of_active;
+ regs->h_allpxl = timing->total_pixel_of_line;
+
+ regs->actln_bg = timing->begin_line_of_active + 1;
+ regs->actln_end = timing->end_line_of_active + 1;
+ regs->v_allln = timing->total_line_of_frame;
+
+ regs->vbie_line = timing->line_number_between_VBIS_VBIE;
+ /* pre vbi should more 6 to avoid garbage line */
+ regs->pvbi_line = (timing->line_number_between_PVBI_VBIS < 6) ?
+ 6 : timing->line_number_between_PVBI_VBIS;
+
+ regs->hdmi_hsynw = timing->hsync;
+ regs->hdmi_vbisw = timing->vsync;
+
+#ifdef DEBUG_DETAIL
+ vpp_show_timing("govrh tg1", 0, timing);
+#endif
+}
+
+int govrh_get_tg_mode(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ return regs->tg_enable.b.mode;
+}
+
+void govrh_set_tg2(govrh_mod_t *base, vpp_clock_t *timing)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ regs->tg_enable.b.mode = 1;
+ regs->dstfmt = 1;
+
+ regs->actpx_bg2 = timing->begin_pixel_of_active;
+ regs->actpx_end2 = timing->end_pixel_of_active;
+ regs->h_allpxl2 = timing->total_pixel_of_line;
+
+ regs->actln_bg2 = timing->begin_line_of_active + 1;
+ regs->actln_end2 = timing->end_line_of_active + 1;
+ regs->v_allln2 = timing->total_line_of_frame;
+
+ regs->vbie_line2 = timing->line_number_between_VBIS_VBIE;
+ regs->pvbi_line2 = (timing->line_number_between_PVBI_VBIS < 6) ?
+ 6 : timing->line_number_between_PVBI_VBIS;
+
+ regs->hdmi_hsynw2 = timing->hsync;
+ regs->hdmi_vbisw2 = timing->vsync;
+
+#ifdef DEBUG_DETAIL
+ vpp_show_timing("govrh tg2", 0, timing);
+#endif
+}
+
+void govrh_get_tg(govrh_mod_t *base, vpp_clock_t *tmr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ tmr->read_cycle = regs->read_cyc;
+ tmr->total_pixel_of_line = regs->h_allpxl;
+ tmr->begin_pixel_of_active = regs->actpx_bg;
+ tmr->end_pixel_of_active = regs->actpx_end;
+
+ tmr->total_line_of_frame = regs->v_allln;
+ tmr->begin_line_of_active = (regs->actln_bg - 1);
+ tmr->end_line_of_active = (regs->actln_end - 1);
+
+ tmr->line_number_between_VBIS_VBIE = regs->vbie_line;
+ tmr->line_number_between_PVBI_VBIS = regs->pvbi_line;
+
+ tmr->hsync = regs->hdmi_hsynw;
+ tmr->vsync = regs->hdmi_vbisw;
+
+ if (regs->tg_enable.b.mode) {
+ tmr->vsync = (tmr->vsync - 1) * 2;
+ tmr->begin_line_of_active = (tmr->begin_line_of_active - 1) * 2;
+ tmr->end_line_of_active = (tmr->end_line_of_active - 1) * 2;
+ tmr->total_line_of_frame = (tmr->total_line_of_frame - 1) * 2;
+ }
+}
+
+int govrh_get_hscale_up(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ int reg;
+
+ reg = regs->hscale_up & 0x1;
+ DBG_DETAIL("(govr %d,reg %d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, reg);
+ return reg;
+}
+
+void govrh_set_direct_path(govrh_mod_t *base, int enable)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, enable);
+
+ regs->dirpath = enable;
+}
+
+vpp_int_err_t govrh_get_int_status(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ vpp_int_err_t int_sts;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ int_sts = 0;
+ if (regs->interrupt.b.err_sts)
+ int_sts |= VPP_INT_ERR_GOVRH_MIF;
+ return int_sts;
+}
+
+void govrh_clean_int_status(govrh_mod_t *base, vpp_int_err_t int_sts)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ unsigned int val;
+
+ DBG_DETAIL("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, int_sts);
+
+ if (int_sts & VPP_INT_ERR_GOVRH_MIF) {
+ val = regs->interrupt.val;
+ val = (val & 0xff) + 0x20000;
+ regs->interrupt.val = val;
+ }
+}
+
+void govrh_set_int_enable(govrh_mod_t *base,
+ vpp_flag_t enable, vpp_int_err_t int_bit)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, enable);
+
+ /* clean status first before enable/disable interrupt */
+ govrh_clean_int_status(base, int_bit);
+
+ if (int_bit & VPP_INT_ERR_GOVRH_MIF)
+ regs->interrupt.b.mem_enable = enable;
+}
+
+void govrh_set_dvo_enable(govrh_mod_t *base, vpp_flag_t enable)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+/* DPRINT("[GOVRH] %s(%d)\n",__FUNCTION__,enable); */
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, enable);
+
+ regs->dvo_set.b.enable = enable;
+#if 1
+ if (enable) {
+ /* GPIO to function pin */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x44, 0x0);
+ /* GPIO pull disable */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x484, 0x0);
+ /* 1:pullup,0:pulldn */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x4C4, 0x0);
+ return;
+ }
+ /* disable dvo */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x44, 0xFFFFFFFF); /* Enable GPIO */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x84, 0x0); /* GPIO output enable */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x484, 0x0); /* GPIO pull disable */
+ vppif_reg32_out(GPIO_BASE_ADDR + 0x4C4, 0x0); /* 1:pullup,0:pulldn */
+#endif
+}
+
+void govrh_set_dvo_sync_polar(govrh_mod_t *base,
+ vpp_flag_t hsync, vpp_flag_t vsync)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, hsync, vsync);
+
+ regs->dvo_set.b.hsync_polar = hsync;
+ regs->dvo_set.b.vsync_polar = vsync;
+}
+
+vdo_color_fmt govrh_get_dvo_color_format(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ if (regs->dvo_pix.b.rgb)
+ return VDO_COL_FMT_ARGB;
+ if (regs->dvo_pix.b.yuv422)
+ return VDO_COL_FMT_YUV422H;
+ return VDO_COL_FMT_YUV444;
+}
+
+void govrh_set_dvo_color_format(govrh_mod_t *base, vdo_color_fmt fmt)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%s)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, vpp_colfmt_str[fmt]);
+
+ switch (fmt) {
+ case VDO_COL_FMT_ARGB:
+ regs->dvo_pix.b.rgb = 1;
+ regs->dvo_pix.b.yuv422 = 0;
+ break;
+ case VDO_COL_FMT_YUV422H:
+ regs->dvo_pix.b.rgb = 0;
+ regs->dvo_pix.b.yuv422 = 1;
+ break;
+ case VDO_COL_FMT_YUV444:
+ default:
+ regs->dvo_pix.b.rgb = 0;
+ regs->dvo_pix.b.yuv422 = 0;
+ break;
+ }
+}
+
+void govrh_set_dvo_outdatw(govrh_mod_t *base, vpp_datawidht_t width)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ unsigned int clk_delay;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, width);
+
+ regs->dvo_set.b.outwidth = (width == VPP_DATAWIDHT_12) ? 1 : 0;
+
+ clk_delay = (width == VPP_DATAWIDHT_24) ?
+ VPP_GOVR_DVO_DELAY_24 : VPP_GOVR_DVO_DELAY_12;
+ govrh_set_dvo_clock_delay(base, ((clk_delay & BIT14) != 0x0),
+ clk_delay & 0x3FFF);
+}
+
+void govrh_set_dvo_clock_delay(govrh_mod_t *base,
+ int inverse, int delay)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d,%d)\n", (base->mod == VPP_MOD_GOVRH) ?
+ 1 : 2, inverse, delay);
+
+#ifdef CONFIG_HW_DVO_DELAY
+ regs = p_govrh->mmio;
+#endif
+
+ regs->dvo_dly_sel.b.inv = inverse;
+ regs->dvo_dly_sel.b.delay = delay;
+}
+
+void govrh_set_colorbar(govrh_mod_t *base,
+ vpp_flag_t enable, int mode, int inv)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, enable);
+
+ regs->cb_enable.b.enable = enable;
+ regs->cb_enable.b.mode = mode;
+ regs->cb_enable.b.inversion = inv;
+}
+
+void govrh_set_contrast(govrh_mod_t *base, unsigned int value)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, value);
+
+ regs->contrast.val = value;
+}
+
+unsigned int govrh_get_contrast(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ return regs->contrast.val;
+}
+
+void govrh_set_brightness(govrh_mod_t *base, unsigned int value)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, value);
+
+ regs->brightness = value;
+}
+
+unsigned int govrh_get_brightness(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ return regs->brightness;
+}
+
+void govrh_set_saturation(govrh_mod_t *base, unsigned int value)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, value);
+
+ regs->saturation.val = value;
+ regs->saturation_enable.b.enable = 1;
+}
+
+unsigned int govrh_get_saturation(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ return regs->saturation.val;
+}
+
+void govrh_set_MIF_enable(govrh_mod_t *base, vpp_flag_t enable)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, enable);
+
+ regs->mif.b.enable = enable;
+}
+
+int govrh_get_MIF_enable(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ return regs->mif.b.enable;
+}
+
+void govrh_set_frame_mode(govrh_mod_t *base,
+ unsigned int width, vdo_color_fmt colfmt)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ int y_byte, c_byte;
+ int enable;
+
+ enable = 1;
+ vpp_get_colfmt_bpp(colfmt, &y_byte, &c_byte);
+ y_byte = width * y_byte / 8;
+ if (y_byte % 8)
+ enable = 0;
+ if (c_byte) {
+ c_byte = width * c_byte / 8;
+ if (c_byte % 8)
+ enable = 0;
+ }
+ if (regs->bufwid % 128)
+ enable = 0;
+ if ((width == 720) && (colfmt == VDO_COL_FMT_RGB_565))
+ enable = 0;
+ regs->mif_frame_mode.b.frame_enable = enable;
+ regs->mif_frame_mode.b.req_num =
+ (y_byte / 128) + ((y_byte % 128) ? 1 : 0);
+ regs->mif_frame_mode.b.req_num_c =
+ (c_byte) ? ((c_byte / 128) + ((c_byte % 128) ? 1 : 0)) : 0;
+}
+
+void govrh_set_color_format(govrh_mod_t *base, vdo_color_fmt format)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%s)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, vpp_colfmt_str[format]);
+
+ if (format < VDO_COL_FMT_ARGB)
+ regs->yuv2rgb.b.rgb_mode = 0;
+
+ switch (format) {
+ case VDO_COL_FMT_YUV420:
+ regs->colfmt = 1;
+ regs->colfmt2 = 0;
+ break;
+ case VDO_COL_FMT_YUV422H:
+ regs->colfmt = 0;
+ regs->colfmt2 = 0;
+ break;
+ case VDO_COL_FMT_YUV444:
+ regs->colfmt = 0;
+ regs->colfmt2 = 1;
+ break;
+ case VDO_COL_FMT_ARGB:
+ regs->yuv2rgb.b.rgb_mode = 1;
+ break;
+ case VDO_COL_FMT_RGB_1555:
+ regs->yuv2rgb.b.rgb_mode = 2;
+ break;
+ case VDO_COL_FMT_RGB_565:
+ regs->yuv2rgb.b.rgb_mode = 3;
+ break;
+ default:
+ DBGMSG("*E* check the parameter\n");
+ return;
+ }
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ govrh_CUR_set_colfmt((govrh_cursor_mod_t *)p_cursor, format);
+#endif
+ govrh_set_frame_mode(base, regs->pixwid, format);
+ regs->saturation_enable.b.format = (regs->yuv2rgb.b.rgb_mode) ? 1 : 0;
+}
+
+vdo_color_fmt govrh_get_color_format(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ switch (regs->yuv2rgb.b.rgb_mode) {
+ case 1:
+ return VDO_COL_FMT_ARGB;
+ case 2:
+ return VDO_COL_FMT_RGB_1555;
+ case 3:
+ return VDO_COL_FMT_RGB_565;
+ default:
+ break;
+ }
+ if (regs->colfmt2)
+ return VDO_COL_FMT_YUV444;
+ if (regs->colfmt)
+ return VDO_COL_FMT_YUV420;
+ return VDO_COL_FMT_YUV422H;
+}
+
+void govrh_set_source_format(govrh_mod_t *base,
+ vpp_display_format_t format)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, format);
+
+ regs->srcfmt = (format == VPP_DISP_FMT_FIELD) ? 1 : 0;
+}
+
+void govrh_set_output_format(govrh_mod_t *base,
+ vpp_display_format_t field)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, field);
+
+ regs->dstfmt = (field == VPP_DISP_FMT_FIELD) ? 1 : 0;
+}
+
+void govrh_set_fb_addr(govrh_mod_t *base,
+ unsigned int y_addr, unsigned int c_addr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ /* DBG_DETAIL("(govr %d,0x%x,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH)? 1:2,y_addr,c_addr); */
+#ifdef DEBUG
+ if ((y_addr % GOVRH_FRAMEBUF_ALIGN) || (c_addr % GOVRH_FRAMEBUF_ALIGN))
+ DBG_ERR("addr should align %d(0x%x,0x%x)\n",
+ GOVRH_FRAMEBUF_ALIGN, y_addr, c_addr);
+#endif
+ regs->ysa = y_addr;
+ regs->csa = c_addr;
+}
+
+void govrh_get_fb_addr(govrh_mod_t *base,
+ unsigned int *y_addr, unsigned int *c_addr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ *y_addr = regs->ysa;
+ *c_addr = regs->csa;
+
+ DBG_DETAIL("(govr %d,0x%x,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, *y_addr, *c_addr);
+}
+
+void govrh_set_fb2_addr(govrh_mod_t *base,
+ unsigned int y_addr, unsigned int c_addr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,0x%x,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, y_addr, c_addr);
+#ifdef DEBUG
+ if ((y_addr % GOVRH_FRAMEBUF_ALIGN) || (c_addr % GOVRH_FRAMEBUF_ALIGN))
+ DBG_ERR("addr should align %d(0x%x,0x%x)\n",
+ GOVRH_FRAMEBUF_ALIGN, y_addr, c_addr);
+#endif
+ regs->ysa2 = y_addr;
+ regs->csa2 = c_addr;
+}
+
+void govrh_get_fb2_addr(govrh_mod_t *base,
+ unsigned int *y_addr, unsigned int *c_addr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ *y_addr = regs->ysa2;
+ *c_addr = regs->csa2;
+
+ DBG_DETAIL("(govr %d,0x%x,0x%x)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, *y_addr, *c_addr);
+}
+
+void govrh_set_fb_width(govrh_mod_t *base, unsigned int width)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, width);
+
+ regs->bufwid = width;
+}
+
+#define GOVRH_CURSOR_RIGHT_POS 6
+void govrh_set_fb_info(govrh_mod_t *base, unsigned int width,
+ unsigned int act_width, unsigned int x_offset, unsigned int y_offset)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,fb_w %d,img_w %d,x %d,y %d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2,
+ width, act_width, x_offset, y_offset);
+
+ regs->pixwid = act_width;
+ regs->bufwid = width;
+ regs->hcrop = x_offset;
+ regs->vcrop = y_offset;
+ govrh_set_frame_mode(base, act_width, govrh_get_color_format(base));
+
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ /* cursor postion over when change resolution */
+ if ((regs->actpx_end - regs->actpx_bg) != regs->pixwid) {
+ if (regs->cur_hcoord.b.end >=
+ (regs->pixwid - GOVRH_CURSOR_RIGHT_POS)) {
+ int pos;
+
+ pos = regs->pixwid - GOVRH_CURSOR_RIGHT_POS;
+ regs->cur_hcoord.b.end = pos;
+ pos -= regs->cur_width;
+ regs->cur_hcoord.b.start = (pos+1);
+
+ p_cursor->posx = pos;
+ }
+ }
+#endif
+}
+
+void govrh_get_fb_info(govrh_mod_t *base, unsigned int *width,
+ unsigned int *act_width, unsigned int *x_offset, unsigned int *y_offset)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ *act_width = regs->pixwid;
+ *width = regs->bufwid;
+ *x_offset = regs->hcrop;
+ *y_offset = regs->vcrop;
+}
+
+void govrh_set_fifo_index(govrh_mod_t *base, unsigned int index)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, index);
+
+ regs->fhi = index;
+}
+
+void govrh_set_reg_level(govrh_mod_t *base, vpp_reglevel_t level)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, level);
+
+ regs->sts.b.level = (level == VPP_REG_LEVEL_1) ? 0 : 1;
+}
+
+void govrh_set_reg_update(govrh_mod_t *base, vpp_flag_t enable)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, enable);
+
+ regs->sts.b.update = enable;
+}
+
+void govrh_set_tg(govrh_mod_t *base,
+ vpp_clock_t *tmr, unsigned int pixel_clock)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ int tg_enable;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, pixel_clock);
+
+ tg_enable = regs->tg_enable.b.enable;
+ regs->tg_enable.b.enable = 0;
+
+ tmr->read_cycle = govrh_set_clock(base, pixel_clock);
+ govrh_set_tg1(base, tmr);
+
+ regs->tg_enable.b.enable = tg_enable;
+}
+
+void govrh_set_csc_mode(govrh_mod_t *base, vpp_csc_t mode)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ vdo_color_fmt src_fmt, dst_fmt;
+ unsigned int enable;
+ unsigned int csc_parm5;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, mode);
+
+ enable = 0;
+ src_fmt = (govrh_get_color_format(base) < VDO_COL_FMT_ARGB) ?
+ VDO_COL_FMT_YUV444 : VDO_COL_FMT_ARGB;
+ dst_fmt = (govrh_get_dvo_color_format(base) < VDO_COL_FMT_ARGB) ?
+ VDO_COL_FMT_YUV444 : VDO_COL_FMT_ARGB;
+ if (src_fmt == VDO_COL_FMT_ARGB)
+ enable |= (dst_fmt != VDO_COL_FMT_ARGB) ? BIT0 : 0x00; /* DVO */
+ else
+ enable |= (dst_fmt == VDO_COL_FMT_ARGB) ? BIT0 : 0x00; /* DVO */
+#ifdef WMT_FTBLK_LVDS
+ vpp_set_clock_enable(DEV_LVDS, 1, 0);
+ if (src_fmt == VDO_COL_FMT_ARGB) {
+ if (lvds_get_colfmt() < VDO_COL_FMT_ARGB)
+ enable |= BIT3;
+ } else {
+ if (lvds_get_colfmt() >= VDO_COL_FMT_ARGB)
+ enable |= BIT3;
+ }
+ vpp_set_clock_enable(DEV_LVDS, 0, 0);
+#endif
+#ifdef WMT_FTBLK_HDMI
+ vpp_set_clock_enable(DEV_HDMI, 1, 0);
+ if (src_fmt == VDO_COL_FMT_ARGB) {
+ if (hdmi_get_output_colfmt() < VDO_COL_FMT_ARGB)
+ enable |= BIT4;
+ } else {
+ if (hdmi_get_output_colfmt() >= VDO_COL_FMT_ARGB)
+ enable |= BIT4;
+ }
+ vpp_set_clock_enable(DEV_HDMI, 0, 0);
+#endif
+ mode = vpp_check_csc_mode(mode, src_fmt, dst_fmt, enable);
+ if (mode >= VPP_CSC_MAX) {
+ enable = 0;
+ /* for internal color bar (YUV base) */
+ mode = VPP_CSC_YUV2RGB_JFIF_0_255;
+ }
+ /* HDMI YUV mode too light 0xff00ff -> 0x01010001 */
+ csc_parm5 = (mode == VPP_CSC_RGB2YUV_JFIF_0_255) ?
+ 0x01010001 : vpp_csc_parm[mode][5];
+
+ regs->dmacsc_coef0 = vpp_csc_parm[mode][0]; /* C1,C2 */
+ regs->dmacsc_coef1 = vpp_csc_parm[mode][1]; /* C3,C4 */
+ regs->dmacsc_coef2 = vpp_csc_parm[mode][2]; /* C5,C6 */
+ regs->dmacsc_coef3 = vpp_csc_parm[mode][3]; /* C7,C8 */
+ regs->dmacsc_coef4 = vpp_csc_parm[mode][4] & 0xFFFF; /* C9 */
+ regs->dmacsc_coef5 = ((vpp_csc_parm[mode][4] & 0xFFFF0000) >> 16) +
+ ((csc_parm5 & 0xFFFF) << 16); /* I,J */
+ regs->dmacsc_coef6 = (csc_parm5 & 0xFFFF0000) >> 16; /* K */
+ regs->csc_mode.val = (vpp_csc_parm[mode][6] & BIT0) +
+ ((vpp_csc_parm[mode][6] & BIT8) ? BIT1 : 0x0);
+
+ /* enable bit0:DVO, bit1:VGA, bit2:DISP, bit3:LVDS, bit4:HDMI */
+ regs->yuv2rgb.val = (regs->yuv2rgb.val & ~0xFF) +
+ ((mode >= VPP_CSC_RGB2YUV_SDTV_0_255) ? 0x1C : 0x18);
+ regs->yuv2rgb.b.dvo = (enable & BIT0) ? 1 : 0;
+#ifdef WMT_FTBLK_LVDS
+ regs->yuv2rgb.b.lvds = (enable & BIT3) ? 1 : 0;
+#endif
+#ifdef WMT_FTBLK_HDMI
+ regs->yuv2rgb.b.hdmi = (enable & BIT4) ? 1 : 0;
+#endif
+}
+
+void govrh_set_media_format(govrh_mod_t *base, vpp_flag_t h264)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, h264);
+
+ regs->mif.b.h264 = h264;
+}
+
+void govrh_HDMI_set_blank_value(govrh_mod_t *base, unsigned int val)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2, val);
+
+ regs->hdmi_3d.b.blank_value = val;
+}
+
+void govrh_HDMI_set_3D_mode(govrh_mod_t *base, int mode)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, mode);
+
+ /* 0-disable,3-frame pack progress(use fb1&2),7-frame pack interlace */
+ regs->hdmi_3d.b.mode = mode;
+}
+
+int govrh_is_top_field(govrh_mod_t *base)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ return (regs->field_status) ? 1 : 0;
+}
+
+/*----------------------- GOVRH IGS --------------------------------------*/
+void govrh_IGS_set_mode(govrh_mod_t *base,
+ int no, int mode_18bit, int msb)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2, no);
+
+ if (no == 0) { /* DVO */
+ /* 0-24bit,10-18bit,01-555,11-565 */
+ regs->igs_mode.b.mode = mode_18bit;
+ regs->igs_mode.b.ldi = msb; /* 0-lsb,1-msb */
+ } else { /* LVDS */
+ /* 0-24bit,10-18bit,01-555,11-565 */
+ regs->igs_mode2.b.mode = mode_18bit;
+ regs->igs_mode2.b.ldi = msb; /* 0-lsb,1-msb */
+ }
+}
+
+void govrh_IGS_set_RGB_swap(govrh_mod_t *base, int mode)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+
+ DBG_DETAIL("(govr %d,%d)\n",
+ (base->mod == VPP_MOD_GOVRH) ? 1 : 2, mode);
+
+ /* 0-RGB [7-0], 1-RGB [0-7], 2-BGR [7-0], 3-BGR [0-7] */
+ regs->dvo_set.b.rgb_swap = mode;
+}
+
+/*----------------------- GOVRH CURSOR --------------------------------------*/
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+#define CONFIG_GOVR2_CURSOR
+void *govrh_CUR_get_mmio(govrh_cursor_mod_t *base)
+{
+#ifdef CONFIG_GOVR2_CURSOR
+ base = (govrh_cursor_mod_t *)
+ ((govrh_get_MIF_enable(p_govrh)) ? p_govrh : p_govrh2);
+#endif
+ return base->mmio;
+}
+
+void govrh_CUR_set_enable(govrh_cursor_mod_t *base, vpp_flag_t enable)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ DBG_DETAIL("(0x%x,%d)\n", base->mmio, enable);
+
+ regs->cur_status.b.enable = enable;
+}
+
+void govrh_CUR_set_fb_addr(govrh_cursor_mod_t *base,
+ unsigned int y_addr, unsigned int c_addr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ DBG_DETAIL("(0x%x,0x%x,0x%x)\n", base->mmio, y_addr, c_addr);
+
+ regs->cur_addr = y_addr;
+}
+
+void govrh_CUR_get_fb_addr(govrh_cursor_mod_t *base,
+ unsigned int *y_addr, unsigned int *c_addr)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ *y_addr = regs->cur_addr;
+ *c_addr = 0;
+
+ DBG_DETAIL("(0x%x,0x%x,0x%x)\n", base->mmio, *y_addr, *c_addr);
+}
+
+void govrh_CUR_set_coordinate(govrh_cursor_mod_t *base,
+ unsigned int x1, unsigned int y1, unsigned int x2, unsigned int y2)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ DBG_DETAIL("(0x%x)\n", base->mmio);
+
+ regs->cur_hcoord.b.start = x1;
+ regs->cur_hcoord.b.end = x2;
+ regs->cur_vcoord.b.start = y1;
+ regs->cur_vcoord.b.end = y2;
+}
+
+void govrh_CUR_set_position(govrh_cursor_mod_t *base,
+ unsigned int x, unsigned int y)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+ unsigned int w, h;
+
+ DBG_DETAIL("(0x%x,%d,%d)\n", base->mmio, x, y);
+
+/* w = regs->cur_hcoord.b.end - regs->cur_hcoord.b.start; */
+ w = regs->cur_width - 1;
+ h = regs->cur_vcoord.b.end - regs->cur_vcoord.b.start;
+ govrh_CUR_set_coordinate(base, x, y, x+w, y+h);
+}
+
+void govrh_CUR_set_color_key_mode(govrh_cursor_mod_t *base,
+ int enable, int alpha, int mode)
+{
+ /*
+ if( colkey_en )
+ colkey_hit = (mode)? (data == colkey):(data != colkey);
+
+ if( (alpha_en) or (colkey_hit) or (alpha value) )
+ show pixel
+ */
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ DBG_DETAIL("(0x%x,%d,%d,%d)\n", base->mmio, enable, alpha, mode);
+
+ regs->cur_color_key.b.alpha = alpha;
+ regs->cur_color_key.b.enable = enable;
+ regs->cur_color_key.b.invert = mode;
+}
+
+void govrh_CUR_set_color_key(govrh_cursor_mod_t *base,
+ int enable, int alpha, unsigned int colkey)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ DBG_DETAIL("(0x%x,0x%x)\n", base->mmio, colkey);
+
+ regs->cur_color_key.b.colkey = colkey;
+ regs->cur_color_key.b.alpha = 0;
+ regs->cur_color_key.b.enable = 0;
+
+#ifdef __KERNEL__
+ if (!(alpha) && enable) {
+ int i, j;
+ unsigned int *ptr1, *ptr2;
+ int yuv2rgb;
+
+ ptr1 = (unsigned int *)mb_phys_to_virt(regs->cur_addr);
+ for (i = 0; i < base->fb_p->fb.fb_h; i++) {
+ for (j = 0; j < base->fb_p->fb.fb_w; j++) {
+ if ((*ptr1 & 0xFFFFFF) == (colkey & 0xFFFFFF))
+ *ptr1 &= 0xFFFFFF;
+ else
+ *ptr1 |= 0xFF000000;
+ ptr1++;
+ }
+ }
+
+ yuv2rgb = (base->colfmt < VDO_COL_FMT_ARGB) ? 1 : 0;
+ ptr1 = (unsigned int *) mb_phys_to_virt(base->cursor_addr1);
+ ptr2 = (unsigned int *) mb_phys_to_virt(base->cursor_addr2);
+ for (i = 0; i < base->fb_p->fb.fb_h; i++) {
+ for (j = 0; j < base->fb_p->fb.fb_w; j++) {
+ *ptr2 = vpp_convert_colfmt(yuv2rgb, *ptr1);
+ ptr1++;
+ ptr2++;
+ }
+ }
+ }
+#endif
+}
+
+void govrh_CUR_set_colfmt(govrh_cursor_mod_t *base,
+ vdo_color_fmt colfmt)
+{
+#ifdef __KERNEL__
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+#endif
+
+ DBG_DETAIL("(0x%x,%s)\n", base->mmio, vpp_colfmt_str[colfmt]);
+
+ colfmt = (colfmt < VDO_COL_FMT_ARGB) ?
+ VDO_COL_FMT_YUV444 : VDO_COL_FMT_ARGB;
+ if (base->fb_p->fb.col_fmt == colfmt)
+ return;
+
+ base->fb_p->fb.col_fmt = colfmt;
+#ifdef __KERNEL__
+ regs->cur_addr = (base->colfmt == colfmt) ?
+ base->cursor_addr1 : base->cursor_addr2;
+#endif
+ DBG_DETAIL("(0x%x)\n", regs->cur_addr);
+}
+
+int govrh_irqproc_set_position(void *arg)
+{
+ govrh_cursor_mod_t *base = (govrh_cursor_mod_t *) arg;
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ int begin;
+ int pos_x, pos_y;
+ int crop_x, crop_y;
+ int width, height;
+ int govrh_w, govrh_h;
+
+ govrh_w = regs->actpx_end - regs->actpx_bg;
+ govrh_h = regs->actln_end - regs->actln_bg;
+ if (regs->tg_enable.b.mode)
+ govrh_h *= 2;
+
+ /* cursor H */
+ width = base->fb_p->fb.img_w;
+ begin = base->posx - base->hotspot_x;
+ if (begin < 0) { /* over left edge */
+ pos_x = 0;
+ crop_x = 0 - begin;
+ width = width - crop_x;
+ } else {
+ pos_x = begin;
+ crop_x = 0;
+ }
+ if ((begin + base->fb_p->fb.img_w) > govrh_w) /* over right edge */
+ width = govrh_w - begin;
+
+/* DPRINT("[CURSOR] H pos %d,hotspot %d,posx %d,crop %d,width %d\n",
+ p_cursor->posx,p_cursor->hotspot_x,pos_x,crop_x,width); */
+
+ /* cursor V */
+ height = base->fb_p->fb.img_h;
+ begin = base->posy - base->hotspot_y;
+ if (begin < 0) { /* over top edge */
+ pos_y = 0;
+ crop_y = 0 - begin;
+ height = height - crop_y;
+ } else {
+ pos_y = begin;
+ crop_y = 0;
+ }
+ if ((begin + base->fb_p->fb.img_h) > govrh_h) /* over right edge */
+ height = govrh_h - begin;
+
+ if (regs->tg_enable.b.mode) {
+ regs->cur_fb_width = (base->fb_p->fb.fb_w*2);
+ pos_y /= 2;
+ crop_y /= 2;
+ height /= 2;
+ } else
+ regs->cur_fb_width = base->fb_p->fb.fb_w;
+
+/* DPRINT("[CURSOR] V pos %d,hotspot %d,posx %d,crop %d,height %d\n",
+ p_cursor->posy,p_cursor->hotspot_y,pos_y,crop_y,height); */
+
+ regs->cur_width = width;
+ regs->cur_vcrop = crop_y;
+ regs->cur_hcrop = crop_x;
+ govrh_CUR_set_coordinate(base, pos_x, pos_y,
+ pos_x + width - 1, pos_y + height - 1);
+ return 0;
+}
+
+void govrh_CUR_irqproc(int arg)
+{
+ if (p_cursor->enable) {
+ if (p_cursor->chg_flag) {
+ p_cursor->hide_cnt = 0;
+ govrh_CUR_set_enable(p_cursor, 1);
+ } else {
+ p_cursor->hide_cnt++;
+ if (p_cursor->hide_cnt >
+ (GOVRH_CURSOR_HIDE_TIME * p_govrh->fb_p->framerate))
+ govrh_CUR_set_enable(p_cursor, 0);
+ }
+ }
+
+ if (p_cursor->chg_flag) {
+ govrh_irqproc_set_position(p_cursor);
+ p_cursor->chg_flag = 0;
+ }
+}
+
+void govrh_CUR_proc_view(govrh_cursor_mod_t *base,
+ int read, vdo_view_t *view)
+{
+ vdo_framebuf_t *fb;
+
+ DBG_DETAIL("(0x%x)\n", base->mmio);
+
+ fb = &base->fb_p->fb;
+ if (read) {
+ view->resx_src = fb->fb_w;
+ view->resy_src = fb->fb_h;
+ view->resx_virtual = fb->img_w;
+ view->resy_virtual = fb->img_h;
+ view->resx_visual = fb->img_w;
+ view->resy_visual = fb->img_h;
+ view->posx = base->posx;
+ view->posy = base->posy;
+ view->offsetx = fb->h_crop;
+ view->offsety = fb->v_crop;
+ } else {
+ base->posx = view->posx;
+ base->posy = view->posy;
+ base->chg_flag = 1;
+ }
+}
+
+void govrh_CUR_set_framebuffer(govrh_cursor_mod_t *base,
+ vdo_framebuf_t *fb)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) govrh_CUR_get_mmio(base);
+
+ DBG_DETAIL("(0x%x)\n", base->mmio);
+
+ regs->cur_addr = fb->y_addr;
+ regs->cur_width = fb->img_w;
+ if (regs->tg_enable.b.mode)
+ regs->cur_fb_width = (fb->fb_w*2);
+ else
+ regs->cur_fb_width = fb->fb_w;
+ regs->cur_vcrop = fb->v_crop;
+ regs->cur_hcrop = fb->h_crop;
+ regs->cur_status.b.out_field = vpp_get_fb_field(fb);
+ govrh_irqproc_set_position(base);
+ base->colfmt = fb->col_fmt;
+ base->cursor_addr1 = fb->y_addr;
+#ifdef __KERNEL__
+ if (base->cursor_addr2 == 0)
+ base->cursor_addr2 = mb_alloc(64*64*4);
+#endif
+ vpp_cache_sync();
+}
+
+void govrh_CUR_init(void *base)
+{
+#ifdef CONFIG_GOVR2_CURSOR
+ volatile struct govrh_regs *regs;
+
+ regs = (volatile struct govrh_regs *) p_govrh->mmio;
+ regs->cur_color_key.b.invert = 1;
+ regs = (volatile struct govrh_regs *) p_govrh2->mmio;
+ regs->cur_color_key.b.invert = 1;
+#else
+ govrh_cursor_mod_t *mod_p;
+ volatile struct govrh_regs *regs;
+
+ mod_p = (govrh_cursor_mod_t *) base;
+ regs = (volatile struct govrh_regs *) mod_p->mmio;
+
+ regs->cur_color_key.b.invert = 1;
+#endif
+}
+#endif /* WMT_FTBLK_GOVRH_CURSOR */
+
+/*----------------------- GOVRH API --------------------------------------*/
+void govrh_vmode_to_timing(vpp_mod_t mod,
+ struct fb_videomode *vmode, vpp_clock_t *t)
+{
+ unsigned int pixel_clock;
+ int temp;
+
+ t->begin_pixel_of_active = vmode->hsync_len + vmode->left_margin;
+ t->end_pixel_of_active = t->begin_pixel_of_active + vmode->xres;
+ t->total_pixel_of_line = t->end_pixel_of_active + vmode->right_margin;
+ t->begin_line_of_active = vmode->vsync_len + vmode->upper_margin;
+ t->end_line_of_active = t->begin_line_of_active + vmode->yres;
+ t->total_line_of_frame = t->end_line_of_active + vmode->lower_margin;
+ t->line_number_between_VBIS_VBIE = vmode->vsync_len + 1;
+ temp = t->total_line_of_frame - t->end_line_of_active;
+ t->line_number_between_PVBI_VBIS = (temp > 2) ? (temp-1) : 1;
+ t->hsync = vmode->hsync_len;
+ t->vsync = vmode->vsync_len;
+
+ pixel_clock = PICOS2KHZ(vmode->pixclock) * 1000;
+ t->read_cycle = vpp_get_base_clock(mod) / pixel_clock;
+}
+
+void govrh_set_videomode(govrh_mod_t *base,
+ struct fb_videomode *p_vmode)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) base->mmio;
+ struct fb_videomode vmode;
+ vpp_clock_t t1;
+ int tg_enable;
+
+ DBG_MSG("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ tg_enable = regs->tg_enable.b.enable;
+ regs->tg_enable.b.enable = 0;
+
+ vmode = *p_vmode;
+ if (vmode.vmode & FB_VMODE_DOUBLE) {
+ vmode.xres *= 2;
+ DBGMSG("[GOVRH] H scale up\n");
+ }
+
+ if (vmode.vmode & FB_VMODE_INTERLACED) {
+ vmode.vsync_len /= 2;
+ vmode.upper_margin /= 2;
+ vmode.yres /= 2;
+ vmode.lower_margin /= 2;
+ }
+
+ govrh_vmode_to_timing(base->mod, &vmode, &t1);
+ t1.read_cycle = govrh_set_clock(base, PICOS2KHZ(vmode.pixclock)*1000);
+ if (vmode.vmode & FB_VMODE_INTERLACED) {
+ t1.total_line_of_frame += 1;
+ t1.begin_line_of_active += 1;
+ t1.end_line_of_active += 1;
+ t1.vsync += 1;
+ }
+ govrh_set_tg1(base, &t1);
+
+ if (vmode.vmode & FB_VMODE_INTERLACED) {
+ vpp_clock_t t2;
+
+ vmode.upper_margin += 1;
+ /* 480i/576i repeat 2 pixel */
+ if (vmode.vmode & FB_VMODE_DOUBLE)
+ vmode.xres *= 2;
+ govrh_vmode_to_timing(base->mod, &vmode, &t2);
+ t2.total_line_of_frame -= 1;
+ t2.vsync += 1;
+ govrh_set_tg2(base, &t2);
+ }
+ govrh_set_output_format(base,
+ (vmode.vmode & FB_VMODE_INTERLACED) ? 1 : 0);
+ regs->hscale_up = (vmode.vmode & FB_VMODE_DOUBLE) ? 1 : 0;
+ regs->vsync_offset.val = (vmode.vmode & FB_VMODE_INTERLACED) ?
+ ((regs->h_allpxl / 2) | BIT16) : 0;
+
+ regs->tg_enable.b.enable = tg_enable;
+}
+
+void govrh_get_videomode(govrh_mod_t *base, struct fb_videomode *vmode)
+{
+ volatile struct govrh_regs *regs
+ = (volatile struct govrh_regs *) base->mmio;
+ vpp_clock_t t;
+ int htotal, vtotal;
+ unsigned int pixel_clock;
+
+ vmode->flag = FB_MODE_IS_FROM_VAR;
+ vmode->name = 0;
+
+ govrh_get_tg(base, &t);
+ pixel_clock = vpp_get_base_clock(base->mod);
+ vmode->pixclock = KHZ2PICOS(pixel_clock/1000);
+
+ vmode->hsync_len = t.hsync;
+ vmode->left_margin = t.begin_pixel_of_active - t.hsync;
+ vmode->xres = t.end_pixel_of_active - t.begin_pixel_of_active;
+ vmode->right_margin = t.total_pixel_of_line - t.end_pixel_of_active;
+
+ vmode->vsync_len = t.vsync;
+ vmode->upper_margin = t.begin_line_of_active - t.vsync;
+ vmode->yres = t.end_line_of_active - t.begin_line_of_active;
+ vmode->lower_margin = t.total_line_of_frame - t.end_line_of_active;
+
+ htotal = vmode->xres + vmode->right_margin
+ + vmode->hsync_len + vmode->left_margin;
+ vtotal = vmode->yres + vmode->lower_margin
+ + vmode->vsync_len + vmode->upper_margin;
+ vmode->vmode = (regs->tg_enable.b.mode) ? FB_VMODE_INTERLACED : 0;
+ if (vmode->vmode & FB_VMODE_INTERLACED)
+ vtotal /= 2;
+ if (vmode->vmode & FB_VMODE_DOUBLE)
+ vtotal *= 2;
+
+ if (htotal && vtotal)
+ vmode->refresh = pixel_clock/(htotal * vtotal);
+ else
+ vmode->refresh = 60;
+
+ { /* get sync polar */
+ int hsync_hi, vsync_hi;
+ vout_inf_mode_t interface;
+
+ switch (base->mod) {
+ case VPP_MOD_GOVRH:
+ interface = VOUT_INF_HDMI;
+ break;
+ case VPP_MOD_GOVRH2:
+ interface = VOUT_INF_DVI;
+ if (lcd_get_type())
+ interface = VOUT_INF_LVDS;
+ break;
+ default:
+ interface = VOUT_INF_DVI;
+ break;
+ }
+
+ switch (interface) {
+ case VOUT_INF_DVI:
+ hsync_hi = (regs->dvo_set.b.hsync_polar) ? 0 : 1;
+ vsync_hi = (regs->dvo_set.b.vsync_polar) ? 0 : 1;
+ break;
+ case VOUT_INF_HDMI:
+ vo_hdmi_set_clock(1);
+ hdmi_get_sync_polar(&hsync_hi, &vsync_hi);
+ vo_hdmi_set_clock(0);
+ break;
+ case VOUT_INF_LVDS:
+ lvds_get_sync_polar(&hsync_hi, &vsync_hi);
+ break;
+ default:
+ hsync_hi = 0;
+ vsync_hi = 0;
+ break;
+ }
+ vmode->sync = (hsync_hi) ? FB_SYNC_HOR_HIGH_ACT : 0;
+ vmode->sync |= (vsync_hi) ? FB_SYNC_VERT_HIGH_ACT : 0;
+ }
+}
+
+void govrh_set_framebuffer(govrh_mod_t *base, vdo_framebuf_t *fb)
+{
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ govrh_set_fb_addr(base, fb->y_addr, fb->c_addr);
+ govrh_set_color_format(base, fb->col_fmt);
+ govrh_set_fb_info(base, fb->fb_w, fb->img_w, fb->h_crop, fb->v_crop);
+ govrh_set_source_format(base, vpp_get_fb_field(fb));
+ govrh_set_csc_mode(base, base->fb_p->csc_mode);
+}
+
+void govrh_get_framebuffer(govrh_mod_t *base, vdo_framebuf_t *fb)
+{
+ volatile struct govrh_regs *regs
+ = (volatile struct govrh_regs *) base->mmio;
+ vpp_clock_t clock;
+ int y_bpp, c_bpp;
+
+ DBG_DETAIL("(govr %d)\n", (base->mod == VPP_MOD_GOVRH) ? 1 : 2);
+
+ govrh_get_fb_addr(base, &fb->y_addr, &fb->c_addr);
+ fb->col_fmt = govrh_get_color_format(base);
+ govrh_get_fb_info(base, &fb->fb_w,
+ &fb->img_w, &fb->h_crop, &fb->v_crop);
+ govrh_get_tg(base, &clock);
+ fb->img_h = clock.end_line_of_active - clock.begin_line_of_active;
+ fb->fb_h = fb->img_h;
+ fb->flag = (regs->srcfmt) ? VDO_FLAG_INTERLACE : 0;
+ vpp_get_colfmt_bpp(fb->col_fmt, &y_bpp, &c_bpp);
+ fb->bpp = y_bpp + c_bpp;
+ fb->y_size = fb->fb_w * fb->fb_h * y_bpp / 8;
+ fb->c_size = fb->fb_w * fb->fb_h * c_bpp / 8;
+}
+
+/*----------------------- GOVRH MODULE API -----------------------------*/
+int govrh_suspend_yaddr;
+void govrh_suspend(govrh_mod_t *base, int sts)
+{
+#ifdef CONFIG_PM
+ volatile struct govrh_regs *regs
+ = (volatile struct govrh_regs *) base->mmio;
+
+ switch (sts) {
+ case 0: /* disable module */
+ base->pm_enable = govrh_get_MIF_enable(base);
+ govrh_set_MIF_enable(base, 0);
+ break;
+ case 1: /* disable tg */
+ base->pm_tg = regs->tg_enable.b.enable;
+ govrh_set_tg_enable(base, 0);
+ break;
+ case 2: /* backup register */
+ base->reg_bk = vpp_backup_reg((unsigned int)base->mmio,
+ (REG_GOVRH_BASE2_END-REG_GOVRH_BASE1_BEGIN));
+ if (base->vo_clock == 0)
+ base->vo_clock = vpp_get_base_clock(base->mod);
+ govrh_suspend_yaddr = 0;
+ break;
+ default:
+ break;
+ }
+#endif
+}
+
+void govrh_resume(govrh_mod_t *base, int sts)
+{
+#ifdef CONFIG_PM
+/* volatile struct govrh_regs *regs
+ = (volatile struct govrh_regs *) base->mmio; */
+
+ switch (sts) {
+ case 0: /* restore register */
+ govrh_set_clock(base, base->vo_clock);
+ vpp_restore_reg((unsigned int)base->mmio,
+ (REG_GOVRH_BASE2_END-REG_GOVRH_BASE1_BEGIN),
+ base->reg_bk);
+ if (base == p_govrh && govrh_suspend_yaddr)
+ govrh_set_fb_addr(p_govrh, govrh_suspend_yaddr, 0);
+ base->reg_bk = 0;
+ break;
+ case 1: /* enable module */
+ govrh_set_MIF_enable(base, base->pm_enable);
+ break;
+ case 2: /* enable tg */
+ govrh_set_tg_enable(base, base->pm_tg);
+ break;
+ default:
+ break;
+ }
+#endif
+}
+
+void govrh_init(void *base)
+{
+ govrh_mod_t *mod_p;
+ volatile struct govrh_regs *regs;
+
+ mod_p = (govrh_mod_t *) base;
+ regs = (volatile struct govrh_regs *) mod_p->mmio;
+
+ govrh_set_reg_level(base, VPP_REG_LEVEL_1);
+ govrh_set_colorbar(base, VPP_FLAG_DISABLE, 1, 0);
+ govrh_set_tg_enable(base, VPP_FLAG_ENABLE);
+ govrh_set_dvo_color_format(base, VDO_COL_FMT_ARGB);
+ govrh_set_dvo_enable(base, VPP_FLAG_DISABLE);
+
+ govrh_set_framebuffer(base, &mod_p->fb_p->fb);
+ govrh_set_fifo_index(base, 0xf);
+ govrh_set_csc_mode(base, mod_p->fb_p->csc_mode);
+
+ govrh_set_reg_update(base, VPP_FLAG_ENABLE);
+ govrh_set_tg_enable(base, VPP_FLAG_ENABLE);
+/* govrh_set_MIF_enable(base, VPP_FLAG_ENABLE); */
+ govrh_set_int_enable(base, VPP_FLAG_ENABLE, mod_p->int_catch);
+}
+
+void govrh_mod_dump_reg(void)
+{
+ govrh_reg_dump((void *)p_govrh);
+}
+
+void govrh_mod_set_enable(vpp_flag_t enable)
+{
+ govrh_set_MIF_enable(p_govrh, enable);
+}
+
+void govrh_mod_set_colorbar(vpp_flag_t enable, int mode, int inv)
+{
+ govrh_set_colorbar(p_govrh, enable, mode, inv);
+}
+
+void govrh_mod_set_tg(vpp_clock_t *tmr, unsigned int pixel_clock)
+{
+ govrh_set_tg(p_govrh, tmr, pixel_clock);
+}
+
+void govrh_mod_get_tg(vpp_clock_t *tmr)
+{
+ govrh_get_tg(p_govrh, tmr);
+}
+
+unsigned int govrh_mod_get_sts(void)
+{
+ return govrh_get_int_status(p_govrh);
+}
+
+void govrh_mod_clr_sts(unsigned int sts)
+{
+ govrh_clean_int_status(p_govrh, sts);
+}
+
+void govrh_mod_suspend(int sts)
+{
+ govrh_suspend(p_govrh, sts);
+}
+
+void govrh_mod_resume(int sts)
+{
+ govrh_resume(p_govrh, sts);
+}
+
+void govrh_mod_set_framebuf(vdo_framebuf_t *fb)
+{
+ govrh_set_framebuffer(p_govrh, fb);
+}
+void govrh_mod_set_addr(unsigned int yaddr, unsigned int caddr)
+{
+ govrh_set_fb_addr(p_govrh, yaddr, caddr);
+}
+void govrh_mod_get_addr(unsigned int *yaddr, unsigned int *caddr)
+{
+ govrh_get_fb_addr(p_govrh, yaddr, caddr);
+}
+void govrh_mod_set_csc(vpp_csc_t mode)
+{
+ govrh_set_csc_mode(p_govrh, mode);
+}
+vdo_color_fmt govrh_mod_get_color_fmt(void)
+{
+ return govrh_get_color_format(p_govrh);
+}
+void govrh_mod_set_color_fmt(vdo_color_fmt colfmt)
+{
+ govrh_set_color_format(p_govrh, colfmt);
+}
+
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+void cursor_mod_set_enable(vpp_flag_t enable)
+{
+ govrh_CUR_set_enable(p_cursor, enable);
+}
+
+void cursor_mod_set_framebuf(vdo_framebuf_t *fb)
+{
+ govrh_CUR_set_framebuffer(p_cursor, fb);
+}
+
+void cursor_mod_set_addr(unsigned int yaddr, unsigned int caddr)
+{
+ govrh_CUR_set_fb_addr(p_cursor, yaddr, caddr);
+}
+
+void cursor_mod_get_addr(unsigned int *yaddr, unsigned int *caddr)
+{
+ govrh_CUR_get_fb_addr(p_cursor, yaddr, caddr);
+}
+
+void cursor_mod_fn_view(int read, vdo_view_t *view)
+{
+ govrh_CUR_proc_view(p_cursor, read, view);
+}
+#endif
+
+#ifdef WMT_FTBLK_GOVRH2
+void govrh2_mod_dump_reg(void)
+{
+ govrh_reg_dump((void *)p_govrh2);
+}
+
+void govrh2_mod_set_enable(vpp_flag_t enable)
+{
+ govrh_set_MIF_enable(p_govrh2, enable);
+}
+
+void govrh2_mod_set_colorbar(vpp_flag_t enable, int mode, int inv)
+{
+ govrh_set_colorbar(p_govrh2, enable, mode, inv);
+}
+
+void govrh2_mod_set_tg(vpp_clock_t *tmr, unsigned int pixel_clock)
+{
+ govrh_set_tg(p_govrh2, tmr, pixel_clock);
+}
+
+void govrh2_mod_get_tg(vpp_clock_t *tmr)
+{
+ govrh_get_tg(p_govrh2, tmr);
+}
+
+unsigned int govrh2_mod_get_sts(void)
+{
+ return govrh_get_int_status(p_govrh2);
+}
+
+void govrh2_mod_clr_sts(unsigned int sts)
+{
+ govrh_clean_int_status(p_govrh2, sts);
+}
+
+void govrh2_mod_suspend(int sts)
+{
+ govrh_suspend(p_govrh2, sts);
+}
+
+void govrh2_mod_resume(int sts)
+{
+ govrh_resume(p_govrh2, sts);
+}
+
+void govrh2_mod_set_framebuf(vdo_framebuf_t *fb)
+{
+ govrh_set_framebuffer(p_govrh2, fb);
+}
+void govrh2_mod_set_addr(unsigned int yaddr, unsigned int caddr)
+{
+ govrh_set_fb_addr(p_govrh2, yaddr, caddr);
+}
+void govrh2_mod_get_addr(unsigned int *yaddr, unsigned int *caddr)
+{
+ govrh_get_fb_addr(p_govrh2, yaddr, caddr);
+}
+void govrh2_mod_set_csc(vpp_csc_t mode)
+{
+ govrh_set_csc_mode(p_govrh2, mode);
+}
+vdo_color_fmt govrh2_mod_get_color_fmt(void)
+{
+ return govrh_get_color_format(p_govrh2);
+}
+void govrh2_mod_set_color_fmt(vdo_color_fmt colfmt)
+{
+ govrh_set_color_format(p_govrh2, colfmt);
+}
+
+#endif
+int govrh_mod_init(void)
+{
+ govrh_mod_t *mod_p;
+ vpp_fb_base_t *mod_fb_p;
+ vdo_framebuf_t *fb_p;
+
+ mod_p = (govrh_mod_t *) vpp_mod_register(VPP_MOD_GOVRH,
+ sizeof(govrh_mod_t), VPP_MOD_FLAG_FRAMEBUF);
+ if (!mod_p) {
+ DPRINT("*E* GOVRH module register fail\n");
+ return -1;
+ }
+
+ /* module member variable */
+ mod_p->mmio = (void *)REG_GOVRH_BASE1_BEGIN;
+ mod_p->int_catch = VPP_INT_NULL; /* VPP_INT_ERR_GOVRH_MIF; */
+
+ /* module member function */
+ mod_p->init = govrh_init;
+ mod_p->dump_reg = govrh_mod_dump_reg;
+ mod_p->set_enable = govrh_mod_set_enable;
+ mod_p->set_colorbar = govrh_mod_set_colorbar;
+ mod_p->set_tg = govrh_mod_set_tg;
+ mod_p->get_tg = govrh_mod_get_tg;
+ mod_p->get_sts = govrh_mod_get_sts;
+ mod_p->clr_sts = govrh_mod_clr_sts;
+ mod_p->suspend = govrh_mod_suspend;
+ mod_p->resume = govrh_mod_resume;
+
+ /* module frame buffer */
+ mod_fb_p = mod_p->fb_p;
+ mod_fb_p->csc_mode = VPP_CSC_RGB2YUV_SDTV_0_255;
+ mod_fb_p->framerate = VPP_HD_DISP_FPS;
+ mod_fb_p->media_fmt = VPP_MEDIA_FMT_MPEG;
+ mod_fb_p->wait_ready = 0;
+ mod_fb_p->capability = BIT(VDO_COL_FMT_YUV420)
+ | BIT(VDO_COL_FMT_YUV422H)
+ | BIT(VDO_COL_FMT_YUV444) | BIT(VDO_COL_FMT_ARGB)
+ | BIT(VDO_COL_FMT_RGB_1555) | BIT(VDO_COL_FMT_RGB_565)
+ | VPP_FB_FLAG_CSC | VPP_FB_FLAG_MEDIA
+ | VPP_FB_FLAG_FIELD;
+
+ /* module frame buffer member function */
+ mod_fb_p->set_framebuf = govrh_mod_set_framebuf;
+ mod_fb_p->set_addr = govrh_mod_set_addr;
+ mod_fb_p->get_addr = govrh_mod_get_addr;
+ mod_fb_p->set_csc = govrh_mod_set_csc;
+ mod_fb_p->get_color_fmt = govrh_mod_get_color_fmt;
+ mod_fb_p->set_color_fmt = govrh_mod_set_color_fmt;
+
+ fb_p = &mod_p->fb_p->fb;
+ fb_p->y_addr = 0;
+ fb_p->c_addr = 0;
+ fb_p->col_fmt = VPP_UBOOT_COLFMT;
+ fb_p->img_w = VPP_HD_DISP_RESX;
+ fb_p->img_h = VPP_HD_DISP_RESY;
+ fb_p->fb_w = VPP_HD_MAX_RESX;
+ fb_p->fb_h = VPP_HD_MAX_RESY;
+ fb_p->h_crop = 0;
+ fb_p->v_crop = 0;
+ fb_p->flag = 0;
+
+ p_govrh = mod_p;
+
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ mod_p = (govrh_mod_t *) vpp_mod_register(VPP_MOD_CURSOR,
+ sizeof(govrh_cursor_mod_t), VPP_MOD_FLAG_FRAMEBUF);
+ if (!mod_p) {
+ DPRINT("*E* CURSOR module register fail\n");
+ return -1;
+ }
+
+ /* module member function */
+ mod_p->init = govrh_CUR_init;
+ mod_p->set_enable = cursor_mod_set_enable;
+ mod_p->mmio = (void *)REG_GOVRH_BASE1_BEGIN;
+
+ /* module frame buffer */
+ mod_fb_p = mod_p->fb_p;
+ mod_fb_p->csc_mode = VPP_CSC_RGB2YUV_SDTV_0_255;
+ mod_fb_p->framerate = VPP_HD_DISP_FPS;
+ mod_fb_p->media_fmt = VPP_MEDIA_FMT_MPEG;
+ mod_fb_p->wait_ready = 0;
+ mod_fb_p->capability = BIT(VDO_COL_FMT_YUV420)
+ | BIT(VDO_COL_FMT_YUV422H)
+ | BIT(VDO_COL_FMT_YUV444) | BIT(VDO_COL_FMT_ARGB)
+ | VPP_FB_FLAG_CSC | VPP_FB_FLAG_MEDIA
+ | VPP_FB_FLAG_FIELD;
+
+ /* module frame buffer member function */
+ mod_fb_p->set_framebuf = cursor_mod_set_framebuf;
+ mod_fb_p->set_addr = cursor_mod_set_addr;
+ mod_fb_p->get_addr = cursor_mod_get_addr;
+ mod_fb_p->fn_view = cursor_mod_fn_view;
+
+ fb_p = &mod_p->fb_p->fb;
+ fb_p->y_addr = 0;
+ fb_p->c_addr = 0;
+ fb_p->col_fmt = VDO_COL_FMT_YUV444;
+ fb_p->img_w = VPP_HD_DISP_RESX;
+ fb_p->img_h = VPP_HD_DISP_RESY;
+ fb_p->fb_w = VPP_HD_MAX_RESX;
+ fb_p->fb_h = VPP_HD_MAX_RESY;
+ fb_p->h_crop = 0;
+ fb_p->v_crop = 0;
+ fb_p->flag = 0;
+
+ p_cursor = (govrh_cursor_mod_t *) mod_p;
+#endif
+
+#ifdef WMT_FTBLK_GOVRH2
+ mod_p = (govrh_mod_t *) vpp_mod_register(VPP_MOD_GOVRH2,
+ sizeof(govrh_mod_t), VPP_MOD_FLAG_FRAMEBUF);
+ if (!mod_p) {
+ DPRINT("*E* GOVRH module register fail\n");
+ return -1;
+ }
+
+ /* module member variable */
+ mod_p->mmio = (void *)REG_GOVRH2_BASE1_BEGIN;
+ mod_p->int_catch = VPP_INT_NULL; /* VPP_INT_ERR_GOVRH_MIF; */
+
+ /* module member function */
+ mod_p->init = govrh_init;
+ mod_p->dump_reg = govrh2_mod_dump_reg;
+ mod_p->set_enable = govrh2_mod_set_enable;
+ mod_p->set_colorbar = govrh2_mod_set_colorbar;
+ mod_p->set_tg = govrh2_mod_set_tg;
+ mod_p->get_tg = govrh2_mod_get_tg;
+ mod_p->get_sts = govrh2_mod_get_sts;
+ mod_p->clr_sts = govrh2_mod_clr_sts;
+ mod_p->suspend = govrh2_mod_suspend;
+ mod_p->resume = govrh2_mod_resume;
+
+ /* module frame buffer */
+ mod_fb_p = mod_p->fb_p;
+ mod_fb_p->csc_mode = VPP_CSC_RGB2YUV_SDTV_0_255;
+ mod_fb_p->framerate = VPP_HD_DISP_FPS;
+ mod_fb_p->media_fmt = VPP_MEDIA_FMT_MPEG;
+ mod_fb_p->wait_ready = 0;
+ mod_fb_p->capability = BIT(VDO_COL_FMT_YUV420)
+ | BIT(VDO_COL_FMT_YUV422H)
+ | BIT(VDO_COL_FMT_YUV444) | BIT(VDO_COL_FMT_ARGB)
+ | BIT(VDO_COL_FMT_RGB_1555) | BIT(VDO_COL_FMT_RGB_565)
+ | VPP_FB_FLAG_CSC | VPP_FB_FLAG_MEDIA | VPP_FB_FLAG_FIELD;
+
+ /* module frame buffer member function */
+ mod_fb_p->set_framebuf = govrh2_mod_set_framebuf;
+ mod_fb_p->set_addr = govrh2_mod_set_addr;
+ mod_fb_p->get_addr = govrh2_mod_get_addr;
+ mod_fb_p->set_csc = govrh2_mod_set_csc;
+ mod_fb_p->get_color_fmt = govrh2_mod_get_color_fmt;
+ mod_fb_p->set_color_fmt = govrh2_mod_set_color_fmt;
+
+ fb_p = &mod_p->fb_p->fb;
+ fb_p->y_addr = 0;
+ fb_p->c_addr = 0;
+ fb_p->col_fmt = VPP_UBOOT_COLFMT;
+ fb_p->img_w = VPP_HD_DISP_RESX;
+ fb_p->img_h = VPP_HD_DISP_RESY;
+ fb_p->fb_w = VPP_HD_MAX_RESX;
+ fb_p->fb_h = VPP_HD_MAX_RESY;
+ fb_p->h_crop = 0;
+ fb_p->v_crop = 0;
+ fb_p->flag = 0;
+
+ p_govrh2 = mod_p;
+#endif
+ return 0;
+}
+module_init(govrh_mod_init);
+#endif /* WMT_FTBLK_GOVRH */
diff --git a/drivers/video/wmt/govrh.h b/drivers/video/wmt/govrh.h
new file mode 100644
index 00000000..b20c31ab
--- /dev/null
+++ b/drivers/video/wmt/govrh.h
@@ -0,0 +1,168 @@
+/*++
+ * linux/drivers/video/wmt/govrh.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp.h"
+
+#ifdef WMT_FTBLK_GOVRH
+
+#ifndef GOVRH_H
+#define GOVRH_H
+
+typedef struct {
+ VPP_MOD_BASE;
+
+ unsigned int *reg_bk2;
+ unsigned int pm_enable;
+ unsigned int pm_tg;
+ unsigned int vo_clock;
+ unsigned int underrun_cnt;
+} govrh_mod_t;
+
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+#define GOVRH_CURSOR_HIDE_TIME 15
+typedef struct {
+ VPP_MOD_BASE;
+
+ unsigned int posx;
+ unsigned int posy;
+ unsigned int hotspot_x;
+ unsigned int hotspot_y;
+ int chg_flag;
+ vdo_color_fmt colfmt;
+ unsigned int cursor_addr1;
+ unsigned int cursor_addr2;
+ int enable;
+ int hide_cnt;
+} govrh_cursor_mod_t;
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifdef GOVRH_C
+#define EXTERN
+#else
+#define EXTERN extern
+#endif
+
+EXTERN govrh_mod_t *p_govrh;
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+EXTERN govrh_cursor_mod_t *p_cursor;
+#endif
+#ifdef WMT_FTBLK_GOVRH2
+EXTERN govrh_mod_t *p_govrh2;
+#endif
+
+EXTERN void govrh_set_tg_enable(govrh_mod_t *base, vpp_flag_t enable);
+EXTERN int govrh_get_tg_mode(govrh_mod_t *base);
+EXTERN int govrh_get_hscale_up(govrh_mod_t *base);
+EXTERN void govrh_set_direct_path(govrh_mod_t *base, int enable);
+EXTERN void govrh_set_dvo_enable(govrh_mod_t *base, vpp_flag_t enable);
+EXTERN void govrh_set_dvo_sync_polar(govrh_mod_t *base,
+ vpp_flag_t hsync, vpp_flag_t vsync);
+EXTERN void govrh_set_dvo_outdatw(govrh_mod_t *base,
+ vpp_datawidht_t width);
+EXTERN void govrh_set_dvo_clock_delay(govrh_mod_t *base,
+ int inverse, int delay);
+EXTERN void govrh_set_colorbar(govrh_mod_t *base, vpp_flag_t enable,
+ int mode, int inv);
+EXTERN void govrh_set_contrast(govrh_mod_t *base, unsigned int value);
+EXTERN unsigned int govrh_get_contrast(govrh_mod_t *base);
+EXTERN void govrh_set_brightness(govrh_mod_t *base, unsigned int value);
+EXTERN unsigned int govrh_get_brightness(govrh_mod_t *base);
+EXTERN void govrh_set_saturation(govrh_mod_t *base, unsigned int value);
+EXTERN unsigned int govrh_get_saturation(govrh_mod_t *base);
+EXTERN void govrh_set_MIF_enable(govrh_mod_t *base, vpp_flag_t enable);
+EXTERN int govrh_get_MIF_enable(govrh_mod_t *base);
+EXTERN void govrh_set_color_format(govrh_mod_t *base,
+ vdo_color_fmt format);
+EXTERN vdo_color_fmt govrh_get_color_format(govrh_mod_t *base);
+EXTERN void govrh_set_source_format(govrh_mod_t *base,
+ vpp_display_format_t format);
+EXTERN void govrh_set_output_format(govrh_mod_t *base,
+ vpp_display_format_t field);
+EXTERN void govrh_set_fb_addr(govrh_mod_t *base,
+ unsigned int y_addr, unsigned int c_addr);
+EXTERN void govrh_get_fb_addr(govrh_mod_t *base,
+ unsigned int *y_addr, unsigned int *c_addr);
+EXTERN void govrh_set_fb_width(govrh_mod_t *base, unsigned int width);
+EXTERN void govrh_set_fb_info(govrh_mod_t *base, unsigned int width,
+ unsigned int act_width, unsigned int x_offset, unsigned int y_offset);
+EXTERN void govrh_get_fb_info(govrh_mod_t *base, unsigned int *width,
+ unsigned int *act_width, unsigned int *x_offset,
+ unsigned int *y_offset);
+EXTERN void govrh_set_fifo_index(govrh_mod_t *base, unsigned int index);
+EXTERN void govrh_set_reg_level(govrh_mod_t *base, vpp_reglevel_t level);
+EXTERN void govrh_set_reg_update(govrh_mod_t *base, vpp_flag_t enable);
+EXTERN void govrh_set_csc_mode(govrh_mod_t *base, vpp_csc_t mode);
+EXTERN void govrh_set_framebuffer(govrh_mod_t *base,
+ vdo_framebuf_t *inbuf);
+EXTERN void govrh_get_framebuffer(govrh_mod_t *base, vdo_framebuf_t *fb);
+EXTERN vdo_color_fmt govrh_get_dvo_color_format(govrh_mod_t *base);
+EXTERN void govrh_set_dvo_color_format(govrh_mod_t *base,
+ vdo_color_fmt fmt);
+EXTERN vpp_int_err_t govrh_get_int_status(govrh_mod_t *base);
+EXTERN void govrh_clean_int_status(govrh_mod_t *base,
+ vpp_int_err_t int_sts);
+EXTERN unsigned int govrh_set_clock(govrh_mod_t *base,
+ unsigned int pixel_clock);
+EXTERN void govrh_set_videomode(govrh_mod_t *base,
+ struct fb_videomode *vmode);
+EXTERN void govrh_get_tg(govrh_mod_t *base, vpp_clock_t *tmr);
+EXTERN void govrh_get_videomode(govrh_mod_t *base,
+ struct fb_videomode *vmode);
+EXTERN void govrh_HDMI_set_blank_value(govrh_mod_t *base,
+ unsigned int val);
+EXTERN void govrh_HDMI_set_3D_mode(govrh_mod_t *base, int mode);
+EXTERN int govrh_is_top_field(govrh_mod_t *base);
+EXTERN void govrh_IGS_set_mode(govrh_mod_t *base,
+ int no, int mode_18bit, int msb);
+EXTERN void govrh_IGS_set_RGB_swap(govrh_mod_t *base, int mode);
+EXTERN int govrh_mod_init(void);
+
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+EXTERN void govrh_CUR_set_enable(govrh_cursor_mod_t *base,
+ vpp_flag_t enable);
+EXTERN void govrh_CUR_set_framebuffer(govrh_cursor_mod_t *base,
+ vdo_framebuf_t *fb);
+EXTERN void govrh_CUR_set_coordinate(govrh_cursor_mod_t *base,
+ unsigned int x1, unsigned int y1, unsigned int x2, unsigned int y2);
+EXTERN void govrh_CUR_set_position(govrh_cursor_mod_t *base,
+ unsigned int x, unsigned int y);
+EXTERN void govrh_CUR_set_color_key_mode(govrh_cursor_mod_t *base,
+ int alpha, int enable, int mode);
+EXTERN void govrh_CUR_set_color_key(govrh_cursor_mod_t *base,
+ int enable, int alpha, unsigned int colkey);
+EXTERN void govrh_CUR_set_colfmt(govrh_cursor_mod_t *base,
+ vdo_color_fmt colfmt);
+EXTERN void govrh_CUR_irqproc(int arg);
+#endif
+
+#undef EXTERN
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* GOVRH_H */
+#endif /* WMT_FTBLK_GOVRH */
diff --git a/drivers/video/wmt/hdmi.c b/drivers/video/wmt/hdmi.c
new file mode 100644
index 00000000..47626d77
--- /dev/null
+++ b/drivers/video/wmt/hdmi.c
@@ -0,0 +1,1614 @@
+/*++
+ * linux/drivers/video/wmt/hdmi.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define HDMI_C
+#undef DEBUG
+/* #define DEBUG */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "hdmi.h"
+#include "vout.h"
+#ifdef __KERNEL__
+#include <asm/div64.h>
+#endif
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define HDMI_XXXX 1 *//*Example*/
+/* #define CONFIG_HDMI_INFOFRAME_DISABLE */
+/* #define CONFIG_HDMI_EDID_DISABLE */
+
+#define HDMI_I2C_FREQ 80000
+
+#define MAX_ACR_TV_NUM 128
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx hdmi_xxx_t; *//*Example*/
+enum hdmi_fifo_slot_t {
+ HDMI_FIFO_SLOT_AVI = 0,
+ HDMI_FIFO_SLOT_VENDOR = 1,
+ HDMI_FIFO_SLOT_AUDIO = 2,
+ HDMI_FIFO_SLOT_CONTROL = 3,
+ HDMI_FIFO_SLOT_MAX = 15
+};
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in hdmi.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int hdmi_xxx; *//*Example*/
+
+/*
+Added by howayhuo
+Some TV need fix the HDMI ACR ration, otherwise these TV will show "overclock"
+Fill in the TV vendor name and TV monitor name to this list if your TV needs fix acr
+*/
+static const tv_name_t fix_acr_tv_list[] = {
+ {"TCL", "RTD2662"}, //another name: TCL L19E09
+};
+
+static tv_name_t *p_fix_acr_tv;
+
+static int fixed_acr_ration_val = 2300;
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void hdmi_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+/*---------------------------- HDMI COMMON API -------------------------------*/
+unsigned int hdmi_reg32_write(U32 offset, U32 mask, U32 shift, U32 val)
+{
+ unsigned int new_val;
+
+ new_val = (vppif_reg32_in(offset) & ~(mask))
+ | (((val) << (shift)) & mask);
+ if (offset == REG_HDMI_GENERAL_CTRL) {
+ new_val &= ~(BIT24 | BIT25 | BIT26);
+ new_val |= g_vpp.hdmi_ctrl;
+ DBG_MSG("[HDMI] reg32_wr 0x%x ctrl 0x%x\n",
+ new_val, g_vpp.hdmi_ctrl);
+ }
+ vppif_reg32_out(offset, new_val);
+ return new_val;
+}
+
+unsigned char hdmi_ecc(unsigned char *buf, int bit_cnt)
+{
+ #define HDMI_CRC_LEN 9
+
+ int crc[HDMI_CRC_LEN], crc_o[HDMI_CRC_LEN];
+ int i, j;
+ int input, result, result_rev = 0;
+
+ for (i = 0; i < HDMI_CRC_LEN; i++)
+ crc[i] = 0;
+
+ for (i = 0; i < bit_cnt; i++) {
+ for (j = 0; j < HDMI_CRC_LEN; j++)
+ crc_o[j] = crc[j];
+ input = (buf[i/8] & (1<<(i%8))) ? 1 : 0;
+ crc[0] = crc_o[7] ^ input;
+ crc[1] = crc_o[0];
+ crc[2] = crc_o[1];
+ crc[3] = crc_o[2];
+ crc[4] = crc_o[3];
+ crc[5] = crc_o[4];
+ crc[6] = crc_o[5] ^ crc_o[7] ^ input;
+ crc[7] = crc_o[6] ^ crc_o[7] ^ input;
+ crc[8] = crc_o[7];
+
+ result = 0;
+ result_rev = 0;
+ for (j = 0; j < HDMI_CRC_LEN - 1; j++) {
+ result += (crc[j] << j);
+ result_rev += (crc[j] << (HDMI_CRC_LEN - 2 - j));
+ }
+ }
+
+/* DPRINT("[HDMI] crc 0x%x, %x %x %x %x %x %x %x\n",result_rev,
+ buf[0],buf[1],buf[2],buf[3],buf[4],buf[5],buf[6]); */
+ return result_rev;
+}
+
+unsigned char hdmi_checksum(unsigned char *header,
+ unsigned char *buf, int cnt)
+{
+ unsigned char sum;
+ int i;
+
+ for (i = 0, sum = 0; i < cnt; i++)
+ sum += buf[i];
+ for (i = 0; i < 3; i++)
+ sum += header[i];
+ return 0 - sum;
+}
+
+#ifdef WMT_FTBLK_HDMI
+/*---------------------------- HDMI HAL --------------------------------------*/
+void hdmi_set_power_down(int pwrdn)
+{
+ DBG_DETAIL("(%d)\n", pwrdn);
+
+ if ((vppif_reg32_read(HDMI_PD) == 0) && (pwrdn == 0)) {
+ return; /* avoid HDMI reset */
+ }
+
+ vppif_reg32_write(HDMI_INTERNAL_LDO, (pwrdn) ? 0 : 1);
+ vppif_reg32_write(HDMI_PD, pwrdn);
+ if (!pwrdn) {
+ vppif_reg32_write(HDMI_RESET_PLL, 1);
+ mdelay(1);
+ vppif_reg32_write(HDMI_RESET_PLL, 0);
+ }
+ mdelay(1);
+ vppif_reg32_write(HDMI_PD_L2HA, pwrdn);
+}
+
+int hdmi_get_power_down(void)
+{
+ return vppif_reg32_read(HDMI_PD);
+}
+
+void hdmi_set_enable(vpp_flag_t enable)
+{
+ hdmi_reg32_write(HDMI_ENABLE, enable);
+ vppif_reg32_write(HDMI_MODE, (enable) ? 0 : 1);
+}
+
+void hdmi_set_avmute(vpp_flag_t mute)
+{
+ vppif_reg32_write(HDMI_AVMUTE_SET_ENABLE, mute);
+}
+
+void hdmi_set_dvi_enable(vpp_flag_t enable)
+{
+ hdmi_reg32_write(HDMI_DVI_MODE_ENABLE, enable);
+}
+
+void hdmi_set_sync_low_active(vpp_flag_t hsync, vpp_flag_t vsync)
+{
+ hdmi_reg32_write(HDMI_HSYNC_LOW_ACTIVE, hsync);
+ hdmi_reg32_write(HDMI_VSYNC_LOW_ACTIVE, vsync);
+}
+
+void hdmi_get_sync_polar(int *hsync_hi, int *vsync_hi)
+{
+ *hsync_hi = (vppif_reg32_read(HDMI_HSYNC_LOW_ACTIVE)) ? 0 : 1;
+ *vsync_hi = (vppif_reg32_read(HDMI_VSYNC_LOW_ACTIVE)) ? 0 : 1;
+}
+
+void hdmi_set_output_colfmt(vdo_color_fmt colfmt)
+{
+ unsigned int val;
+
+ switch (colfmt) {
+ default:
+ case VDO_COL_FMT_ARGB:
+ val = 0;
+ break;
+ case VDO_COL_FMT_YUV444:
+ val = 1;
+ break;
+ case VDO_COL_FMT_YUV422H:
+ case VDO_COL_FMT_YUV422V:
+ val = 2;
+ break;
+ }
+ hdmi_reg32_write(HDMI_CONVERT_YUV422, (val == 2) ? 1 : 0);
+ hdmi_reg32_write(HDMI_OUTPUT_FORMAT, val);
+}
+
+vdo_color_fmt hdmi_get_output_colfmt(void)
+{
+ unsigned int val;
+
+ val = vppif_reg32_read(HDMI_OUTPUT_FORMAT);
+ switch (val) {
+ default:
+ case 0:
+ return VDO_COL_FMT_ARGB;
+ case 1:
+ return VDO_COL_FMT_YUV444;
+ case 2:
+ return VDO_COL_FMT_YUV422H;
+ }
+ return VDO_COL_FMT_ARGB;
+}
+
+int hdmi_get_plugin(void)
+{
+ int plugin;
+
+ if (vppif_reg32_read(HDMI_HOTPLUG_IN_INT)) {
+ plugin = vppif_reg32_read(HDMI_HOTPLUG_IN);
+ } else {
+ int tre_en;
+
+ tre_en = vppif_reg32_read(HDMI_TRE_EN);
+ vppif_reg32_write(HDMI_TRE_EN, 0);
+ plugin = vppif_reg32_read(HDMI_RSEN);
+ vppif_reg32_write(HDMI_TRE_EN, tre_en);
+ }
+ return plugin;
+}
+
+int hdmi_get_plug_status(void)
+{
+ int reg;
+
+ reg = vppif_reg32_in(REG_HDMI_HOTPLUG_DETECT);
+ return reg & 0x3000000;
+}
+
+void hdmi_clear_plug_status(void)
+{
+ vppif_reg32_write(HDMI_HOTPLUG_IN_STS, 1);
+ vppif_reg32_write(HDMI_HOTPLUG_OUT_STS, 1);
+}
+
+void hdmi_enable_plugin(int enable)
+{
+ vppif_reg32_write(HDMI_HOTPLUG_OUT_INT, enable);
+ vppif_reg32_write(HDMI_HOTPLUG_IN_INT, enable);
+}
+
+void hdmi_write_fifo(enum hdmi_fifo_slot_t no, unsigned int *buf, int cnt)
+{
+ int i;
+
+ if (no > HDMI_FIFO_SLOT_MAX)
+ return;
+#ifdef DEBUG
+{
+ char *ptr;
+
+ DPRINT("[HDMI] wr fifo %d,cnt %d", no, cnt);
+ ptr = (char *) buf;
+ for (i = 0; i < cnt; i++) {
+ if ((i % 4) == 0)
+ DPRINT("\n %02d :", i);
+ DPRINT(" 0x%02x", ptr[i]);
+ }
+ DPRINT("\n[HDMI] AVI info package end\n");
+}
+#endif
+
+ vppif_reg32_out(REG_HDMI_FIFO_CTRL, (no << 8));
+ cnt = (cnt + 3) / 4;
+ for (i = 0; i < cnt; i++)
+ vppif_reg32_out(REG_HDMI_WR_FIFO_ADDR + 4 * i, buf[i]);
+ vppif_reg32_write(HDMI_INFOFRAME_WR_STROBE, 1);
+}
+
+void hdmi_read_fifo(enum hdmi_fifo_slot_t no, unsigned int *buf, int cnt)
+{
+ int i;
+ int rdy;
+
+ if (no > HDMI_FIFO_SLOT_MAX)
+ return;
+
+ rdy = vppif_reg32_read(HDMI_INFOFRAME_FIFO1_RDY);
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_RDY, 0);
+
+ no = no - 1;
+ vppif_reg32_out(REG_HDMI_FIFO_CTRL, (no << 8));
+ vppif_reg32_write(HDMI_INFOFRAME_RD_STROBE, 1);
+ cnt = (cnt + 3) / 4;
+ for (i = 0; i < cnt; i++)
+ buf[i] = vppif_reg32_in(REG_HDMI_RD_FIFO_ADDR + 4 * i);
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_RDY, rdy);
+#ifdef DEBUG
+{
+ char *ptr;
+
+ cnt *= 4;
+ DPRINT("[HDMI] rd fifo %d,cnt %d", no, cnt);
+ ptr = (char *) buf;
+ for (i = 0; i < cnt; i++) {
+ if ((i % 4) == 0)
+ DPRINT("\n %02d :", i);
+ DPRINT(" 0x%02x", ptr[i]);
+ }
+ DPRINT("\n[HDMI] AVI info package end\n");
+}
+#endif
+}
+
+#if 1
+int hdmi_ddc_delay_us = 5;
+int hdmi_ddc_ctrl_delay_us = 5;
+
+#define HDMI_DDC_OUT
+#define HDMI_DDC_DELAY hdmi_ddc_delay_us
+#define HDMI_DDC_CHK_DELAY 1
+#define HDMI_DDC_CTRL_DELAY hdmi_ddc_ctrl_delay_us
+#else
+#define HDMI_DDC_OUT
+#define HDMI_DDC_DELAY 1
+#define HDMI_DDC_CHK_DELAY 1
+#define HDMI_DDC_CTRL_DELAY 1
+#endif
+
+#define HDMI_STATUS_START BIT16
+#define HDMI_STATUS_STOP BIT17
+#define HDMI_STATUS_WR_AVAIL BIT18
+#define HDMI_STATUS_CP_USE BIT19
+#define HDMI_STATUS_SW_READ BIT25
+int hdmi_DDC_check_status(unsigned int checkbits, int condition)
+{
+ int status = 1;
+ unsigned int i = 0, maxloop;
+
+ maxloop = 500 / HDMI_DDC_CHK_DELAY;
+ udelay(HDMI_DDC_DELAY);
+
+ if (condition) { /* wait 1 --> 0 */
+ while ((vppif_reg32_in(REG_HDMI_I2C_CTRL2) & checkbits)
+ && (i < maxloop)) {
+ udelay(HDMI_DDC_CHK_DELAY);
+ if (++i == maxloop)
+ status = 0;
+ }
+ } else { /* wait 0 --> 1 */
+ while (!(vppif_reg32_in(REG_HDMI_I2C_CTRL2) & checkbits)
+ && (i < maxloop)) {
+ udelay(HDMI_DDC_CHK_DELAY);
+ if (++i == maxloop)
+ status = 0;
+ }
+ }
+
+ if ((status == 0) && (checkbits != HDMI_STATUS_SW_READ)) {
+ unsigned int reg;
+
+ reg = vppif_reg32_in(REG_HDMI_I2C_CTRL2);
+ DBG_DETAIL("[HDMI] status timeout check 0x%x,wait to %s\n",
+ checkbits, (condition) ? "0" : "1");
+ DBG_DETAIL("[HDMI] 0x%x,sta %d,stop %d,wr %d,rd %d,cp %d\n",
+ reg, (reg & HDMI_STATUS_START) ? 1 : 0,
+ (reg & HDMI_STATUS_STOP) ? 1 : 0,
+ (reg & HDMI_STATUS_WR_AVAIL) ? 1 : 0,
+ (reg & HDMI_STATUS_SW_READ) ? 1 : 0,
+ (reg & HDMI_STATUS_CP_USE) ? 1 : 0);
+ }
+ return status;
+}
+
+void hdmi_DDC_set_freq(unsigned int hz)
+{
+ unsigned int clock;
+ unsigned int div;
+
+ clock = 25000000*15/100; /* RTC clock source */
+ div = clock / hz;
+
+ vppif_reg32_write(HDMI_I2C_CLK_DIVIDER, div);
+ DBG_DETAIL("[HDMI] set freq(%d,clk %d,div %d)\n", hz, clock, div);
+}
+
+void hdmi_DDC_reset(void)
+{
+ vppif_reg32_write(HDMI_I2C_SW_RESET, 1);
+ udelay(1);
+ vppif_reg32_write(HDMI_I2C_SW_RESET, 0);
+}
+
+int hdmi_DDC_read_func(char addr, int index, char *buf, int length)
+{
+ int status = 1;
+ unsigned int i = 0;
+ int err_cnt = 0;
+
+ DBG_DETAIL("[HDMI] read DDC(index 0x%x,len %d),reg 0x%x\n",
+ index, length, vppif_reg32_in(REG_HDMI_I2C_CTRL2));
+
+#ifdef CONFIG_HDMI_EDID_DISABLE
+ return status;
+#endif
+
+ hdmi_DDC_set_freq(g_vpp.hdmi_i2c_freq);
+ /* enhanced DDC read */
+ if (index >= 256) {
+ /* sw start, write data avail */
+ vppif_reg32_write(REG_HDMI_I2C_CTRL2, BIT18 + BIT16, 16, 0x5);
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait start & wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_START +
+ HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* start\n");
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+ /* Slave address */
+ vppif_reg32_write(HDMI_WR_DATA, 0x60);
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* slave addr 0x%x\n", addr);
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+ /* Offset */
+ vppif_reg32_write(HDMI_WR_DATA, 0x1);
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* index 0x%x\n", index);
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+ index -= 256;
+ }
+
+ /* START */
+#ifdef HDMI_DDC_OUT
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, 0x50000);
+#else
+#if 0
+ vppif_reg32_write(HDMI_SW_START_REQ, 1); /* sw start */
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+#endif
+ /* sw start, write data avail */
+ vppif_reg32_write(REG_HDMI_I2C_CTRL2, BIT18 + BIT16, 16, 0x5);
+#endif
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait start & wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_START +
+ HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* start\n");
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+ /* Slave address */
+#ifdef HDMI_DDC_OUT
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, 0x400A0);
+#else
+ vppif_reg32_write(HDMI_WR_DATA, addr);
+ udelay(HDMI_DDC_DELAY);
+ while (vppif_reg32_read(HDMI_WR_DATA) != addr)
+ ;
+ udelay(HDMI_DDC_DELAY);
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+#endif
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* slave addr 0x%x\n", addr);
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+ /* Offset */
+#ifdef HDMI_DDC_OUT
+{
+ unsigned int reg;
+
+ reg = 0x40000;
+ reg |= index;
+
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, reg);
+}
+#else
+ vppif_reg32_write(HDMI_WR_DATA, index);
+ udelay(HDMI_DDC_DELAY);
+ while (vppif_reg32_read(HDMI_WR_DATA) != index)
+ ;
+ udelay(HDMI_DDC_DELAY);
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+#endif
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* index 0x%x\n", index);
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+/* vppif_reg32_write(HDMI_WR_DATA,addr+1); */
+
+ /* START */
+#ifdef HDMI_DDC_OUT
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, 0x50000);
+#else
+#if 0
+ vppif_reg32_write(HDMI_SW_START_REQ, 1); /* sw start */
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+#endif
+ /* sw start, write data avail */
+ vppif_reg32_write(REG_HDMI_I2C_CTRL2, BIT18 + BIT16, 16, 0x5);
+#endif
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait start & wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_START +
+ HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* restart\n");
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+ /* Slave Address + 1 */
+#ifdef HDMI_DDC_OUT
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, 0x400A1);
+#else
+ vppif_reg32_write(HDMI_WR_DATA, addr + 1);
+ udelay(HDMI_DDC_DELAY);
+ while (vppif_reg32_read(HDMI_WR_DATA) != (addr + 1))
+ ;
+ udelay(HDMI_DDC_DELAY);
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+#endif
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* slave addr 0x%x\n", addr + 1);
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+
+ /* Read Data */
+ for (i = 0; i < length; i++) {
+#ifdef HDMI_DDC_OUT
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, 0x40000);
+#else
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+ /* hdmi_reg32_write(HDMI_WR_DATA_AVAIL,1); */
+#endif
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_WR_AVAIL, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* wr ACK(%d)\n", i);
+ err_cnt++;
+ goto ddc_read_fail;
+ /* break; */
+ }
+
+ /* wait sw read not set */
+ status = hdmi_DDC_check_status(HDMI_STATUS_SW_READ, 0);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* read avail(%d)\n", i);
+ if (i == 0) {
+ err_cnt++;
+ /* goto ddc_read_fail; */
+ } else {
+ /* g_vpp.dbg_hdmi_ddc_read_err++; */
+ }
+ goto ddc_read_fail;
+ /* break; */
+ }
+
+ *buf++ = vppif_reg32_read(HDMI_RD_DATA);
+ udelay(HDMI_DDC_DELAY);
+#ifdef HDMI_DDC_OUT
+ vppif_reg32_out(REG_HDMI_I2C_CTRL2, 0x2000000);
+#else
+ vppif_reg32_write(HDMI_SW_READ, 1);
+#endif
+ udelay(HDMI_DDC_DELAY);
+ }
+
+ /* STOP */
+#if 0
+ vppif_reg32_write(HDMI_SW_STOP_REQ, 1); /* sw stop */
+ vppif_reg32_write(HDMI_WR_DATA_AVAIL, 1); /* write data avail */
+#endif
+ /* sw stop, write data avail */
+ vppif_reg32_write(REG_HDMI_I2C_CTRL2, BIT18 + BIT17, 17, 3);
+ udelay(HDMI_DDC_CTRL_DELAY);
+ /* wait start & wr data avail */
+ status = hdmi_DDC_check_status(HDMI_STATUS_STOP +
+ HDMI_STATUS_WR_AVAIL + HDMI_STATUS_CP_USE, 1);
+ if (status == 0) {
+ DBGMSG("[HDMI] *E* stop\n");
+ err_cnt++;
+ goto ddc_read_fail;
+ }
+ udelay(HDMI_DDC_DELAY);
+
+ddc_read_fail:
+ if (err_cnt)
+ DBGMSG("[HDMI] *E* read DDC %d\n", err_cnt);
+ return (err_cnt) ? 1 : 0;
+}
+
+int hdmi_DDC_read(char addr, int index, char *buf, int length)
+{
+ int retry = 3;
+ do {
+ if (hdmi_DDC_read_func(addr, index, buf, length) == 0)
+ break;
+ hdmi_DDC_reset();
+ DPRINT("[HDMI] *W* DDC reset\n");
+ retry--;
+ } while (retry);
+ return (retry == 0) ? 1 : 0;
+}
+
+void hdmi_audio_enable(vpp_flag_t enable)
+{
+ vppif_reg32_write(HDMI_AUD_ENABLE, enable);
+}
+
+void hdmi_audio_mute(vpp_flag_t enable)
+{
+ vppif_reg32_write(HDMI_AUD_MUTE, enable);
+}
+
+/*----------------------- HDMI API --------------------------------------*/
+void hdmi_write_packet(unsigned int header, unsigned char *packet,
+ int cnt)
+{
+ unsigned char buf[36];
+ int i;
+ enum hdmi_fifo_slot_t no;
+
+#ifdef CONFIG_HDMI_INFOFRAME_DISABLE
+ return;
+#endif
+ memcpy(&buf[0], &header, 3);
+ buf[3] = hdmi_ecc((unsigned char *)&header, 24);
+ for (i = 0; i < cnt / 7; i++) {
+ memcpy(&buf[4+8*i], &packet[7*i], 7);
+ buf[11+8*i] = hdmi_ecc(&packet[7*i], 56);
+ }
+
+ switch (header & 0xFF) {
+ case HDMI_PACKET_INFOFRAME_AVI:
+ no = HDMI_FIFO_SLOT_AVI;
+ break;
+ case HDMI_PACKET_INFOFRAME_AUDIO:
+ no = HDMI_FIFO_SLOT_AUDIO;
+ break;
+ case HDMI_PACKET_INFOFRAME_VENDOR:
+ no = HDMI_FIFO_SLOT_VENDOR;
+ break;
+ default:
+ no = HDMI_FIFO_SLOT_CONTROL;
+ break;
+ }
+ hdmi_write_fifo(no, (unsigned int *)buf, (4 + 8 * (cnt / 7)));
+}
+
+void hdmi_tx_null_packet(void)
+{
+ hdmi_write_packet(HDMI_PACKET_NULL, 0, 0);
+}
+
+void hdmi_tx_general_control_packet(int mute)
+{
+ unsigned char buf[7];
+ memset(buf, 0x0, 7);
+ buf[0] = (mute) ? 0x01 : 0x10;
+ buf[1] = HDMI_COLOR_DEPTH_24 | (HDMI_PHASE_4 << 4);
+ hdmi_write_packet(HDMI_PACKET_GENERAL_CTRL, buf, 7);
+}
+
+int hdmi_get_pic_aspect(hdmi_video_code_t vic)
+{
+ switch (vic) {
+ case HDMI_640x480p60_4x3:
+ case HDMI_720x480p60_4x3:
+ case HDMI_1440x480i60_4x3:
+ case HDMI_1440x240p60_4x3:
+ case HDMI_2880x480i60_4x3:
+ case HDMI_2880x240p60_4x3:
+ case HDMI_1440x480p60_4x3:
+ case HDMI_720x576p50_4x3:
+ case HDMI_1440x576i50_4x3:
+ case HDMI_1440x288p50_4x3:
+ case HDMI_2880x576i50_4x3:
+ case HDMI_2880x288p50_4x3:
+ case HDMI_1440x576p50_4x3:
+ return HDMI_PIC_ASPECT_4_3;
+ default:
+ break;
+ }
+ return HDMI_PIC_ASPECT_16_9;
+}
+
+int hdmi_get_vic(int resx, int resy, int fps, int interlace)
+{
+ hdmi_vic_t info;
+ int i;
+
+ info.resx = resx;
+ info.resy = resy;
+ info.freq = fps;
+ info.option = (interlace) ? HDMI_VIC_INTERLACE : HDMI_VIC_PROGRESS;
+ info.option |= (vout_check_ratio_16_9(resx, resy)) ?
+ HDMI_VIC_16x9 : HDMI_VIC_4x3;
+ for (i = 0; i < HDMI_VIDEO_CODE_MAX; i++) {
+ if (memcmp(&hdmi_vic_info[i], &info, sizeof(hdmi_vic_t)) == 0)
+ return i;
+ }
+ return HDMI_UNKNOW;
+}
+
+void hdmi_tx_avi_infoframe_packet(vdo_color_fmt colfmt,
+ hdmi_video_code_t vic)
+{
+ unsigned int header;
+ unsigned char buf[28];
+ unsigned char temp;
+
+ memset(buf, 0x0, 28);
+ header = HDMI_PACKET_INFOFRAME_AVI + (0x2 << 8) + (0x0d << 16);
+ buf[1] = HDMI_SI_NO_DATA + (HDMI_BI_V_H_VALID << 2) +
+ (HDMI_AF_INFO_NO_DATA << 4);
+ switch (colfmt) {
+ case VDO_COL_FMT_YUV422H:
+ case VDO_COL_FMT_YUV422V:
+ temp = HDMI_OUTPUT_YUV422;
+ break;
+ case VDO_COL_FMT_YUV444:
+ temp = HDMI_OUTPUT_YUV444;
+ break;
+ case VDO_COL_FMT_ARGB:
+ default:
+ temp = HDMI_OUTPUT_RGB;
+ break;
+ }
+ buf[1] += (temp << 5);
+ buf[2] = HDMI_ASPECT_RATIO_PIC + (hdmi_get_pic_aspect(vic) << 4) +
+ (HDMI_COLORIMETRY_ITU709 << 6);
+ buf[3] = 0x84;
+ buf[4] = vic;
+ switch (vic) {
+ case HDMI_1440x480i60_16x9:
+ case HDMI_1440x576i50_16x9:
+ buf[5] = HDMI_PIXEL_REP_2;
+ break;
+ default:
+ buf[5] = HDMI_PIXEL_REP_NO;
+ break;
+ }
+ buf[0] = hdmi_checksum((unsigned char *)&header, buf, 28);
+ hdmi_write_packet(header, buf, 28);
+}
+
+void hdmi_tx_audio_infoframe_packet(int channel, int freq)
+{
+ unsigned int header;
+ unsigned char buf[28];
+
+ memset(buf, 0x0, 28);
+ header = HDMI_PACKET_INFOFRAME_AUDIO + (0x1 << 8) + (0x0a << 16);
+ buf[1] = (channel - 1) + (HDMI_AUD_TYPE_REF_STM << 4);
+ buf[2] = 0x0; /* HDMI_AUD_SAMPLE_24 + (freq << 2); */
+ buf[3] = 0x00;
+ buf[4] = 0x0;
+ buf[5] = 0x0; /* 0 db */
+ buf[0] = hdmi_checksum((unsigned char *)&header, buf, 28);
+ hdmi_write_packet(header, buf, 28);
+}
+
+void hdmi_tx_vendor_specific_infoframe_packet(void)
+{
+ unsigned int header;
+ unsigned char buf[28];
+ unsigned char structure_3d, meta_present;
+ unsigned char hdmi_video_format;
+
+ /* 0-No,1-1 byte param,2-3D format */
+ hdmi_video_format = (g_vpp.hdmi_3d_type) ? 2 : 0;
+ /* HDMI_3D_STRUCTURE_XXX; */
+ structure_3d = (g_vpp.hdmi_3d_type == 1) ? 0 : g_vpp.hdmi_3d_type;
+ meta_present = 0;
+
+ memset(buf, 0x0, 28);
+ header = HDMI_PACKET_INFOFRAME_VENDOR + (0x1 << 8) + (0xa << 16);
+ buf[1] = 0x3;
+ buf[2] = 0xC;
+ buf[3] = 0x0;
+ buf[4] = (hdmi_video_format << 5);
+ buf[5] = (structure_3d << 4) + ((meta_present) ? 0x8 : 0x0);
+ buf[6] = 0x0; /* 3D_Ext_Data */
+#if 0 /* metadata present */
+ buf[7] = 0x0; /* 3D_Metadata_type,3D_Metadata_Length(N) */
+ buf[8] = 0x0; /* 3D Metadata 1_N */
+#endif
+ buf[0] = hdmi_checksum((unsigned char *)&header, buf, 28);
+ hdmi_write_packet(header, buf, 28);
+}
+
+/*
+--> Added by howayhuo.
+Some TV (example: TCL L19E09) will overclock if ACR ratio too large,
+So we need decrease the ACR ratio for some special TV
+*/
+static void print_acr(void)
+{
+ int i;
+
+ if(p_fix_acr_tv != NULL) {
+ for(i = 0; i < MAX_ACR_TV_NUM; i++) {
+ if(strlen(p_fix_acr_tv[i].vendor_name) == 0
+ || strlen(p_fix_acr_tv[i].monitor_name) == 0)
+ break;
+
+ if(i == 0)
+ printk("ACR TV Name:\n");
+
+ printk(" %s,%s\n", p_fix_acr_tv[i].vendor_name, p_fix_acr_tv[i].monitor_name);
+ }
+ }
+}
+
+static void acr_init(void)
+{
+ char buf[512] = {0};
+ int buflen = 512;
+ unsigned long val;
+ int i, j, k, tv_num;
+ int ret, to_save_vendor;
+ tv_name_t tv_name;
+
+ if(p_fix_acr_tv != NULL) {
+ kfree(p_fix_acr_tv);
+ p_fix_acr_tv = NULL;
+ }
+
+ if(wmt_getsyspara("wmt.acr.ratio", buf, &buflen) == 0) {
+ ret = strict_strtoul(buf, 10, &val);
+ if(ret) {
+ printk("[HDMI] Wrong wmt.acr.ratio value: %s\n", buf);
+ return;
+ }
+ if(val >= 0 && val < 0xFFFFF) // total 20 bits
+ fixed_acr_ration_val = (int)val;
+ else
+ printk("[HDMI] Invalid Fixed ACR Ratio: %lu\n", val);
+ }
+
+ if(fixed_acr_ration_val == 0)
+ return;
+
+ /*
+ For example: setenv wmt.acr.tv 'TCH,RTD2662;PHL,Philips 244E'
+ */
+ if(wmt_getsyspara("wmt.acr.tv", buf, &buflen) != 0) {
+ p_fix_acr_tv = (tv_name_t *)kzalloc(sizeof(fix_acr_tv_list) + sizeof(tv_name_t), GFP_KERNEL);
+ if(p_fix_acr_tv) {
+ memcpy(p_fix_acr_tv, fix_acr_tv_list, sizeof(fix_acr_tv_list));
+ print_acr();
+ } else
+ printk("[HDMI] malloc for ACR fail. malloc len = %d\n",
+ sizeof(fix_acr_tv_list) + sizeof(tv_name_t));
+
+ return;
+ }
+
+ tv_num = 0;
+ buflen = strlen(buf);
+ if(buflen == 0)
+ return;
+
+ if(buflen == sizeof(buf)) {
+ printk("[HDMI] wmt.acr.tv too long\n");
+ return;
+ }
+
+ for(i = 0; i < buflen; i++) {
+ if(buf[i] == ',')
+ tv_num++;
+ }
+
+ /*
+ Limit TV Number
+ */
+ if(tv_num > MAX_ACR_TV_NUM)
+ tv_num = MAX_ACR_TV_NUM;
+
+ if(tv_num == 0)
+ return;
+
+ printk("acr_tv_num = %d\n", tv_num);
+ p_fix_acr_tv = (tv_name_t *)kzalloc((tv_num + 1) * sizeof(tv_name_t), GFP_KERNEL);
+ if(!p_fix_acr_tv) {
+ printk("[HDMI] malloc for ACR fail. malloc len = %d\n",
+ sizeof(fix_acr_tv_list) + sizeof(tv_name_t));
+ return;
+ }
+ memset(&tv_name, 0, sizeof(tv_name_t));
+
+ j = 0;
+ k = 0;
+ to_save_vendor= 1;
+ for(i = 0; i < buflen + 1; i++) {
+ if(buf[i] != ',' && buf[i] != ';' && buf[i] != '\0') {
+ if(to_save_vendor) {
+ if(k < VENDOR_NAME_LEN)
+ tv_name.vendor_name[k] = buf[i];
+ } else {
+ if(k < MONITOR_NAME_LEN)
+ tv_name.monitor_name[k] = buf[i];
+ }
+ k++;
+ } else if(buf[i] == ',') {
+ to_save_vendor = 0;
+ k = 0;
+ } else {
+ if(strlen(tv_name.vendor_name) == 0 || strlen(tv_name.monitor_name) == 0) {
+ printk("[HDMI] Wrong wmt.acr.tv format\n");
+ kfree(p_fix_acr_tv);
+ p_fix_acr_tv = NULL;
+ break;
+ } else {
+ if(j < tv_num) {
+ memcpy(p_fix_acr_tv + j, &tv_name, sizeof(tv_name_t));
+ memset(&tv_name, 0, sizeof(tv_name_t));
+ j++;
+ }
+
+ if(j == tv_num)
+ break;
+ }
+
+ if(buf[i]== ';') {
+ to_save_vendor = 1;
+ k = 0;
+ } else
+ break;
+ }
+ }
+
+ print_acr();
+}
+
+void acr_exit(void)
+{
+ if(p_fix_acr_tv != NULL) {
+ kfree(p_fix_acr_tv);
+ p_fix_acr_tv = NULL;
+ }
+}
+
+static int use_fix_acr_ratio(void)
+{
+ int i;
+
+ if(fixed_acr_ration_val == 0 || p_fix_acr_tv == NULL)
+ return 0;
+
+ for(i = 0; i < MAX_ACR_TV_NUM; i++) {
+ if(strlen(p_fix_acr_tv[i].vendor_name) == 0
+ || strlen(p_fix_acr_tv[i].monitor_name) == 0)
+ break;
+
+ if(!strcmp(edid_parsed.tv_name.vendor_name, p_fix_acr_tv[i].vendor_name)
+ && !strcmp(edid_parsed.tv_name.monitor_name, p_fix_acr_tv[i].monitor_name)) {
+ printk("TV is \"%s %s\". Use fixed HDMI ACR Ratio: %d\n",
+ edid_parsed.tv_name.vendor_name,
+ edid_parsed.tv_name.monitor_name,
+ fixed_acr_ration_val);
+ return 1;
+ }
+ }
+
+ return 0;
+}
+/*
+<-- end added by howayhuo
+*/
+
+void hdmi_set_audio_n_cts(unsigned int freq)
+{
+ unsigned int n, cts;
+
+ n = 128 * freq / 1000;
+#ifdef __KERNEL__
+{
+ unsigned int tmp;
+ unsigned int pll_clk;
+
+ pll_clk = auto_pll_divisor(DEV_I2S, GET_FREQ, 0, 0);
+ tmp = (vppif_reg32_in(AUDREGF_BASE_ADDR+0x70) & 0xF);
+
+ switch (tmp) {
+ case 0 ... 4:
+ tmp = 0x01 << tmp;
+ break;
+ case 9 ... 12:
+ tmp = 3 * (0x1 << (tmp-9));
+ break;
+ default:
+ tmp = 1;
+ break;
+ }
+
+ {
+ unsigned long long tmp2;
+ unsigned long long div2;
+ unsigned long mod;
+
+ tmp2 = g_vpp.hdmi_pixel_clock;
+ tmp2 = tmp2 * n * tmp;
+ div2 = pll_clk;
+ mod = do_div(tmp2, div2);
+ cts = tmp2;
+ }
+ DBGMSG("[HDMI] i2s %d,cts %d,reg 0x%x\n", pll_clk, cts,
+ vppif_reg32_in(AUDREGF_BASE_ADDR + 0x70));
+}
+
+ vppif_reg32_write(HDMI_AUD_N_20BITS, n);
+ if(use_fix_acr_ratio())
+ vppif_reg32_write(HDMI_AUD_ACR_RATIO, fixed_acr_ration_val);
+ else
+ vppif_reg32_write(HDMI_AUD_ACR_RATIO, cts - 1);
+#else
+#if 1
+ cts = g_vpp.hdmi_pixel_clock / 1000;
+#else
+ cts = vpp_get_base_clock(VPP_MOD_GOVRH) / 1000;
+#endif
+ vppif_reg32_write(HDMI_AUD_N_20BITS, n);
+ if(use_fix_acr_ratio())
+ vppif_reg32_write(HDMI_AUD_ACR_RATIO, fixed_acr_ration_val);
+ else
+ vppif_reg32_write(HDMI_AUD_ACR_RATIO, cts - 2);
+#endif
+
+#if 1 /* auto detect CTS */
+ vppif_reg32_write(HDMI_AUD_CTS_SELECT, 0);
+ cts = 0;
+#else
+ vppif_reg32_write(HDMI_AUD_CTS_SELECT, 1);
+#endif
+ vppif_reg32_write(HDMI_AUD_CTS_LOW_12BITS, cts & 0xFFF);
+ vppif_reg32_write(HDMI_AUD_CTS_HI_8BITS, (cts & 0xFF000) >> 12);
+
+ DBGMSG("[HDMI] set audio freq %d,n %d,cts %d,tmds %d\n",
+ freq, n, cts, g_vpp.hdmi_pixel_clock);
+}
+void hdmi_config_audio(vout_audio_t *info)
+{
+ unsigned int freq;
+
+ hdmi_info.channel = info->channel;
+ hdmi_info.freq = info->sample_rate;
+
+ /* enable ARF & ARFP clock */
+ REG32_VAL(PM_CTRL_BASE_ADDR + 0x254) |= (BIT4 | BIT3);
+ hdmi_tx_audio_infoframe_packet(info->channel - 1, info->sample_rate);
+ hdmi_audio_enable(VPP_FLAG_DISABLE);
+ vppif_reg32_write(HDMI_AUD_LAYOUT, (info->channel == 8) ? 1 : 0);
+ vppif_reg32_write(HDMI_AUD_2CH_ECO, 1);
+
+ switch (info->sample_rate) {
+ case 32000:
+ freq = 0x3;
+ break;
+ case 44100:
+ freq = 0x0;
+ break;
+ case 88200:
+ freq = 0x8;
+ break;
+ case 176400:
+ freq = 0xC;
+ break;
+ default:
+ case 48000:
+ freq = 0x2;
+ break;
+ case 96000:
+ freq = 0xA;
+ break;
+ case 192000:
+ freq = 0xE;
+ break;
+ case 768000:
+ freq = 0x9;
+ break;
+ }
+ vppif_reg32_out(REG_HDMI_AUD_CHAN_STATUS0, (freq << 24) + 0x4);
+ vppif_reg32_out(REG_HDMI_AUD_CHAN_STATUS1, 0x0);
+ vppif_reg32_out(REG_HDMI_AUD_CHAN_STATUS2, 0xb);
+ vppif_reg32_out(REG_HDMI_AUD_CHAN_STATUS3, 0x0);
+ vppif_reg32_out(REG_HDMI_AUD_CHAN_STATUS4, 0x0);
+ vppif_reg32_out(REG_HDMI_AUD_CHAN_STATUS5, 0x0);
+
+ hdmi_set_audio_n_cts(info->sample_rate);
+ vppif_reg32_write(HDMI_AUD_ACR_ENABLE, VPP_FLAG_ENABLE);
+ vppif_reg32_write(HDMI_AUD_AIPCLK_RATE, 0);
+ hdmi_audio_enable(hdmi_get_plugin() ?
+ VPP_FLAG_ENABLE : VPP_FLAG_DISABLE);
+
+}
+
+void hdmi_config_video(hdmi_info_t *info)
+{
+ hdmi_set_output_colfmt(info->outfmt);
+ hdmi_tx_avi_infoframe_packet(info->outfmt, info->vic);
+ hdmi_tx_vendor_specific_infoframe_packet();
+}
+
+void hdmi_set_option(unsigned int option)
+{
+ vdo_color_fmt colfmt;
+ int temp;
+
+ hdmi_set_dvi_enable((option & EDID_OPT_HDMI) ?
+ VPP_FLAG_DISABLE : VPP_FLAG_ENABLE);
+ hdmi_audio_enable((option & EDID_OPT_AUDIO) ?
+ VPP_FLAG_ENABLE : VPP_FLAG_DISABLE);
+
+ colfmt = hdmi_get_output_colfmt();
+ switch (colfmt) {
+ case VDO_COL_FMT_YUV422H:
+ temp = option & EDID_OPT_YUV422;
+ break;
+ case VDO_COL_FMT_YUV444:
+ temp = option & EDID_OPT_YUV444;
+ break;
+ default:
+ temp = 1;
+ break;
+ }
+ if (temp == 0) {
+ hdmi_set_output_colfmt(VDO_COL_FMT_ARGB);
+ DBG_MSG("[HDMI] TV not support %s,use default RGB\n",
+ vpp_colfmt_str[colfmt]);
+ }
+ DBG_MSG("[HDMI] set option(8-HDMI,6-AUDIO) 0x%x\n", option);
+}
+
+void hdmi_config(hdmi_info_t *info)
+{
+ vout_audio_t audio_info;
+ int h_porch;
+ int delay_cfg;
+ vpp_clock_t clock;
+
+ vppif_reg32_write(HDMI_HDEN, 0);
+ vppif_reg32_write(HDMI_INFOFRAME_SELECT, 0);
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_RDY, 0);
+ hdmi_config_video(info);
+
+ govrh_get_tg(p_govrh, &clock);
+ h_porch = clock.total_pixel_of_line - clock.end_pixel_of_active; /*fp*/
+ delay_cfg = 47 - h_porch;
+ if (delay_cfg <= 0)
+ delay_cfg = 1;
+ h_porch = clock.begin_pixel_of_active; /* bp */
+ h_porch = (h_porch - (delay_cfg + 1) - 26) / 32;
+ if (h_porch <= 0)
+ h_porch = 1;
+ if (h_porch >= 8)
+ h_porch = 0;
+ hdmi_reg32_write(HDMI_CP_DELAY, delay_cfg);
+ vppif_reg32_write(HDMI_HORIZ_BLANK_MAX_PCK, h_porch);
+ DBGMSG("[HDMI] H blank max pck %d,delay %d\n", h_porch, delay_cfg);
+
+ audio_info.fmt = 16;
+ audio_info.channel = info->channel;
+ audio_info.sample_rate = info->freq;
+ hdmi_config_audio(&audio_info);
+
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_ADDR, 0);
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_LEN, 2);
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_RDY, 1);
+
+ hdmi_set_option(info->option);
+ vppif_reg32_write(HDMI_TRE_EN,
+ (g_vpp.hdmi_pixel_clock < 40000000) ? 3 : 2);
+}
+
+/*----------------------- Module API --------------------------------------*/
+void hdmi_set_cp_enable(vpp_flag_t enable)
+{
+ if (!g_vpp.hdmi_cp_enable)
+ enable = 0;
+
+ if (hdmi_cp)
+ hdmi_cp->enable(enable);
+
+#ifdef __KERNEL__
+ if (hdmi_cp && hdmi_cp->poll) {
+ vpp_irqproc_del_work(VPP_INT_GOVRH_VBIS, (void *)hdmi_cp->poll);
+ if (enable)
+ vpp_irqproc_work(VPP_INT_GOVRH_VBIS,
+ (void *)hdmi_cp->poll, 0, 0, 0);
+ }
+#endif
+}
+
+int hdmi_check_cp_int(void)
+{
+ int ret = 0;
+
+ if (hdmi_cp)
+ ret = hdmi_cp->interrupt();
+ return ret;
+}
+
+void hdmi_get_bksv(unsigned int *bksv)
+{
+ if (hdmi_cp)
+ hdmi_cp->get_bksv(bksv);
+}
+
+#ifdef __KERNEL__
+void hdmi_hotplug_notify(int plug_status)
+{
+ if (g_vpp.hdmi_disable)
+ return;
+ if (g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ vpp_netlink_notify_plug(VPP_VOUT_ALL, 0);
+ vpp_netlink_notify_plug(VPP_VOUT_ALL, 1);
+ return;
+ }
+ vpp_netlink_notify_plug(VPP_VOUT_NUM_HDMI, plug_status);
+}
+#else
+#define hdmi_hotplug_notify
+#endif
+
+int hdmi_check_plugin(int hotplug)
+{
+ static int last_plugin = -1;
+ int plugin;
+ int flag;
+
+ if (g_vpp.hdmi_disable)
+ return 0;
+
+ plugin = hdmi_get_plugin();
+ hdmi_clear_plug_status();
+#ifdef __KERNEL__
+ /* disable HDMI before change clock */
+ if (plugin == 0) {
+ hdmi_set_enable(0);
+ hdmi_set_power_down(1);
+ }
+ vpp_set_clock_enable(DEV_HDMII2C, plugin, 1);
+ vpp_set_clock_enable(DEV_HDCE, plugin, 1);
+
+ /* slow down clock for plugout */
+ flag = (auto_pll_divisor(DEV_HDMILVDS, GET_FREQ, 0, 0)
+ == 8000000) ? 0 : 1;
+ if ((plugin != flag) && !g_vpp.virtual_display) {
+ int pixclk;
+
+ pixclk = (plugin) ? g_vpp.hdmi_pixel_clock : 8000000;
+ auto_pll_divisor(DEV_HDMILVDS, SET_PLLDIV, 0, pixclk);
+ }
+#endif
+ if (last_plugin != plugin) {
+ DPRINT("[HDMI] HDMI plug%s,hotplug %d\n", (plugin) ?
+ "in" : "out", hotplug);
+ last_plugin = plugin;
+ }
+#if 0 /* Denzel test */
+ if (plugin == 0)
+ hdmi_set_dvi_enable(VPP_FLAG_ENABLE);
+#endif
+ return plugin;
+}
+
+void hdmi_reg_dump(void)
+{
+ DPRINT("========== HDMI register dump ==========\n");
+ vpp_reg_dump(REG_HDMI_BEGIN, REG_HDMI_END - REG_HDMI_BEGIN);
+ vpp_reg_dump(REG_HDMI2_BEGIN, REG_HDMI2_END - REG_HDMI2_BEGIN);
+
+ DPRINT("---------- HDMI common ----------\n");
+ DPRINT("enable %d,hden %d,reset %d,dvi %d\n",
+ vppif_reg32_read(HDMI_ENABLE), vppif_reg32_read(HDMI_HDEN),
+ vppif_reg32_read(HDMI_RESET),
+ vppif_reg32_read(HDMI_DVI_MODE_ENABLE));
+ DPRINT("colfmt %d,conv 422 %d,hsync low %d,vsync low %d\n",
+ vppif_reg32_read(HDMI_OUTPUT_FORMAT),
+ vppif_reg32_read(HDMI_CONVERT_YUV422),
+ vppif_reg32_read(HDMI_HSYNC_LOW_ACTIVE),
+ vppif_reg32_read(HDMI_VSYNC_LOW_ACTIVE));
+ DPRINT("dbg bus sel %d,state mach %d\n",
+ vppif_reg32_read(HDMI_DBG_BUS_SELECT),
+ vppif_reg32_read(HDMI_STATE_MACHINE_STATUS));
+ DPRINT("eep reset %d,encode %d,eess %d\n",
+ vppif_reg32_read(HDMI_EEPROM_RESET),
+ vppif_reg32_read(HDMI_ENCODE_ENABLE),
+ vppif_reg32_read(HDMI_EESS_ENABLE));
+ DPRINT("verify pj %d,auth test %d,cipher %d\n",
+ vppif_reg32_read(HDMI_VERIFY_PJ_ENABLE),
+ vppif_reg32_read(HDMI_AUTH_TEST_KEY),
+ vppif_reg32_read(HDMI_CIPHER_1_1));
+ DPRINT("preamble %d\n", vppif_reg32_read(HDMI_PREAMBLE));
+
+ DPRINT("---------- HDMI hotplug ----------\n");
+ DPRINT("plug %s\n", vppif_reg32_read(HDMI_HOTPLUG_IN) ? "in" : "out");
+ DPRINT("plug in enable %d, status %d\n",
+ vppif_reg32_read(HDMI_HOTPLUG_IN_INT),
+ vppif_reg32_read(HDMI_HOTPLUG_IN_STS));
+ DPRINT("plug out enable %d, status %d\n",
+ vppif_reg32_read(HDMI_HOTPLUG_OUT_INT),
+ vppif_reg32_read(HDMI_HOTPLUG_OUT_STS));
+ DPRINT("debounce detect %d,sample %d\n",
+ vppif_reg32_read(HDMI_DEBOUNCE_DETECT),
+ vppif_reg32_read(HDMI_DEBOUNCE_SAMPLE));
+
+ DPRINT("---------- I2C ----------\n");
+ DPRINT("enable %d,exit FSM %d,key read %d\n",
+ vppif_reg32_read(HDMI_I2C_ENABLE),
+ vppif_reg32_read(HDMI_FORCE_EXIT_FSM),
+ vppif_reg32_read(HDMI_KEY_READ_WORD));
+ DPRINT("clk divid %d,rd data 0x%x,wr data 0x%x\n",
+ vppif_reg32_read(HDMI_I2C_CLK_DIVIDER),
+ vppif_reg32_read(HDMI_RD_DATA),
+ vppif_reg32_read(HDMI_WR_DATA));
+ DPRINT("start %d,stop %d,wr avail %d\n",
+ vppif_reg32_read(HDMI_SW_START_REQ),
+ vppif_reg32_read(HDMI_SW_STOP_REQ),
+ vppif_reg32_read(HDMI_WR_DATA_AVAIL));
+ DPRINT("status %d,sw read %d,sw i2c req %d\n",
+ vppif_reg32_read(HDMI_I2C_STATUS),
+ vppif_reg32_read(HDMI_SW_READ),
+ vppif_reg32_read(HDMI_SW_I2C_REQ));
+
+ DPRINT("---------- AUDIO ----------\n");
+ DPRINT("enable %d,sub pck %d,spflat %d\n",
+ vppif_reg32_read(HDMI_AUD_ENABLE),
+ vppif_reg32_read(HDMI_AUD_SUB_PACKET),
+ vppif_reg32_read(HDMI_AUD_SPFLAT));
+ DPRINT("aud pck insert reset %d,enable %d,delay %d\n",
+ vppif_reg32_read(HDMI_AUD_PCK_INSERT_RESET),
+ vppif_reg32_read(HDMI_AUD_PCK_INSERT_ENABLE),
+ vppif_reg32_read(HDMI_AUD_INSERT_DELAY));
+ DPRINT("avmute set %d,clr %d,pixel repete %d\n",
+ vppif_reg32_read(HDMI_AVMUTE_SET_ENABLE),
+ vppif_reg32_read(HDMI_AVMUTE_CLR_ENABLE),
+ vppif_reg32_read(HDMI_AUD_PIXEL_REPETITION));
+ DPRINT("acr ratio %d,acr enable %d,mute %d\n",
+ vppif_reg32_read(HDMI_AUD_ACR_RATIO),
+ vppif_reg32_read(HDMI_AUD_ACR_ENABLE),
+ vppif_reg32_read(HDMI_AUD_MUTE));
+ DPRINT("layout %d,pwr save %d,n 20bits %d\n",
+ vppif_reg32_read(HDMI_AUD_LAYOUT),
+ vppif_reg32_read(HDMI_AUD_PWR_SAVING),
+ vppif_reg32_read(HDMI_AUD_N_20BITS));
+ DPRINT("cts low 12 %d,hi 8 %d,cts sel %d\n",
+ vppif_reg32_read(HDMI_AUD_CTS_LOW_12BITS),
+ vppif_reg32_read(HDMI_AUD_CTS_HI_8BITS),
+ vppif_reg32_read(HDMI_AUD_CTS_SELECT));
+ DPRINT("aipclk rate %d\n", vppif_reg32_read(HDMI_AUD_AIPCLK_RATE));
+
+ DPRINT("---------- INFOFRAME ----------\n");
+ DPRINT("sel %d,hor blank pck %d\n",
+ vppif_reg32_read(HDMI_INFOFRAME_SELECT),
+ vppif_reg32_read(HDMI_HORIZ_BLANK_MAX_PCK));
+ DPRINT("fifo1 ready %d,addr 0x%x,len %d\n",
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO1_RDY),
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO1_ADDR),
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO1_LEN));
+ DPRINT("fifo2 ready %d,addr 0x%x,len %d\n",
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO2_RDY),
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO2_ADDR),
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO2_LEN));
+ DPRINT("wr strobe %d,rd strobe %d,fifo addr %d\n",
+ vppif_reg32_read(HDMI_INFOFRAME_WR_STROBE),
+ vppif_reg32_read(HDMI_INFOFRAME_RD_STROBE),
+ vppif_reg32_read(HDMI_INFOFRAME_FIFO_ADDR));
+
+ {
+ int i;
+ unsigned int buf[32];
+
+ for (i = 0; i <= vppif_reg32_read(HDMI_INFOFRAME_FIFO1_LEN); i++) {
+ DPRINT("----- infoframe %d -----\n", i);
+ hdmi_read_fifo(i, buf, 32);
+ vpp_reg_dump((unsigned int) buf, 32);
+ }
+ }
+
+ DPRINT("---------- HDMI test ----------\n");
+ DPRINT("ch0 enable %d, data 0x%x\n",
+ vppif_reg32_read(HDMI_CH0_TEST_MODE_ENABLE),
+ vppif_reg32_read(HDMI_CH0_TEST_DATA));
+ DPRINT("ch1 enable %d, data 0x%x\n",
+ vppif_reg32_read(HDMI_CH1_TEST_MODE_ENABLE),
+ vppif_reg32_read(HDMI_CH1_TEST_DATA));
+ DPRINT("ch2 enable %d, data 0x%x\n",
+ vppif_reg32_read(HDMI_CH2_TEST_MODE_ENABLE),
+ vppif_reg32_read(HDMI_CH2_TEST_DATA));
+
+ if (hdmi_cp)
+ hdmi_cp->dump();
+}
+
+#ifdef CONFIG_PM
+static unsigned int *hdmi_pm_bk;
+static unsigned int *hdmi_pm_bk2;
+static unsigned int hdmi_pm_enable;
+static unsigned int hdmi_pm_enable2;
+static int hdmi_plug_enable = 0xFF;
+extern struct switch_dev vpp_sdev;
+static int hdmi_resume_plug_cnt;
+#define HDMI_RESUME_PLUG_MS 50
+#define HDMI_RESUME_PLUG_CNT 20
+static void hdmi_do_resume_plug(struct work_struct *ptr)
+{
+ vout_t *vo;
+ int plugin;
+ struct delayed_work *dwork = to_delayed_work(ptr);
+
+ plugin = hdmi_check_plugin(0);
+ vo = vout_get_entry(VPP_VOUT_NUM_HDMI);
+ vout_change_status(vo, VPP_VOUT_STS_PLUGIN, plugin);
+ if (plugin)
+ hdmi_hotplug_notify(1);
+ hdmi_resume_plug_cnt --;
+ if (hdmi_resume_plug_cnt && (vpp_sdev.state == 0))
+ schedule_delayed_work(dwork,
+ msecs_to_jiffies(HDMI_RESUME_PLUG_MS));
+}
+
+DECLARE_DELAYED_WORK(hdmi_resume_work, hdmi_do_resume_plug);
+
+void hdmi_suspend(int sts)
+{
+ vo_hdmi_set_clock(1);
+ switch (sts) {
+ case 0: /* disable module */
+ cancel_delayed_work_sync(&hdmi_resume_work);
+ hdmi_pm_enable = vppif_reg32_read(HDMI_ENABLE);
+ hdmi_reg32_write(HDMI_ENABLE, 0);
+ hdmi_pm_enable2 = vppif_reg32_read(HDMI_HDEN);
+ vppif_reg32_write(HDMI_HDEN, 0);
+ if (hdmi_plug_enable == 0xFF)
+ hdmi_plug_enable =
+ vppif_reg32_read(HDMI_HOTPLUG_OUT_INT);
+ hdmi_enable_plugin(0);
+ break;
+ case 1: /* disable tg */
+ break;
+ case 2: /* backup register */
+ hdmi_pm_bk = vpp_backup_reg(REG_HDMI_BEGIN,
+ (REG_HDMI_END - REG_HDMI_BEGIN));
+ hdmi_pm_bk2 = vpp_backup_reg(REG_HDMI2_BEGIN,
+ (REG_HDMI2_END - REG_HDMI2_BEGIN));
+ switch_set_state(&vpp_sdev, 0);
+ hdmi_resume_plug_cnt = 20;
+ break;
+ default:
+ break;
+ }
+ vo_hdmi_set_clock(0);
+}
+
+void hdmi_resume(int sts)
+{
+ vo_hdmi_set_clock(1);
+ switch (sts) {
+ case 0: /* restore register */
+ vpp_restore_reg(REG_HDMI_BEGIN,
+ (REG_HDMI_END - REG_HDMI_BEGIN), hdmi_pm_bk);
+ vpp_restore_reg(REG_HDMI2_BEGIN,
+ (REG_HDMI2_END - REG_HDMI2_BEGIN), hdmi_pm_bk2);
+ hdmi_pm_bk = 0;
+ hdmi_pm_bk2 = 0;
+ hdmi_config(&hdmi_info); /* re-config HDMI info frame */
+ if (g_vpp.hdmi_cp_p && hdmi_cp)
+ hdmi_cp->init();
+ break;
+ case 1: /* enable module */
+ hdmi_reg32_write(HDMI_ENABLE, hdmi_pm_enable);
+ vppif_reg32_write(HDMI_HDEN, hdmi_pm_enable2);
+ break;
+ case 2: /* enable tg */
+ hdmi_check_plugin(0);
+ hdmi_clear_plug_status();
+ hdmi_enable_plugin(hdmi_plug_enable);
+ hdmi_plug_enable = 0xFF;
+ if (vpp_sdev.state == 0) {
+ hdmi_resume_plug_cnt = HDMI_RESUME_PLUG_CNT;
+ schedule_delayed_work(&hdmi_resume_work,
+ msecs_to_jiffies(HDMI_RESUME_PLUG_MS));
+ }
+ break;
+ default:
+ break;
+ }
+ vo_hdmi_set_clock(0);
+}
+#else
+#define hdmi_suspend NULL
+#define hdmi_resume NULL
+#endif
+
+void hdmi_init(void)
+{
+ struct fb_videomode vmode;
+
+ g_vpp.hdmi_pixel_clock = vpp_get_base_clock(VPP_MOD_GOVRH);
+ g_vpp.hdmi_i2c_freq = HDMI_I2C_FREQ;
+ g_vpp.hdmi_i2c_udelay = 0;
+ g_vpp.hdmi_ctrl = 0x1000000;
+ g_vpp.hdmi_audio_pb4 = 0x0;
+ g_vpp.hdmi_audio_pb1 = 0x0;
+
+ hdmi_info.outfmt = hdmi_get_output_colfmt();
+ govrh_get_videomode(p_govrh, &vmode);
+ hdmi_info.vic = hdmi_get_vic(vmode.xres, vmode.yres, vmode.refresh,
+ (vmode.vmode & FB_VMODE_INTERLACED) ? 1 : 0);
+ hdmi_info.channel = 2;
+ hdmi_info.freq = 48000;
+ hdmi_info.option = EDID_OPT_AUDIO + EDID_OPT_HDMI;
+
+ if (g_vpp.govrh_preinit) {
+ DBGMSG("[HDMI] hdmi_init for uboot logo\n");
+ } else {
+ /* bit8-HDMI SDA,bit9-HDMI SCL,bit10-Hotplug,bit26-CEC */
+ /* GPIO disable GPIO function */
+ vppif_reg32_write(GPIO_BASE_ADDR+0x54, 0x4000700, 0, 0);
+ /* GPIO4 disable GPIO out */
+ vppif_reg32_write(GPIO_BASE_ADDR+0x494, 0x4000700, 0, 0);
+#if 0
+ /* Suspend GPIO output enable */
+ vppif_reg32_write(GPIO_BASE_ADDR+0x80, BIT23, 23, 1);
+ /* Suspend GPIO output high */
+ vppif_reg32_write(GPIO_BASE_ADDR+0xC0, BIT23, 23, 1);
+ /* Wake3 disable pull ctrl */
+ vppif_reg32_write(GPIO_BASE_ADDR+0x480, BIT19, 19, 0);
+#endif
+ vppif_reg32_write(HDMI_REG_LEVEL, 1);
+ vppif_reg32_write(HDMI_REG_UPDATE, 1);
+ vppif_reg32_write(HDMI_LDI_SHIFT_LEFT, 1);
+ vppif_reg32_out(REG_HDMI_STATUS, 0x0008c000);
+ vppif_reg32_out(REG_HDMI_TEST, 0x00450409);
+ vppif_reg32_out(REG_HDMI_TEST2, 0x00005022);
+ vppif_reg32_out(REG_HDMI_TEST3,
+ (g_vpp.hdmi_sp_mode) ? 0x00010100 : 0x00000100);
+ hdmi_set_enable(VPP_FLAG_DISABLE);
+ hdmi_set_dvi_enable(VPP_FLAG_DISABLE);
+ vppif_reg32_write(HDMI_CIPHER_1_1, 0);
+
+ vppif_reg32_write(HDMI_INFOFRAME_SRAM_ENABLE, 1);
+ vppif_reg32_write(HDMI_INFOFRAME_SELECT, 0);
+ vppif_reg32_write(HDMI_INFOFRAME_FIFO1_RDY, 0);
+
+ vppif_reg32_out(HDMI_BASE_ADDR+0x3ec, 0x0);
+ vppif_reg32_out(HDMI_BASE_ADDR+0x3e8, 0x1);
+
+ /* vppif_reg32_write(HDMI_AUD_LAYOUT, 1); */
+ hdmi_DDC_reset();
+ hdmi_DDC_set_freq(g_vpp.hdmi_i2c_freq);
+ vppif_reg32_write(HDMI_I2C_ENABLE, 1);
+ }
+ g_vpp.hdmi_init = 1;
+ if (hdmi_cp)
+ hdmi_cp->init();
+
+ acr_init();
+}
+#endif /* WMT_FTBLK_HDMI */
diff --git a/drivers/video/wmt/hdmi.h b/drivers/video/wmt/hdmi.h
new file mode 100644
index 00000000..83845100
--- /dev/null
+++ b/drivers/video/wmt/hdmi.h
@@ -0,0 +1,345 @@
+/*++
+ * linux/drivers/video/wmt/hdmi.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef HDMI_H
+/* To assert that only one occurrence is included */
+#define HDMI_H
+/*-------------------- MODULE DEPENDENCY -------------------------------------*/
+#include "vpp.h"
+
+/* following is the C++ header */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+/* #define HDMI_XXXX 1 *//*Example*/
+#define CONFIG_HDMI_INTERRUPT
+#define HDMI_PLUG_DELAY 300 /* plug stable delay ms */
+#define HDMI_CP_TIME 3 /* should more than 2 seconds */
+
+typedef enum {
+ HDMI_PACKET_NULL = 0x0,
+ HDMI_PACKET_AUD_CLOCK_REGEN = 0x1,
+ HDMI_PACKET_AUD_SAMPLE = 0x2,
+ HDMI_PACKET_GENERAL_CTRL = 0x3,
+ HDMI_PACKET_ACP = 0x4,
+ HDMI_PACKET_ISRC1 = 0x5,
+ HDMI_PACKET_ISRC2 = 0x6,
+ HDMI_PACKET_AUD_ONE_BIT_SAMPLE = 0x7,
+ HDMI_PACKET_AUD_DST = 0x8,
+ HDMI_PACKET_AUD_HBR = 0x9,
+ HDMI_PACKET_GAMUT_METADATA = 0xA,
+ HDMI_PACKET_INFOFRAME_VENDOR = 0x81,
+ HDMI_PACKET_INFOFRAME_AVI = 0x82,
+ HDMI_PACKET_INFOFRAME_SRC_PRODUCT_DESC = 0x83,
+ HDMI_PACKET_INFOFRAME_AUDIO = 0x84,
+ HDMI_PACKET_INFOFRAME_MPEG_SOURCE = 0x85
+} hdmi_packet_type_t;
+
+/* color depth (CD field) */
+#define HDMI_COLOR_DEPTH_24 0x4
+#define HDMI_COLOR_DEPTH_30 0x5
+#define HDMI_COLOR_DEPTH_36 0x6
+#define HDMI_COLOR_DEPTH_48 0x7
+
+/* pixel packing phase (PP field) */
+#define HDMI_PHASE_4 0x0
+#define HDMI_PHASE_1 0x1
+#define HDMI_PHASE_2 0x2
+#define HDMI_PHASE_3 0x3
+
+/* Scan Information (AVI InfoFrame S0/S1) */
+#define HDMI_SI_NO_DATA 0x0
+#define HDMI_SI_OVERSCAN 0x1
+#define HDMI_SI_UNDERSCAN 0x2
+
+/* Bar Info (AVI InfoFrame B0/B1) */
+#define HDMI_BI_DATA_NOT_VALID 0x0
+#define HDMI_BI_VERT_VALID 0x1
+#define HDMI_BI_HORIZ_VALID 0x2
+#define HDMI_BI_V_H_VALID 0x3
+
+/* Active Format Information Present (AVI InfoFrame A0) */
+#define HDMI_AF_INFO_NO_DATA 0x0
+#define HDMI_AF_INFO_PRESENT 0x1
+
+/* RGB or YCbCr (AVI InfoFrame Y0/Y1) */
+#define HDMI_OUTPUT_RGB 0x0
+#define HDMI_OUTPUT_YUV422 0x1
+#define HDMI_OUTPUT_YUV444 0x2
+
+/* Aspect Ratio (AVI InfoFrame R0/R1/R2/R3) */
+#define HDMI_ASPECT_RATIO_PIC 0x8
+#define HDMI_ASPECT_RATIO_4_3 0x9
+#define HDMI_ASPECT_RATIO_16_9 0xA
+#define HDMI_ASPECT_RATIO_14_9 0xB
+
+/* Picture Aspect Ratio (AVI InfoFrame M0/M1) */
+#define HDMI_PIC_ASPECT_NO_DATA 0x0
+#define HDMI_PIC_ASPECT_4_3 0x1
+#define HDMI_PIC_ASPECT_16_9 0x2
+
+/* Colorimetry (AVI InfoFrame C0/C1) */
+#define HDMI_COLORIMETRY_NO 0x0
+#define HDMI_COLORIMETRY_ITU601 0x1
+#define HDMI_COLORIMETRY_ITU709 0x2
+
+/* Non-uniform Picture Scaling (AVI InfoFrame SC0/SC1) */
+#define HDMI_NUSCALE_NO 0x0
+#define HDMI_NUSCALE_HOR 0x1
+#define HDMI_NUSCALE_VERT 0x2
+#define HDMI_NUSCALE_HOR_VERT 0x3
+
+/* Pixel Repetition (AVI InfoFrame PR0/PR1/PR2/PR3) */
+#define HDMI_PIXEL_REP_NO 0x0
+#define HDMI_PIXEL_REP_2 0x1
+#define HDMI_PIXEL_REP_3 0x2
+#define HDMI_PIXEL_REP_4 0x3
+#define HDMI_PIXEL_REP_5 0x4
+#define HDMI_PIXEL_REP_6 0x5
+#define HDMI_PIXEL_REP_7 0x6
+#define HDMI_PIXEL_REP_8 0x7
+#define HDMI_PIXEL_REP_9 0x8
+#define HDMI_PIXEL_REP_10 0x9
+
+/* Video Code */
+typedef enum {
+ HDMI_UNKNOW = 0,
+ HDMI_640x480p60_4x3,
+ HDMI_720x480p60_4x3,
+ HDMI_720x480p60_16x9,
+ HDMI_1280x720p60_16x9,
+ HDMI_1920x1080i60_16x9,
+ HDMI_1440x480i60_4x3,
+ HDMI_1440x480i60_16x9,
+ HDMI_1440x240p60_4x3,
+ HDMI_1440x240p60_16x9,
+ HDMI_2880x480i60_4x3, /* 10 */
+ HDMI_2880x480i60_16x9,
+ HDMI_2880x240p60_4x3,
+ HDMI_2880x240p60_16x9,
+ HDMI_1440x480p60_4x3,
+ HDMI_1440x480p60_16x9,
+ HDMI_1920x1080p60_16x9,
+ HDMI_720x576p50_4x3,
+ HDMI_720x576p50_16x9,
+ HDMI_1280x720p50_16x9,
+ HDMI_1920x1080i50_16x9, /* 20 */
+ HDMI_1440x576i50_4x3,
+ HDMI_1440x576i50_16x9,
+ HDMI_1440x288p50_4x3,
+ HDMI_1440x288p50_16x9,
+ HDMI_2880x576i50_4x3,
+ HDMI_2880x576i50_16x9,
+ HDMI_2880x288p50_4x3,
+ HDMI_2880x288p50_16x9,
+ HDMI_1440x576p50_4x3,
+ HDMI_1440x576p50_16x9, /* 30 */
+ HDMI_1920x1080p50_16x9,
+ HDMI_1920x1080p24_16x9,
+ HDMI_1920x1080p25_16x9,
+ HDMI_1920x1080p30_16x9,
+ HDMI_VIDEO_CODE_MAX
+} hdmi_video_code_t;
+
+/* Audio Channel Count (Audio InfoFrame CC0/CC1/CC2) */
+typedef enum {
+ HDMI_AUD_CHAN_REF_STM = 0,
+ HDMI_AUD_CHAN_2CH,
+ HDMI_AUD_CHAN_3CH,
+ HDMI_AUD_CHAN_4CH,
+ HDMI_AUD_CHAN_5CH,
+ HDMI_AUD_CHAN_6CH,
+ HDMI_AUD_CHAN_7CH,
+ HDMI_AUD_CHAN_8CH
+} hdmi_audio_channel_count_t;
+
+/* Audio Coding type (Audio InfoFrame CT0/CT1/CT2/CT3) */
+#define HDMI_AUD_TYPE_REF_STM 0x0
+#define HDMI_AUD_TYPE_PCM 0x1
+#define HDMI_AUD_TYPE_AC3 0x2
+#define HDMI_AUD_TYPE_MPEG1 0x3
+#define HDMI_AUD_TYPE_MP3 0x4
+#define HDMI_AUD_TYPE_MPEG2 0x5
+#define HDMI_AUD_TYPE_AAC 0x6
+#define HDMI_AUD_TYPE_DTS 0x7
+#define HDMI_AUD_TYPE_ATRAC 0x8
+
+/* Audio Sample size (Audio InfoFrame SS0/SS1) */
+#define HDMI_AUD_SAMPLE_REF_STM 0x0
+#define HDMI_AUD_SAMPLE_16 0x1
+#define HDMI_AUD_SAMPLE_20 0x2
+#define HDMI_AUD_SAMPLE_24 0x3
+
+/* Audio Sample frequency (Audio InfoFrame SF0/SF1/SF2) */
+#define HDMI_AUD_FREQ_REF_STM 0x0
+#define HDMI_AUD_FREQ_32K 0x1
+#define HDMI_AUD_FREQ_44_1K 0x2
+#define HDMI_AUD_FREQ_48K 0x3
+#define HDMI_AUD_FREQ_88_2K 0x4
+#define HDMI_AUD_FREQ_96K 0x5
+#define HDMI_AUD_FREQ_176_4K 0x6
+#define HDMI_AUD_FREQ_192K 0x7
+
+/* 3D_Structure */
+#define HDMI_3D_STRUCTURE_FRAME_PACKING 0x0
+#define HDMI_3D_STRUCTURE_FIELD_ALTERNATIVE 0x1
+#define HDMI_3D_STRUCTURE_LINE_ALTERNATIVE 0x2
+#define HDMI_3D_STRUCTURE_SIDE_BY_SIDE_FULL 0x3
+#define HDMI_3D_STRUCTURE_L_DEPTH 0x4
+#define HDMI_3D_STRUCTURE_L_DEP_GRA_GDEP 0x5
+#define HDMI_3D_STRUCTURE_TOP_AND_BOTTOM 0x6
+#define HDMI_3D_STRUCTURE_SIZE_BY_SIZE_HALF 0x8
+
+/*-------------------- EXPORTED PRIVATE TYPES---------------------------------*/
+/* typedef void hdmi_xxx_t; *//*Example*/
+typedef struct {
+ /* video */
+ vdo_color_fmt outfmt;
+ int vic;
+
+ /* audio */
+ int channel;
+ int freq;
+
+ /* option */
+ int option;
+
+} hdmi_info_t;
+
+#define HDMI_VIC_INTERLACE BIT(0)
+#define HDMI_VIC_PROGRESS 0
+#define HDMI_VIC_4x3 BIT(1)
+#define HDMI_VIC_16x9 0
+
+typedef struct {
+ unsigned short resx;
+ unsigned short resy;
+ char freq;
+ char option;
+} hdmi_vic_t;
+
+typedef struct {
+ void (*init)(void);
+ void (*enable)(int on);
+ int (*poll)(void);
+ void (*dump)(void);
+ int (*interrupt)(void);
+ void (*get_bksv)(unsigned int *bksv);
+} hdmi_cp_t;
+
+/*-------------------- EXPORTED PRIVATE VARIABLES ---------------------------*/
+#ifdef VPP_C
+#define EXTERN
+
+const hdmi_vic_t hdmi_vic_info[HDMI_VIDEO_CODE_MAX] = {
+ { 0, 0, 0, 0 }, /* HDMI_UNKNOW = 0 */
+ { 640, 480, 60, HDMI_VIC_4x3 }, /* HDMI_640x480p60_4x3 */
+ { 720, 480, 60, HDMI_VIC_4x3 }, /* HDMI_720x480p60_4x3 */
+ { 720, 480, 60, 0 }, /* HDMI_720x480p60_16x9 */
+ { 1280, 720, 60, 0 }, /* HDMI_1280x720p60_16x9 */
+ { 1920, 1080, 60, HDMI_VIC_INTERLACE }, /* HDMI_1920x1080i60_16x9 */
+ { 720, 480, 60, HDMI_VIC_4x3 | HDMI_VIC_INTERLACE }, /*1440x480i60_4x3*/
+ { 720, 480, 60, HDMI_VIC_INTERLACE }, /* HDMI_1440x480i60_16x9 */
+ { 720, 240, 60, HDMI_VIC_4x3 }, /* HDMI_1440x240p60_4x3 */
+ { 720, 240, 60, 0 }, /* HDMI_1440x240p60_16x9 */
+ { 2880, 480, 60, HDMI_VIC_4x3 | HDMI_VIC_INTERLACE },/*2880x480i60_4x3*/
+ { 2880, 480, 60, HDMI_VIC_INTERLACE }, /* HDMI_2880x480i60_16x9 */
+ { 2880, 240, 60, HDMI_VIC_4x3 }, /* HDMI_2880x240p60_4x3 */
+ { 2880, 240, 60, 0 }, /* HDMI_2880x240p60_16x9 */
+ { 1440, 480, 60, HDMI_VIC_4x3 }, /* HDMI_1440x480p60_4x3 */
+ { 1440, 480, 60, 0 }, /* HDMI_1440x480p60_16x9 */
+ { 1920, 1080, 60, 0 }, /* HDMI_1920x1080p60_16x9 */
+ { 720, 576, 50, HDMI_VIC_4x3 }, /* HDMI_720x576p50_4x3 */
+ { 720, 576, 50, 0 }, /* HDMI_720x576p50_16x9 */
+ { 1280, 720, 50, 0 }, /* HDMI_1280x720p50_16x9 */
+ { 1920, 1080, 50, HDMI_VIC_INTERLACE }, /* HDMI_1920x1080i50_16x9, 20 */
+ { 720, 576, 50, HDMI_VIC_INTERLACE | HDMI_VIC_4x3 }, /*1440x576i50_4x3*/
+ { 720, 576, 50, HDMI_VIC_INTERLACE }, /* HDMI_1440x576i50_16x9 */
+ { 720, 288, 50, HDMI_VIC_4x3 }, /* HDMI_1440x288p50_4x3 */
+ { 720, 288, 50, 0 }, /* HDMI_1440x288p50_16x9 */
+ { 2880, 576, 50, HDMI_VIC_INTERLACE | HDMI_VIC_4x3}, /*2880x576i50_4x3*/
+ { 2880, 576, 50, HDMI_VIC_INTERLACE }, /* HDMI_2880x576i50_16x9 */
+ { 2880, 288, 50, HDMI_VIC_4x3 }, /* HDMI_2880x288p50_4x3 */
+ { 2880, 288, 50, 0 }, /* HDMI_2880x288p50_16x9 */
+ { 1440, 576, 50, HDMI_VIC_4x3 }, /* HDMI_1440x576p50_4x3 */
+ { 1440, 576, 50, 0 }, /* HDMI_1440x576p50_16x9, // 30 */
+ { 1920, 1080, 50, 0 }, /* HDMI_1920x1080p50_16x9 */
+ { 1920, 1080, 24, 0 }, /* HDMI_1920x1080p24_16x9 */
+ { 1920, 1080, 25, 0 }, /* HDMI_1920x1080p25_16x9 */
+ { 1920, 1080, 30, 0 } /* HDMI_1920x1080p30_16x9 */
+};
+#else
+#define EXTERN extern
+
+EXTERN const hdmi_vic_t hdmi_vic_info[HDMI_VIDEO_CODE_MAX];
+#endif /* ifdef HDMI_C */
+
+EXTERN hdmi_cp_t *hdmi_cp;
+EXTERN int hdmi_ri_tm_cnt;
+EXTERN hdmi_info_t hdmi_info;
+
+/* EXTERN int hdmi_xxx; *//*Example*/
+#undef EXTERN
+
+/*--------------------- EXPORTED PRIVATE MACROS -----------------------------*/
+/* #define HDMI_XXX_YYY xxxx *//*Example*/
+/*--------------------- EXPORTED PRIVATE FUNCTIONS -------------------------*/
+/* extern void hdmi_xxx(void); *//*Example*/
+
+void hdmi_init(void);
+void hdmi_reg_dump(void);
+void hdmi_audio_enable(vpp_flag_t enable);
+void hdmi_audio_mute(vpp_flag_t enable);
+void hdmi_set_enable(vpp_flag_t enable);
+void hdmi_set_avmute(vpp_flag_t mute);
+void hdmi_set_dvi_enable(vpp_flag_t enable);
+void hdmi_set_cp_enable(vpp_flag_t enable);
+int hdmi_check_cp_int(void);
+void hdmi_config(hdmi_info_t *info);
+int hdmi_DDC_read(char addr, int index, char *buf, int length);
+int hdmi_get_plugin(void);
+int hdmi_get_plug_status(void);
+void hdmi_clear_plug_status(void);
+void hdmi_suspend(int sts);
+void hdmi_resume(int sts);
+void hdmi_set_power_down(int pwrdn);
+void hdmi_enable_plugin(int enable);
+vdo_color_fmt hdmi_get_output_colfmt(void);
+void hdmi_set_sync_low_active(vpp_flag_t hsync, vpp_flag_t vsync);
+void hdmi_get_sync_polar(int *hsync_hi, int *vsync_hi);
+int hdmi_check_plugin(int hotplug);
+void hdmi_set_cypher(int func);
+void hdmi_set_option(unsigned int option);
+void hdmi_get_bksv(unsigned int *bksv);
+int hdmi_check_cp_dev_cnt(void);
+void hdmi_hotplug_notify(int plug_status);
+void hdmi_tx_vendor_specific_infoframe_packet(void);
+int hdmi_get_vic(int resx, int resy, int fps, int interlace);
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* ifndef HDMI_H */
+
diff --git a/drivers/video/wmt/hdmi_cp.c b/drivers/video/wmt/hdmi_cp.c
new file mode 100644
index 00000000..10fdead0
--- /dev/null
+++ b/drivers/video/wmt/hdmi_cp.c
@@ -0,0 +1,45 @@
+/*++
+ * linux/drivers/video/wmt/hdmi_cp.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define HDMI_CP_C
+// #define DEBUG
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "hdmi_cp.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define HDMI_CP_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx hdmi_cp_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in hdmi_cp.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int hdmi_cp_xxx; *//*Example*/
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void hdmi_cp_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+
diff --git a/drivers/video/wmt/hdmi_cp.h b/drivers/video/wmt/hdmi_cp.h
new file mode 100644
index 00000000..55ab1bab
--- /dev/null
+++ b/drivers/video/wmt/hdmi_cp.h
@@ -0,0 +1,63 @@
+/*++
+ * linux/drivers/video/wmt/hdmi_cp.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef HDMI_CP_H
+/* To assert that only one occurrence is included */
+#define HDMI_CP_H
+/*-------------------- MODULE DEPENDENCY -------------------------------------*/
+#include "vpp.h"
+
+/* following is the C++ header */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+/* #define HDMI_CP_XXXX 1 *//*Example*/
+/* #define CONFIG_HDMI_CP_DISABLE */
+
+/*-------------------- EXPORTED PRIVATE TYPES---------------------------------*/
+/* typedef void hdmi_cp_xxx_t; *//*Example*/
+
+/*-------------------- EXPORTED PRIVATE VARIABLES ----------------------------*/
+#ifdef HDMI_CP_C
+#define EXTERN
+
+#else
+#define EXTERN extern
+
+#endif /* ifdef HDMI_CP_C */
+
+/* EXTERN int hdmi_cp_xxx; *//*Example*/
+#undef EXTERN
+
+/*--------------------- EXPORTED PRIVATE MACROS ------------------------------*/
+/* #define HDMI_CP_XXX_YYY xxxx *//*Example*/
+/*--------------------- EXPORTED PRIVATE FUNCTIONS --------------------------*/
+/* extern void hdmi_cp_xxx(void); *//*Example*/
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* ifndef HDMI_CP_H */
+
diff --git a/drivers/video/wmt/hw/wmt-cec-reg.h b/drivers/video/wmt/hw/wmt-cec-reg.h
new file mode 100644
index 00000000..e93939fd
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-cec-reg.h
@@ -0,0 +1,169 @@
+/*++
+ * linux/drivers/video/wmt/hw/wmt-cec-reg.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_CEC_REG_H
+#define WMT_CEC_REG_H
+
+#define WMT_FTBLK_CEC
+
+#define CEC_BASE_ADDR (LVDS_BASE_ADDR + 0x100)
+#define CEC_BASE2_ADDR (LVDS_BASE_ADDR + 0x200)
+
+#define REG_CEC_BEGIN (CEC_BASE_ADDR + 0x0)
+#define REG_CEC_ENABLE (CEC_BASE_ADDR + 0x0)
+#define REG_CEC_ENCODE_NUMBER (CEC_BASE_ADDR + 0x4)
+#define REG_CEC_ENCODE_HEADER (CEC_BASE_ADDR + 0x8)
+#define REG_CEC_ENCODE_DATA (CEC_BASE_ADDR + 0xC) /* Data1(0x0C) - Data15(0x44) */
+#define REG_CEC_DECODE_RESET (CEC_BASE_ADDR + 0x48)
+#define REG_CEC_DECODE_START (CEC_BASE_ADDR + 0x4C)
+#define REG_CEC_DECODE_HEADER (CEC_BASE_ADDR + 0x50)
+#define REG_CEC_DECODE_DATA (CEC_BASE_ADDR + 0x54) /* Data1(0x54) - Data15(0x8C) */
+#define REG_CEC_WR_START_SET0 (CEC_BASE_ADDR + 0x90) /* val * CEC_CLK = 3.7 ms */
+#define REG_CEC_WR_START_SET1 (CEC_BASE_ADDR + 0x94) /* val * CEC_CLK = 4.5 ms */
+#define REG_CEC_WR_LOGIC0_SET0 (CEC_BASE_ADDR + 0x98) /* val * CEC_CLK = 0.6 ms */
+#define REG_CEC_WR_LOGIC0_SET1 (CEC_BASE_ADDR + 0x9C) /* val * CEC_CLK = 2.4 ms */
+#define REG_CEC_WR_LOGIC1_SET0 (CEC_BASE_ADDR + 0xA0) /* val * CEC_CLK = 1.5 ms */
+#define REG_CEC_WR_LOGIC1_SET1 (CEC_BASE_ADDR + 0xA4) /* val * CEC_CLK = 2.4 ms */
+#define REG_CEC_RD_START_L_SET0 (CEC_BASE_ADDR + 0xA8) /* val * CEC_CLK = 3.5 ms */
+#define REG_CEC_RD_START_R_SET0 (CEC_BASE_ADDR + 0xAC) /* val * CEC_CLK = 3.9 ms */
+#define REG_CEC_RD_START_L_SET1 (CEC_BASE_ADDR + 0xB0) /* val * CEC_CLK = 4.3 ms */
+#define REG_CEC_RD_START_R_SET1 (CEC_BASE_ADDR + 0xB4) /* val * CEC_CLK = 4.7 ms */
+#define REG_CEC_RD_LOGIC0_L_SET0 (CEC_BASE_ADDR + 0xB8) /* val * CEC_CLK = 1.3 ms */
+#define REG_CEC_RD_LOGIC0_R_SET0 (CEC_BASE_ADDR + 0xBC) /* val * CEC_CLK = 1.7 ms */
+#define REG_CEC_RD_LOGIC0_L_SET1 (CEC_BASE_ADDR + 0xC0) /* val * CEC_CLK = 2.05 ms */
+#define REG_CEC_RD_LOGIC0_R_SET1 (CEC_BASE_ADDR + 0xC4) /* val * CEC_CLK = 2.75 ms */
+#define REG_CEC_RD_LOGIC1_L_SET0 (CEC_BASE_ADDR + 0xC8) /* val * CEC_CLK = 0.4 ms */
+#define REG_CEC_RD_LOGIC1_R_SET0 (CEC_BASE_ADDR + 0xCC) /* val * CEC_CLK = 0.8 ms */
+#define REG_CEC_RD_LOGIC1_L_SET1 (CEC_BASE_ADDR + 0xD0) /* val * CEC_CLK = 2.05 ms */
+#define REG_CEC_RD_LOGIC1_R_SET1 (CEC_BASE_ADDR + 0xD4) /* val * CEC_CLK = 2.75 ms */
+#define REG_CEC_PHYSICAL_ADDR (CEC_BASE_ADDR + 0xD8)
+#define REG_CEC_LOGICAL_ADDR (CEC_BASE_ADDR + 0xDC)
+#define REG_CEC_WR_RETRY (CEC_BASE_ADDR + 0xE0)
+#define REG_CEC_FREE_3X (CEC_BASE_ADDR + 0xE4)
+#define REG_CEC_WR_SET0_ERROR (CEC_BASE_ADDR + 0xE8) /* val * CEC_CLK = 2.25 ms */
+#define REG_CEC_WR_SET1_ERROR (CEC_BASE_ADDR + 0xEC)
+#define REG_CEC_REJECT (CEC_BASE_ADDR + 0xF0)
+#define REG_CEC_RD_L_SET0_ERROR (CEC_BASE_ADDR + 0xF4) /* val * CEC_CLK = 1.82 ms */
+#define REG_CEC_RD_R_SET1_ERROR (CEC_BASE_ADDR + 0xF8) /* val * CEC_CLK = 2.38 ms */
+#define REG_CEC_RD_L_ERROR (CEC_BASE_ADDR + 0xFC) /* val * CEC_CLK = 2.87 ms */
+
+#define REG_CEC_RX_TRIG_RANGE (CEC_BASE2_ADDR + 0x00)
+#define REG_CEC_RX_SAMPLE_L_RANGE (CEC_BASE2_ADDR + 0x04) /* val * CEC_CLK = 0.85 ms */
+#define REG_CEC_RX_SAMPLE_R_RANGE (CEC_BASE2_ADDR + 0x08) /* val * CEC_CLK = 1.25 ms */
+#define REG_CEC_COMP_DISABLE (CEC_BASE2_ADDR + 0x0C)
+#define REG_CEC_ERR_HANDLE_DISABLE (CEC_BASE2_ADDR + 0x10)
+#define REG_CEC_STATUS (CEC_BASE2_ADDR + 0x14)
+#define REG_CEC_INT_ENABLE (CEC_BASE2_ADDR + 0x18)
+#define REG_CEC_DECODE_FULL_DISABLE (CEC_BASE2_ADDR + 0x1C)
+#define REG_CEC_STATUS4_DISABLE (CEC_BASE2_ADDR + 0x20)
+#define REG_CEC_RD_ENCODE_ENABLE (CEC_BASE2_ADDR + 0x24)
+#define REG_CEC_DIS_ARB_CHECK (CEC_BASE2_ADDR + 0x28)
+#define REG_CEC_END (CEC_BASE2_ADDR + 0x28)
+
+/* REG_CEC_ENABLE,0x0 */
+#define CEC_WR_START REG_CEC_ENABLE, BIT0, 0
+
+/* REG_CEC_ENCODE_NUMBER,0x4 */
+#define CEC_WR_NUM REG_CEC_ENCODE_NUMBER, 0xFF, 0x0
+
+/* REG_CEC_ENCODE_HEADER,0x8 */
+#define CEC_WR_HEADER_ACK REG_CEC_ENCODE_HEADER, BIT0, 0
+#define CEC_WR_HEADER_EOM REG_CEC_ENCODE_HEADER, BIT1, 1
+#define CEC_WR_HEADER_DATA REG_CEC_ENCODE_HEADER, 0xFF0, 4
+
+/* REG_CEC_ENCODE_DATA,Data1(0x0C) - Data15(0x44) */
+#define CEC_WR_DATA_ACK REG_CEC_ENCODE_DATA, BIT0, 0
+#define CEC_WR_DATA_EOM REG_CEC_ENCODE_DATA, BIT1, 1
+#define CEC_WR_DATA REG_CEC_ENCODE_DATA, 0xFF0, 4
+
+/* REG_CEC_DECODE_RESET,0x48 */
+#define CEC_FINISH_RESET REG_CEC_DECODE_RESET, BIT0, 0
+
+/* REG_CEC_DECODE_START,0x4C */
+#define CEC_RD_START REG_CEC_DECODE_START, BIT0, 0
+#define CEC_RD_ALL_ACK REG_CEC_DECODE_START, BIT1, 1
+#define CEC_RD_FINISH REG_CEC_DECODE_START, BIT2, 2
+
+/* REG_CEC_DECODE_HEADER,0x50 */
+#define CEC_RD_HEADER_ACK REG_CEC_DECODE_HEADER, BIT0, 0
+#define CEC_RD_HEADER_EOM REG_CEC_DECODE_HEADER, BIT1, 1
+#define CEC_RD_HEADER_DATA REG_CEC_DECODE_HEADER, 0xFF0, 4
+
+/* REG_CEC_DECODE_DATA,Data1(0x54) - Data15(0x8C) */
+#define CEC_RD_DATA_ACK REG_CEC_DECODE_DATA, BIT0, 0
+#define CEC_RD_DATA_EOM REG_CEC_DECODE_DATA, BIT1, 1
+#define CEC_RD_DATA REG_CEC_DECODE_DATA, 0xFF0, 4
+
+/* REG_CEC_LOGICAL_ADDR,0xDC */
+#define CEC_LOGICAL_ADDR1 REG_CEC_LOGICAL_ADDR, 0xF, 0
+#define CEC_LOGICAL_ADDR2 REG_CEC_LOGICAL_ADDR, 0xF0, 4
+#define CEC_LOGICAL_ADDR3 REG_CEC_LOGICAL_ADDR, 0xF00, 8
+#define CEC_LOGICAL_ADDR4 REG_CEC_LOGICAL_ADDR, 0xF000, 12
+#define CEC_LOGICAL_ADDR5 REG_CEC_LOGICAL_ADDR, 0xF0000, 16
+#define CEC_ADDR_VALID1 REG_CEC_LOGICAL_ADDR, BIT24, 24
+#define CEC_ADDR_VALID2 REG_CEC_LOGICAL_ADDR, BIT25, 25
+#define CEC_ADDR_VALID3 REG_CEC_LOGICAL_ADDR, BIT26, 26
+#define CEC_ADDR_VALID4 REG_CEC_LOGICAL_ADDR, BIT27, 27
+#define CEC_ADDR_VALID5 REG_CEC_LOGICAL_ADDR, BIT28, 28
+
+/* REG_CEC_WR_RETRY,0xE0 */
+#define CEC_WR_RETRY REG_CEC_WR_RETRY, 0xF, 0
+
+/* REG_CEC_FREE_3X,0xE4 */
+#define CEC_FREE_3X REG_CEC_FREE_3X, 0xF, 0
+#define CEC_FREE_5X REG_CEC_FREE_3X, 0xF00, 8
+#define CEC_FREE_7X REG_CEC_FREE_3X, 0xF0000, 16
+
+/* REG_CEC_REJECT,0xF0 */
+#define CEC_REJECT_NEXT_DECODE REG_CEC_REJECT, BIT0, 0 /*read enable*/
+
+/* REG_CEC_COMP_DISABLE,0x0C */
+#define CEC_COMP_DISABLE REG_CEC_COMP_DISABLE, BIT0, 0
+
+/* REG_CEC_ERR_HANDLE_DISABLE,0x10 */
+#define CEC_ERR_HANDLE_DISABLE REG_CEC_ERR_HANDLE_DISABLE, BIT0, 0
+#define CEC_NO_ACK_DISABLE REG_CEC_ERR_HANDLE_DISABLE, BIT1, 1
+
+/* REG_CEC_STATUS,0x14 */
+#define CEC_R1_ENCODE_OK REG_CEC_STATUS, BIT0, 0 /* write finish */
+#define CEC_R1_DECODE_OK REG_CEC_STATUS, BIT1, 1 /* read finish */
+#define CEC_R1_ERROR REG_CEC_STATUS, BIT2, 2 /* read error */
+#define CEC_R1_ARB_FAIL REG_CEC_STATUS, BIT3, 3 /* wr arb fail */
+#define CEC_R1_NO_ACK REG_CEC_STATUS, BIT4, 4 /* wr no ack */
+
+/* REG_CEC_DECODE_FULL_DISABLE,0x1C */
+#define CEC_DECODE_FULL_DISABLE REG_CEC_DECODE_FULL_DISABLE, BIT0, 0
+
+/* REG_CEC_STATUS4_DISABLE,0x20 */
+#define CEC_STATUS4_START_DISABLE REG_CEC_STATUS4_DISABLE, BIT0, 0
+#define CEC_STATUS4_LOGIC0_DISABLE REG_CEC_STATUS4_DISABLE, BIT1, 1
+#define CEC_STATUS4_LOGIC1_DISABLE REG_CEC_STATUS4_DISABLE, BIT2, 2
+
+/* REG_CEC_RD_ENCODE_ENABLE,0x24 */
+#define CEC_RD_ENCODE_ENABLE REG_CEC_RD_ENCODE_ENABLE, BIT0, 0 /* 1 : read self write and all dest data */
+
+/* REG_CEC_DIS_ARB_CHECK,0x28 */
+#define CEC_ARB_CHECK_DISABLE REG_CEC_DIS_ARB_CHECK, BIT0, 0 /* 1 : disable arbitration check */
+
+#endif /* WMT_CEC_REG_H */
+
diff --git a/drivers/video/wmt/hw/wmt-govrh-reg.h b/drivers/video/wmt/hw/wmt-govrh-reg.h
new file mode 100644
index 00000000..4e2e2341
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-govrh-reg.h
@@ -0,0 +1,362 @@
+/*++
+ * linux/drivers/video/wmt/hw/wmt-govrh-reg.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_GOVRH_REG_H
+#define WMT_GOVRH_REG_H
+
+/* feature */
+#define WMT_FTBLK_GOVRH
+#ifndef CONFIG_UBOOT
+#define WMT_FTBLK_GOVRH_CURSOR
+#endif
+#define WMT_FTBLK_GOVRH2
+
+#define GOVRH_FRAMEBUF_ALIGN 128
+
+struct govrh_regs {
+ /* base1 */
+ unsigned int cur_addr; /* 0x00 */
+ unsigned int cur_width;
+ unsigned int cur_fb_width;
+ unsigned int cur_vcrop;
+ unsigned int cur_hcrop; /* 0x10 */
+ union {
+ unsigned int val;
+ struct {
+ unsigned int start : 11;
+ unsigned int reserved : 5;
+ unsigned int end : 11;
+ } b;
+ } cur_hcoord; /* 0x14 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int start : 11;
+ unsigned int reserved : 5;
+ unsigned int end : 11;
+ } b;
+ } cur_vcoord; /* 0x18 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int enable : 1;
+ unsigned int reserved : 7;
+ unsigned int out_field : 1; /* 0:frame,1-field */
+ } b;
+ } cur_status; /* 0x1C */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int colkey : 24;
+ unsigned int enable : 1;
+ unsigned int invert : 1;
+ unsigned int reserved : 2;
+ unsigned int alpha : 1;
+ } b;
+ } cur_color_key; /* 0x20 */
+
+ unsigned int reserved[3];
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int rgb : 1;
+ unsigned int yuv422 : 1;
+ } b;
+ } dvo_pix; /* 0x30 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int delay : 14;
+ unsigned int inv : 1;
+ } b;
+ } dvo_dly_sel; /* 0x34 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int cur_enable : 1;
+ unsigned int mem_enable : 1;
+ unsigned int reserved : 7;
+ unsigned int err_sts : 1;
+ unsigned int reserved2 : 6;
+ unsigned int cur_sts : 1;
+ unsigned int mem_sts : 1;
+ } b;
+ } interrupt; /* 0x38 */
+
+ unsigned int dvo_blank_data;
+ unsigned int dirpath; /* 0x40 */
+ union {
+ unsigned int val;
+ struct {
+ unsigned int v : 8;
+ unsigned int u : 8;
+ unsigned int y : 8;
+ } b;
+ } saturation; /* 0x44 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int enable : 1;
+ unsigned int format : 1; /* 0:YCbCr, 1:RGB */
+ } b;
+ } saturation_enable; /* 0x48 */
+
+ unsigned int reserved2[13];
+ union {
+ unsigned int val;
+ struct {
+ unsigned int enable : 1;
+ unsigned int reserved : 7;
+ unsigned int h264 : 1;
+ } b;
+ } mif; /* 0x80 */
+
+ unsigned int colfmt; /* 0x84, 0:422,1:420 */
+ unsigned int srcfmt; /* 0x88, 0:frame,1:field */
+ unsigned int dstfmt; /* 0x8C, 0:frame,1:field */
+ unsigned int ysa; /* 0x90 */
+ unsigned int csa;
+ unsigned int pixwid;
+ unsigned int bufwid;
+ unsigned int vcrop; /* 0xA0 */
+ unsigned int hcrop;
+ unsigned int fhi;
+ unsigned int colfmt2; /* 0xAC, 1-444,other refer 0x84 */
+ unsigned int ysa2; /* 0xB0 */
+ unsigned int csa2;
+ union {
+ unsigned int val;
+ struct {
+ unsigned int req_num : 8; /* Y & RGB */
+ unsigned int req_num_c : 8; /* C */
+ unsigned int frame_enable : 1;
+ } b;
+ } mif_frame_mode; /* 0xB8 */
+
+ unsigned int reserved3[10];
+ union {
+ unsigned int val;
+ struct {
+ unsigned int update : 1;
+ unsigned int reserved : 7;
+ unsigned int level : 1; /* 0:level 1, 1:level2 */
+ } b;
+ } sts; /* 0xE4 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int fixed : 1; /* 0-top, 1-bottom */
+ unsigned int enable : 1;
+ } b;
+ } swfld; /* 0xE8 */
+
+ unsigned int reserved4[5];
+ /* base2 */
+ union {
+ unsigned int val;
+ struct {
+ unsigned int enable : 1;
+ unsigned int reserved : 7;
+ unsigned int mode : 1; /* 0-frame,1-field */
+ } b;
+ } tg_enable; /* 0x100 */
+
+ unsigned int read_cyc;
+ unsigned int h_allpxl;
+ unsigned int v_allln;
+ unsigned int actln_bg; /* 0x110 */
+ unsigned int actln_end;
+ unsigned int actpx_bg;
+ unsigned int actpx_end;
+ unsigned int vbie_line; /* 0x120 */
+ unsigned int pvbi_line;
+ unsigned int hdmi_vbisw;
+ unsigned int hdmi_hsynw;
+ union {
+ unsigned int val;
+ struct {
+ unsigned int offset : 12;
+ unsigned int reserved : 4;
+ unsigned int field_invert : 1;
+ } b;
+ } vsync_offset; /* 0x130 */
+
+ unsigned int field_status; /* 0x134, 1-BOTTOM,0-TOP */
+ unsigned int reserved5[1]; /* 0x138 */
+ union {
+ unsigned int val;
+ struct {
+ unsigned int mode : 3; /* 011-frame packing progressive format,111-frame packing interlace format */
+ unsigned int inv_filed_polar : 1;
+ unsigned int blank_value : 16;
+ unsigned int reserved : 11;
+ unsigned int addr_sel : 1; /* in frame packing interlace mode */
+ } b;
+ } hdmi_3d; /* 0x13C */
+
+ unsigned int reserved5_2[2];
+ union {
+ unsigned int val;
+ struct {
+ unsigned int outwidth : 1; /* 0-24bit,1-12bit */
+ unsigned int hsync_polar : 1; /* 0-active high,1-active low */
+ unsigned int enable : 1;
+ unsigned int vsync_polar : 1; /* 0-active high,1-active low */
+ unsigned int reserved : 4;
+ unsigned int rgb_swap : 2; /* 0-RGB[7:0],1-RGB[0:7],2-BGR[7:0],3-BGR[0:7] */
+ unsigned int reserved2 : 6;
+ unsigned int blk_dis : 1; /* 0-Blank Data,1-Embeded sync CCIR656 */
+ } b;
+ } dvo_set; /* 0x148 */
+
+ unsigned int reserved6;
+ union {
+ unsigned int val;
+ struct {
+ unsigned int enable : 1;
+ unsigned int reserved1 : 7;
+ unsigned int mode : 1;
+ unsigned int reserved2 : 7;
+ unsigned int inversion : 1;
+ } b;
+ } cb_enable; /* 0x150 */
+
+ unsigned int reserved7;
+ unsigned int h_allpxl2;
+ unsigned int v_allln2;
+ unsigned int actln_bg2; /* 0x160 */
+ unsigned int actln_end2;
+ unsigned int actpx_bg2;
+ unsigned int actpx_end2;
+ unsigned int vbie_line2; /* 0x170 */
+ unsigned int pvbi_line2;
+ unsigned int hdmi_vbisw2;
+ unsigned int hdmi_hsynw2;
+ union {
+ unsigned int val;
+ struct {
+ unsigned int outwidth : 1; /* 0-24bit,1-12bit */
+ unsigned int hsync_polar : 1; /* 0-active high,1-active low */
+ unsigned int enable : 1;
+ unsigned int vsync_polar : 1; /* 0-active high,1-active low */
+ } b;
+ } lvds_ctrl; /* 0x180 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int pix : 2; /* 0-YUV444,1-RGB,2-YUV422,3-RGB */
+ } b;
+ } lvds_ctrl2; /* 0x184 */
+
+ unsigned int reserved_dac[12];
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int praf : 8;
+ unsigned int pbaf : 8;
+ unsigned int yaf : 8;
+ } b;
+ } contrast; /* 0x1B8 */
+
+ unsigned int brightness;
+ unsigned int dmacsc_coef0; /* 0x1C0 */
+ unsigned int dmacsc_coef1;
+ unsigned int dmacsc_coef2;
+ unsigned int dmacsc_coef3;
+ unsigned int dmacsc_coef4; /* 0x1D0 */
+ unsigned int reserved8;
+ unsigned int dmacsc_coef5;
+ unsigned int dmacsc_coef6;
+ union {
+ unsigned int val;
+ struct {
+ unsigned int mode : 1; /* 1: YUV2RGB, 0: RGB2YUV */
+ unsigned int clamp : 1; /* 0:Y,1:Y-16 */
+ } b;
+ } csc_mode; /* 0x1E0 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int dvo : 1;
+ unsigned int vga : 1;
+ unsigned int reserved1 : 1;
+ unsigned int dac_clkinv : 1;
+ unsigned int blank_zero : 1;
+ unsigned int disp : 1;
+ unsigned int lvds : 1;
+ unsigned int hdmi : 1;
+ unsigned int rgb_mode : 2; /* 0-YUV, 1-RGB24, 2-1555, 3-565 */
+ } b;
+ } yuv2rgb; /* 0x1E4 */
+
+ unsigned int h264_input_en; /* 0x1E8 */
+ unsigned int reserved9;
+ unsigned int lvds_clkinv; /* 0x1F0 */
+ unsigned int hscale_up; /* 0x1F4 */
+ union {
+ unsigned int val;
+ struct {
+ unsigned int mode : 3; /* 0:888,1:555,2:666,3:565,4:original */
+ unsigned int reserved : 5;
+ unsigned int ldi : 1; /* 0:shift right,1:shift left */
+ } b;
+ } igs_mode; /* 0x1F8 */
+
+ union {
+ unsigned int val;
+ struct {
+ unsigned int mode : 3; /* 0:888,1:555,2:666,3:565,4:original */
+ unsigned int reserved : 5;
+ unsigned int ldi : 1; /* 0:shift right,1:shift left */
+ } b;
+ } igs_mode2; /* 0x1FC */
+};
+
+/* GOVRH */
+#define REG_GOVRH_BASE1_BEGIN (GOVRH_BASE1_ADDR+0x00)
+#define REG_GOVRH_YSA (GOVRH_BASE1_ADDR+0x90)
+#define REG_GOVRH_CSA (GOVRH_BASE1_ADDR+0x94)
+#define REG_GOVRH_BASE1_END (GOVRH_BASE1_ADDR+0xe8)
+#define REG_GOVRH_BASE2_BEGIN (GOVRH_BASE2_ADDR+0x00)
+#define REG_GOVRH_BASE2_END (GOVRH_BASE2_ADDR+0xFC)
+
+/* GOVRH2 */
+#define REG_GOVRH2_BASE1_BEGIN (GOVRH2_BASE1_ADDR+0x00)
+#define REG_GOVRH2_YSA (GOVRH2_BASE1_ADDR+0x90)
+#define REG_GOVRH2_CSA (GOVRH2_BASE1_ADDR+0x94)
+#define REG_GOVRH2_BASE1_END (GOVRH2_BASE1_ADDR+0xe8)
+#define REG_GOVRH2_BASE2_BEGIN (GOVRH2_BASE2_ADDR+0x00)
+#define REG_GOVRH2_BASE2_END (GOVRH2_BASE2_ADDR+0xFC)
+
+#endif /* WMT_GOVRH_REG_H */
diff --git a/drivers/video/wmt/hw/wmt-hdmi-reg.h b/drivers/video/wmt/hw/wmt-hdmi-reg.h
new file mode 100644
index 00000000..919d6ab3
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-hdmi-reg.h
@@ -0,0 +1,266 @@
+/*++
+ * linux/drivers/video/wmt/hw/wmt-hdmi-reg.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_HDMI_REG_H
+#define WMT_HDMI_REG_H
+
+#define WMT_FTBLK_HDMI
+
+#define HDMI_BASE_ADDR (HDMI_TRANSMITTE_BASE_ADDR + 0xC000)
+
+/* HDMI registers */
+#define REG_HDMI_BEGIN (HDMI_BASE_ADDR + 0x100)
+#define REG_HDMI_CTRL (HDMI_BASE_ADDR + 0x120)
+#define REG_HDMI_I2C_CTRL (HDMI_BASE_ADDR + 0x124)
+#define REG_HDMI_I2C_CTRL2 (HDMI_BASE_ADDR + 0x128)
+#define REG_HDMI_GENERAL_CTRL (HDMI_BASE_ADDR + 0x280)
+#define REG_HDMI_INFOFRAME_CTRL (HDMI_BASE_ADDR + 0x284)
+#define REG_HDMI_AUD_INSERT_CTRL (HDMI_BASE_ADDR + 0x294)
+#define REG_HDMI_AUD_RATIO (HDMI_BASE_ADDR + 0x29c)
+#define REG_HDMI_AUD_ENABLE (HDMI_BASE_ADDR + 0x2a0)
+#define REG_HDMI_AUD_MODE (HDMI_BASE_ADDR + 0x2ac)
+#define REG_HDMI_AUD_CHAN_STATUS0 (HDMI_BASE_ADDR + 0x390)
+#define REG_HDMI_AUD_CHAN_STATUS1 (HDMI_BASE_ADDR + 0x394)
+#define REG_HDMI_AUD_CHAN_STATUS2 (HDMI_BASE_ADDR + 0x398)
+#define REG_HDMI_AUD_CHAN_STATUS3 (HDMI_BASE_ADDR + 0x39c)
+#define REG_HDMI_AUD_CHAN_STATUS4 (HDMI_BASE_ADDR + 0x3a0)
+#define REG_HDMI_AUD_CHAN_STATUS5 (HDMI_BASE_ADDR + 0x3a4)
+#define REG_HDMI_AUD_SAMPLE_RATE1 (HDMI_BASE_ADDR + 0x3a8)
+#define REG_HDMI_AUD_SAMPLE_RATE2 (HDMI_BASE_ADDR + 0x3ac)
+
+/* HDMI info WR FIFO 0x3c0 - 0x3e0 */
+#define REG_HDMI_WR_FIFO_ADDR (HDMI_BASE_ADDR + 0x3c0)
+#define REG_HDMI_FIFO_CTRL (HDMI_BASE_ADDR + 0x3e4)
+#define REG_HDMI_CHANNEL_TEST (HDMI_BASE_ADDR + 0x3e8)
+#define REG_HDMI_HOTPLUG_DETECT (HDMI_BASE_ADDR + 0x3ec)
+#define REG_HDMI_HOTPLUG_DEBOUNCE (HDMI_BASE_ADDR + 0x3f0)
+#define REG_HDMI_TMDS_CTRL (HDMI_BASE_ADDR + 0x3f8)
+
+/* HDMI info RD FIFO 0x400 - 0x420 */
+#define REG_HDMI_RD_FIFO_ADDR (HDMI_BASE_ADDR + 0x400)
+
+#define REG_HDMI_END (HDMI_BASE_ADDR + 0x420)
+
+/* REG_HDMI_CTRL,0x120 */
+#define HDMI_EEPROM_RESET REG_HDMI_CTRL, BIT0, 0
+#define HDMI_ENCODE_ENABLE REG_HDMI_CTRL, BIT1, 1
+#define HDMI_HDEN REG_HDMI_CTRL, BIT2, 2
+#define HDMI_EESS_ENABLE REG_HDMI_CTRL, BIT3, 3
+#define HDMI_VERIFY_PJ_ENABLE REG_HDMI_CTRL, BIT4, 4
+#define HDMI_I2C_ENABLE REG_HDMI_CTRL, BIT5, 5
+#define HDMI_AUTH_TEST_KEY REG_HDMI_CTRL, BIT6, 6
+#define HDMI_CIPHER_1_1 REG_HDMI_CTRL, BIT8, 8
+#define HDMI_PREAMBLE REG_HDMI_CTRL, 0xF000, 12
+#define HDMI_ENCODE_WINDOW REG_HDMI_CTRL, 0x700000, 20
+
+/* REG_HDMI_I2C_CTRL,0x124 */
+#define HDMI_FORCE_EXIT_FSM REG_HDMI_I2C_CTRL, BIT7, 7
+#define HDMI_KEY_READ_WORD REG_HDMI_I2C_CTRL, 0xFF00, 8
+#define HDMI_I2C_SW_RESET REG_HDMI_I2C_CTRL, 0x8000, 15
+#define HDMI_I2C_CLK_DIVIDER REG_HDMI_I2C_CTRL, 0xFFFF0000, 16
+
+/* REG_HDMI_I2C_CTRL2,0x128 */
+#define HDMI_WR_DATA REG_HDMI_I2C_CTRL2, 0xFF, 0
+#define HDMI_RD_DATA REG_HDMI_I2C_CTRL2, 0xFF00, 8
+#define HDMI_SW_START_REQ REG_HDMI_I2C_CTRL2, BIT16, 16
+#define HDMI_SW_STOP_REQ REG_HDMI_I2C_CTRL2, BIT17, 17
+#define HDMI_WR_DATA_AVAIL REG_HDMI_I2C_CTRL2, BIT18, 18
+#define HDMI_I2C_STATUS REG_HDMI_I2C_CTRL2, BIT19, 19 /* 0-not using, 1-in using */
+#define HDMI_CP_KEY_REQ REG_HDMI_I2C_CTRL2, BIT20, 20
+#define HDMI_CP_KEY_READ REG_HDMI_I2C_CTRL2, BIT21, 21
+#define HDMI_CP_KEY_LAST REG_HDMI_I2C_CTRL2, BIT22, 22
+#define HDMI_CP_SRC_SEL REG_HDMI_I2C_CTRL2, BIT24, 24
+#define HDMI_SW_READ REG_HDMI_I2C_CTRL2, BIT25, 25
+#define HDMI_SW_I2C_REQ REG_HDMI_I2C_CTRL2, BIT26, 26
+#define HDMI_KSV_LIST_AVAIL REG_HDMI_I2C_CTRL2, BIT27, 27
+#define HDMI_KSV_VERIFY_DONE REG_HDMI_I2C_CTRL2, BIT28, 28
+
+/* REG_HDMI_GENERAL_CTRL,0x280 */
+#define HDMI_RESET REG_HDMI_GENERAL_CTRL, BIT0, 0
+#define HDMI_ENABLE REG_HDMI_GENERAL_CTRL, BIT1, 1
+#define HDMI_DVI_MODE_ENABLE REG_HDMI_GENERAL_CTRL, BIT6, 6
+#define HDMI_OUTPUT_FORMAT REG_HDMI_GENERAL_CTRL, 0x180, 7 /* 0-RGB,1-YUV444,2-YUV422 */
+#define HDMI_CONVERT_YUV422 REG_HDMI_GENERAL_CTRL, BIT9, 9
+#define HDMI_HSYNC_LOW_ACTIVE REG_HDMI_GENERAL_CTRL, BIT10, 10 /* 0-active high,1-active low */
+#define HDMI_DBG_BUS_SELECT REG_HDMI_GENERAL_CTRL, BIT11, 11 /* 0-before,1-after */
+#define HDMI_VSYNC_LOW_ACTIVE REG_HDMI_GENERAL_CTRL, BIT13, 13 /* 0-active high,1-active low */
+#define HDMI_CP_DELAY REG_HDMI_GENERAL_CTRL, 0x7F0000, 16 /* delay for CP after HSYNC raising edge */
+#define HDMI_VSYNC_384_ENABLE REG_HDMI_GENERAL_CTRL, BIT24, 24
+#define HDMI_VSYNC_385_507_ENABLE REG_HDMI_GENERAL_CTRL, BIT25, 25
+#define HDMI_VSYNC_650_ENABLE REG_HDMI_GENERAL_CTRL, BIT26, 26
+#define HDMI_STATE_MACHINE_STATUS REG_HDMI_GENERAL_CTRL, 0xF8000000, 27
+
+/* REG_HDMI_INFOFRAME_CTRL,0x284 */
+#define HDMI_INFOFRAME_SELECT REG_HDMI_INFOFRAME_CTRL, BIT0, 0 /* 0-fifo1,1-fifo2 */
+#define HDMI_INFOFRAME_FIFO1_RDY REG_HDMI_INFOFRAME_CTRL, BIT1, 1 /* Info frame FIFO 1 ready */
+#define HDMI_INFOFRAME_FIFO2_RDY REG_HDMI_INFOFRAME_CTRL, BIT2, 2 /* Info frame FIFO 2 ready */
+#define HDMI_INFOFRAME_FIFO1_ADDR REG_HDMI_INFOFRAME_CTRL, 0xF0, 4 /* Info frame FIFO 1 start address */
+#define HDMI_INFOFRAME_FIFO1_LEN REG_HDMI_INFOFRAME_CTRL, 0x1F00, 8 /* Info frame FIFO 1 length */
+#define HDMI_INFOFRAME_FIFO2_ADDR REG_HDMI_INFOFRAME_CTRL, 0xF0000, 16 /* Info frame FIFO 2 start address */
+#define HDMI_INFOFRAME_FIFO2_LEN REG_HDMI_INFOFRAME_CTRL, 0x1F00000, 20 /* Info frame FIFO 2 length */
+#define HDMI_HORIZ_BLANK_MAX_PCK REG_HDMI_INFOFRAME_CTRL, 0x70000000, 28 /* Max packets that insert during HSYNC */
+
+/* REG_HDMI_AUD_INSERT_CTRL,0x294 */
+#define HDMI_AUD_PCK_INSERT_RESET REG_HDMI_AUD_INSERT_CTRL, BIT0, 0
+#define HDMI_AUD_PCK_INSERT_ENABLE REG_HDMI_AUD_INSERT_CTRL, BIT1, 1
+#define HDMI_AVMUTE_SET_ENABLE REG_HDMI_AUD_INSERT_CTRL, BIT2, 2
+#define HDMI_AVMUTE_CLR_ENABLE REG_HDMI_AUD_INSERT_CTRL, BIT3, 3
+#define HDMI_AUD_INSERT_DELAY REG_HDMI_AUD_INSERT_CTRL, 0xFFF0, 4
+#define HDMI_AUD_PIXEL_REPETITION REG_HDMI_AUD_INSERT_CTRL, 0xC0000000, 30 /* 0-none,1-2 times,2-4 times */
+
+/* REG_HDMI_AUD_RATIO,0x29c */
+#define HDMI_AUD_ACR_RATIO REG_HDMI_AUD_RATIO, 0x0FFFFF00, 8
+#define HDMI_AUD_ACR_ENABLE REG_HDMI_AUD_RATIO, BIT28, 28
+#define HDMI_AUD_MUTE REG_HDMI_AUD_RATIO, BIT29, 29
+
+/* REG_HDMI_AUD_ENABLE,0x2a0 */
+#define HDMI_AUD_ENABLE REG_HDMI_AUD_ENABLE, BIT0, 0
+
+/* REG_HDMI_AUD_MODE,0x2ac */
+#define HDMI_AUD_SUB_PACKET REG_HDMI_AUD_MODE, 0xF, 0
+#define HDMI_AUD_SPFLAT REG_HDMI_AUD_MODE, 0xF0, 4
+#define HDMI_AUD_2CH_ECO REG_HDMI_AUD_MODE, BIT8, 8
+#define HDMI_AUD_LAYOUT REG_HDMI_AUD_MODE, BIT10, 10 /* 0-2 channel,1-8 channel */
+#define HDMI_AUD_PWR_SAVING REG_HDMI_AUD_MODE, BIT11, 11 /* 0-normal, 1-power saving */
+
+/* REG_HDMI_AUD_CHAN_STATUS0,0x390 */
+/* REG_HDMI_AUD_CHAN_STATUS1,0x394 */
+/* REG_HDMI_AUD_CHAN_STATUS2,0x398 */
+/* REG_HDMI_AUD_CHAN_STATUS3,0x39c */
+/* REG_HDMI_AUD_CHAN_STATUS4,0x3a0 */
+/* REG_HDMI_AUD_CHAN_STATUS5,0x3a4 */
+
+/* REG_HDMI_AUD_SAMPLE_RATE1,0x3a8 */
+#define HDMI_AUD_N_20BITS REG_HDMI_AUD_SAMPLE_RATE1, 0xFFFFF, 0
+#define HDMI_AUD_CTS_LOW_12BITS REG_HDMI_AUD_SAMPLE_RATE1, 0xFFF00000, 20
+
+/* REG_HDMI_AUD_SAMPLE_RATE2,0x3ac */
+#define HDMI_AUD_CTS_HI_8BITS REG_HDMI_AUD_SAMPLE_RATE2, 0xFF, 0
+#define HDMI_AUD_AIPCLK_RATE REG_HDMI_AUD_SAMPLE_RATE2, 0x30000000, 28 /* 0-N/2,1-N,2-N/4,3-N*2 */
+#define HDMI_AUD_CTS_SELECT REG_HDMI_AUD_SAMPLE_RATE2, BIT30, 30 /* 0-auto, 1-fixed from register */
+
+/* 0x3c0 - 0x3e0 : Info frame FIFO data */
+
+/* REG_HDMI_FIFO_CTRL,0x3e4 */
+#define HDMI_INFOFRAME_WR_STROBE REG_HDMI_FIFO_CTRL, BIT0, 0
+#define HDMI_INFOFRAME_RD_STROBE REG_HDMI_FIFO_CTRL, BIT1, 1
+#define HDMI_INFOFRAME_FIFO_ADDR REG_HDMI_FIFO_CTRL, 0xFF00, 8
+
+/* REG_HDMI_CHANNEL_TEST,0x3e8 */
+#define HDMI_CH1_TEST_MODE_ENABLE REG_HDMI_CHANNEL_TEST, BIT26, 26
+#define HDMI_CH1_TEST_DATA REG_HDMI_CHANNEL_TEST, 0x3FF0000, 16
+#define HDMI_CH0_TEST_MODE_ENABLE REG_HDMI_CHANNEL_TEST, BIT10, 10
+#define HDMI_CH0_TEST_DATA REG_HDMI_CHANNEL_TEST, 0x3FF, 0
+
+/* REG_HDMI_HOTPLUG_DETECT,0x3ec */
+#define HDMI_HOTPLUG_IN REG_HDMI_HOTPLUG_DETECT, BIT31, 31 /* 0-plug out,1-plug in */
+#define HDMI_HOTPLUG_OUT_STS REG_HDMI_HOTPLUG_DETECT, BIT25, 25
+#define HDMI_HOTPLUG_IN_STS REG_HDMI_HOTPLUG_DETECT, BIT24, 24
+#define HDMI_HOTPLUG_OUT_INT REG_HDMI_HOTPLUG_DETECT, BIT17, 17
+#define HDMI_HOTPLUG_IN_INT REG_HDMI_HOTPLUG_DETECT, BIT16, 16
+#define HDMI_CH2_TEST_MODE_ENABLE REG_HDMI_HOTPLUG_DETECT, BIT10, 10
+#define HDMI_CH2_TEST_DATA REG_HDMI_HOTPLUG_DETECT, 0x3FF, 0
+
+/* REG_HDMI_HOTPLUG_DEBOUNCE,0x3f0 */
+#define HDMI_DEBOUNCE_DETECT REG_HDMI_HOTPLUG_DEBOUNCE, 0x1FF0000, 16
+#define HDMI_DEBOUNCE_SAMPLE REG_HDMI_HOTPLUG_DEBOUNCE, 0xFF, 0
+
+/* REG_HDMI_TMDS_CTRL,0x3f8 */
+#define HDMI_CLOCK_SELECT REG_HDMI_TMDS_CTRL, BIT16, 16 /* 0-clk 1x, 1-clk 2x */
+#define HDMI_INFOFRAME_SRAM_ENABLE REG_HDMI_TMDS_CTRL, BIT10, 10
+#define HDMI_TMDS_TST_FORMAT REG_HDMI_TMDS_CTRL, BIT1, 1
+#define HDMI_TMDS_TST_ENABLE REG_HDMI_TMDS_CTRL, BIT0, 0
+
+/* 0x400 - 0x420 : HDMI info frame FIFO data (RO) */
+
+#define REG_HDMI2_BEGIN (HDMI_BASE2_ADDR + 0x00)
+#define REG_HDMI_STATUS (HDMI_BASE2_ADDR + 0x00)
+#define REG_HDMI_TEST (HDMI_BASE2_ADDR + 0x04)
+#define REG_HDMI_LEVEL (HDMI_BASE2_ADDR + 0x08)
+#define REG_HDMI_IGS (HDMI_BASE2_ADDR + 0x0C)
+#define REG_HDMI_SET (HDMI_BASE2_ADDR + 0x10)
+#define REG_HDMI_SET2 (HDMI_BASE2_ADDR + 0x14)
+#define REG_HDMI_DETECT (HDMI_BASE2_ADDR + 0x18)
+#define REG_HDMI_TEST2 (HDMI_BASE2_ADDR + 0x1C)
+#define REG_HDMI_TEST3 (HDMI_BASE2_ADDR + 0x20)
+#define REG_HDMI_DFTSET2 (HDMI_BASE2_ADDR + 0x24)
+#define REG_HDMI2_END (HDMI_BASE2_ADDR + 0x28)
+
+/* REG_HDMI_STATUS,0x00 */
+#define HDMI_INTERNAL_LDO REG_HDMI_STATUS, 0x80000, 19
+#define HDMI_TEST REG_HDMI_STATUS, 0xF00, 8
+#define HDMI_DUAL_CHANNEL REG_HDMI_STATUS, BIT4, 4
+#define HDMI_INV_CLK REG_HDMI_STATUS, BIT0, 0
+
+/* REG_HDMI_TEST,0x04 */
+#define HDMI_PLL_R_F REG_HDMI_TEST, BIT18, 18
+#define HDMI_PLL_CPSET REG_HDMI_TEST, 0x30000, 16
+#define HDMI_PLLCK_DLY REG_HDMI_TEST, 0x7000, 12
+#define HDMI_TRE_EN REG_HDMI_TEST, 0x600, 9
+#define HDMI_PD REG_HDMI_TEST, BIT8, 8
+#define HDMI_VBG_SEL REG_HDMI_TEST, 0xC, 2
+#define HDMI_DRV_PDMODE REG_HDMI_TEST, BIT0, 0
+
+/* REG_HDMI_LEVEL,0x08 */
+#define HDMI_REG_LEVEL REG_HDMI_LEVEL, BIT8, 8
+#define HDMI_REG_UPDATE REG_HDMI_LEVEL, BIT0, 0
+
+/* REG_HDMI_IGS,0x0C */
+#define HDMI_LDI_SHIFT_LEFT REG_HDMI_IGS, BIT8, 8 /* 0-shift right,1-shift left */
+#define HDMI_IGS_BPP_TYPE REG_HDMI_IGS, 0x7, 0 /* 0-888,1-555,2-666,3-565 */
+
+/* REG_HDMI_SET,0x10 */
+#define HDMI_VSYNC_POLAR_LO REG_HDMI_SET, BIT3, 3 /* 0-active high,1-active low */
+#define HDMI_DVO_ENABLE REG_HDMI_SET, BIT2, 2
+#define HDMI_HSYNC_POLAR_LO REG_HDMI_SET, BIT1, 1 /* 0-active high,1-active low */
+#define HDMI_OUT_DATA_12 REG_HDMI_SET, BIT0, 0 /* 0-24bit,1-12bit */
+
+/* REG_HDMI_SET2,0x14 */
+#define HDMI_COLFMT_YUV422 REG_HDMI_SET2, BIT1, 1 /* 0-RGB or YUV444,1-YUV422 */
+#define HDMI_COLFMT_RGB REG_HDMI_SET2, BIT0, 0
+
+/* REG_HDMI_DETECT,0x18 */
+#define HDMI_RSEN REG_HDMI_DETECT, BIT8, 8
+#define HDMI_PLL_READY REG_HDMI_DETECT, BIT0, 0
+
+/* REG_HDMI_TEST2,0x1C */
+#define HDMI_PLL_TSYNC REG_HDMI_TEST2, BIT0, 0
+#define HDMI_TP2S_TYPE REG_HDMI_TEST2, BIT1, 1
+#define HDMI_DIV_SEL REG_HDMI_TEST2, 0xC, 2
+#define HDMI_PD_V2I REG_HDMI_TEST2, BIT4, 4
+#define HDMI_VCO_SX REG_HDMI_TEST2, BIT5, 5
+#define HDMI_VCO_MODE REG_HDMI_TEST2, BIT6, 6
+#define HDMI_VSREF_SEL REG_HDMI_TEST2, 0x300, 8
+#define HDMI_MODE REG_HDMI_TEST2, BIT10, 10
+#define HDMI_PD_L2HA REG_HDMI_TEST2, BIT11, 11
+#define HDMI_PD_L2HB REG_HDMI_TEST2, BIT12, 12
+#define HDMI_L2HA_HSEN REG_HDMI_TEST2, BIT13, 13
+#define HDMI_RESA_EN REG_HDMI_TEST2, BIT14, 14
+#define HDMI_RESA_S REG_HDMI_TEST2, BIT15, 15
+#define HDMI_PLL_LPFS REG_HDMI_TEST2, 0x30000, 16
+
+/* REG_HDMI_DFTSET2,0x24 */
+#define HDMI_RESET_PLL REG_HDMI_DFTSET2, BIT16, 16
+
+#endif /* WMT_HDMI_REG_H */
+
diff --git a/drivers/video/wmt/hw/wmt-lvds-reg.h b/drivers/video/wmt/hw/wmt-lvds-reg.h
new file mode 100644
index 00000000..d258d6e8
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-lvds-reg.h
@@ -0,0 +1,92 @@
+/*++
+ * linux/drivers/video/wmt/hw/wmt-lvds-reg.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_LVDS_REG_H
+#define WMT_LVDS_REG_H
+
+#define WMT_FTBLK_LVDS
+
+#define REG_LVDS_BEGIN (LVDS_BASE_ADDR + 0x00)
+#define REG_LVDS_STATUS (LVDS_BASE_ADDR + 0x00)
+#define REG_LVDS_TEST (LVDS_BASE_ADDR + 0x04)
+#define REG_LVDS_LEVEL (LVDS_BASE_ADDR + 0x08)
+#define REG_LVDS_IGS (LVDS_BASE_ADDR + 0x0C)
+#define REG_LVDS_SET (LVDS_BASE_ADDR + 0x10)
+#define REG_LVDS_SET2 (LVDS_BASE_ADDR + 0x14)
+#define REG_LVDS_DETECT (LVDS_BASE_ADDR + 0x18)
+#define REG_LVDS_TEST2 (LVDS_BASE_ADDR + 0x1C)
+#define REG_LVDS_END (LVDS_BASE_ADDR + 0x1C)
+
+/* REG_LVDS_STATUS,0x00 */
+#define LVDS_TEST REG_LVDS_STATUS, 0xF00, 8
+#define LVDS_DUAL_CHANNEL REG_LVDS_STATUS, BIT4, 4
+#define LVDS_INV_CLK REG_LVDS_STATUS, BIT0, 0
+
+/* REG_LVDS_TEST,0x04 */
+#define LVDS_PLL_R_F REG_LVDS_TEST, BIT18, 18
+#define LVDS_PLL_CPSET REG_LVDS_TEST, 0x30000, 16
+#define LVDS_PLLCK_DLY REG_LVDS_TEST, 0x7000, 12
+#define LVDS_TRE_EN REG_LVDS_TEST, 0x600, 9
+#define LVDS_PD REG_LVDS_TEST, BIT8, 8
+#define LVDS_VBG_SEL REG_LVDS_TEST, 0xC, 2
+#define LVDS_DRV_PDMODE REG_LVDS_TEST, BIT0, 0
+
+/* REG_LVDS_LEVEL,0x08 */
+#define LVDS_REG_LEVEL REG_LVDS_LEVEL, BIT8, 8
+#define LVDS_REG_UPDATE REG_LVDS_LEVEL, BIT0, 0
+
+/* REG_LVDS_IGS,0x0C */
+#define LVDS_LDI_SHIFT_LEFT REG_LVDS_IGS, BIT8, 8 /* 0-shift right,1-shift left */
+#define LVDS_IGS_BPP_TYPE REG_LVDS_IGS, 0x7, 0 /* 0-888,1-555,2-666,3-565 */
+
+/* REG_LVDS_SET,0x10 */
+#define LVDS_VSYNC_POLAR_LO REG_LVDS_SET, BIT3, 3 /* 0-active high,1-active low */
+#define LVDS_DVO_ENABLE REG_LVDS_SET, BIT2, 2
+#define LVDS_HSYNC_POLAR_LO REG_LVDS_SET, BIT1, 1 /* 0-active high,1-active low */
+#define LVDS_OUT_DATA_12 REG_LVDS_SET, BIT0, 0 /* 0-24bit,1-12bit */
+
+/* REG_LVDS_SET2,0x14 */
+#define LVDS_COLFMT_YUV422 REG_LVDS_SET2, BIT1, 1 /* 0-RGB or YUV444,1-YUV422 */
+#define LVDS_COLFMT_RGB REG_LVDS_SET2, BIT0, 0
+
+/* REG_LVDS_DETECT,0x18 */
+#define LVDS_RSEN REG_LVDS_DETECT, BIT8, 8
+#define LVDS_PLL_READY REG_LVDS_DETECT, BIT0, 0
+
+/* REG_LVDS_TEST2,0x1C */
+#define LVDS_PLL_TSYNC REG_LVDS_TEST2, BIT0, 0
+#define LVDS_TP2S_TYPE REG_LVDS_TEST2, BIT1, 1
+#define LVDS_DIV_SEL REG_LVDS_TEST2, 0xC, 2
+#define LVDS_PD_V2I REG_LVDS_TEST2, BIT4, 4
+#define LVDS_VCO_SX REG_LVDS_TEST2, BIT5, 5
+#define LVDS_VCO_MODE REG_LVDS_TEST2, BIT6, 6
+#define LVDS_VSREF_SEL REG_LVDS_TEST2, 0x300, 8
+#define LVDS_MODE REG_LVDS_TEST2, BIT10, 10
+#define LVDS_PD_L2HA REG_LVDS_TEST2, BIT11, 11
+#define LVDS_PD_L2HB REG_LVDS_TEST2, BIT12, 12
+#define LVDS_L2HA_HSEN REG_LVDS_TEST2, BIT13, 13
+#define LVDS_RESA_EN REG_LVDS_TEST2, BIT14, 14
+#define LVDS_RESA_S REG_LVDS_TEST2, BIT15, 15
+#define LVDS_PLL_LPFS REG_LVDS_TEST2, 0x30000, 16
+
+#endif /* WMT_LVDS_REG_H */
diff --git a/drivers/video/wmt/hw/wmt-scl-reg.h b/drivers/video/wmt/hw/wmt-scl-reg.h
new file mode 100644
index 00000000..cabb75a7
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-scl-reg.h
@@ -0,0 +1,374 @@
+/*++
+ * linux/drivers/video/wmt/hw/wmt-scl-reg.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_SCL_REG_H
+#define WMT_SCL_REG_H
+
+/* feature */
+#define WMT_FTBLK_SCL
+
+/* constant */
+#define WMT_SCL_RCYC_MIN 0 /* 1T */
+#define WMT_SCL_H_DIV_MAX 8192
+#define WMT_SCL_V_DIV_MAX 8192
+#define WMT_SCL_FB_WIDTH_MAX 8192
+
+#define WMT_SCL_SCALE_DST_H_MAX 1920 /* bypass no limit */
+
+/* registers */
+#define REG_SCL_BASE1_BEGIN (SCL_BASE_ADDR + 0x00)
+#define REG_SCL_EN (SCL_BASE_ADDR + 0x00)
+#define REG_SCL_UPD (SCL_BASE_ADDR + 0x04)
+#define REG_SCL_SEL (SCL_BASE_ADDR + 0x08)
+#define REG_SCL_HXWIDTH (SCL_BASE_ADDR + 0x3c)
+#define REG_SCLR2_CTL (SCL_BASE_ADDR + 0x40)
+#define REG_SCLR2_YSA (SCL_BASE_ADDR + 0x44)
+#define REG_SCLR2_CSA (SCL_BASE_ADDR + 0x48)
+#define REG_SCLR2_H_SIZE (SCL_BASE_ADDR + 0x4C)
+#define REG_SCLR2_CROP (SCL_BASE_ADDR + 0x50)
+#define REG_ALFA_MD (SCL_BASE_ADDR + 0x54)
+#define REG_ALFA_FXD (SCL_BASE_ADDR + 0x58)
+#define REG_ALFA_COLORKEY (SCL_BASE_ADDR + 0x5C)
+#define REG_ALFA_COLORKEY_RGB (SCL_BASE_ADDR + 0x60)
+#define REG_SCL_VXWIDTH (SCL_BASE_ADDR + 0x70)
+#define REG_SCL_SCLUP_EN (SCL_BASE_ADDR + 0x74)
+
+#define REG_SCL_VSCALE1 (SCL_BASE_ADDR + 0x78)
+#define REG_SCL_VSCALE2 (SCL_BASE_ADDR + 0x7c)
+#define REG_SCL_VSCALE3 (SCL_BASE_ADDR + 0x80)
+#define REG_SCL_HSCALE1 (SCL_BASE_ADDR + 0x84)
+#define REG_SCL_HSCALE2 (SCL_BASE_ADDR + 0x88)
+#define REG_SCL_HSCALE3 (SCL_BASE_ADDR + 0x8c)
+#define REG_SCLR_REQ_NUM (SCL_BASE_ADDR + 0x90)
+#define REG_SCL_SCLDW (SCL_BASE_ADDR + 0x94)
+#define REG_SCL_426_SW (SCL_BASE_ADDR + 0x98)
+#define REG_SCL_VBYPASS (SCL_BASE_ADDR + 0x9C)
+/* SCL_TG */
+#define REG_SCL_TG_CTL (SCL_BASE_ADDR + 0xa0)
+#define REG_SCL_TG_TOTAL (SCL_BASE_ADDR + 0xa4)
+#define REG_SCL_TG_V_ACTIVE (SCL_BASE_ADDR + 0xa8)
+#define REG_SCL_TG_H_ACTIVE (SCL_BASE_ADDR + 0xac)
+#define REG_SCL_TG_VBI (SCL_BASE_ADDR + 0xb0)
+#define REG_SCL_TG_WATCHDOG (SCL_BASE_ADDR + 0xb4)
+#define REG_SCL_TG_STS (SCL_BASE_ADDR + 0xb8)
+#define REG_SCL_TG_GOVW (SCL_BASE_ADDR + 0xbc)
+/* SCLR */
+#define REG_SCLR_CTL (SCL_BASE_ADDR + 0xc0)
+#define REG_SCLR_YSA (SCL_BASE_ADDR + 0xc4)
+#define REG_SCLR_CSA (SCL_BASE_ADDR + 0xc8)
+#define REG_SCLR_H_SIZE (SCL_BASE_ADDR + 0xcc)
+#define REG_SCLR_CROP (SCL_BASE_ADDR + 0xd0)
+#define REG_SCLR_FIFO_CTL (SCL_BASE_ADDR + 0xd4)
+/* SCLW */
+#define REG_SCLW_CTL (SCL_BASE_ADDR + 0xe0)
+#define REG_SCLW_YSA (SCL_BASE_ADDR + 0xe4)
+#define REG_SCLW_CSA (SCL_BASE_ADDR + 0xe8)
+#define REG_SCLW_Y_TIME (SCL_BASE_ADDR + 0xec)
+#define REG_SCLW_C_TIME (SCL_BASE_ADDR + 0xf0)
+#define REG_SCLW_FF_CTL (SCL_BASE_ADDR + 0xf4)
+#define REG_SCLW_INT (SCL_BASE_ADDR + 0xf8)
+#define REG_SCL_TRUE_BILINEAR (SCL_BASE_ADDR + 0xfc)
+
+#define REG_SCL_BASE1_END (SCL_BASE_ADDR + 0xFC)
+
+/* SCL444 CSC */
+#define REG_SCL_BASE2_BEGIN (SCL_BASE2_ADDR + 0x00)
+#define REG_SCL_CSC_CTL (SCL_BASE2_ADDR + 0x00)
+#define REG_SCL_CSC1 (SCL_BASE2_ADDR + 0x04)
+#define REG_SCL_CSC2 (SCL_BASE2_ADDR + 0x08)
+#define REG_SCL_CSC3 (SCL_BASE2_ADDR + 0x0c)
+#define REG_SCL_CSC4 (SCL_BASE2_ADDR + 0x10)
+#define REG_SCL_CSC5 (SCL_BASE2_ADDR + 0x14)
+#define REG_SCL_CSC6 (SCL_BASE2_ADDR + 0x18)
+#define REG_SCL_ARGB_ALPHA (SCL_BASE2_ADDR + 0x1C)
+#define REG_SCL_IGS (SCL_BASE2_ADDR + 0x20)
+#define REG_SCL_R2_CSC (SCL_BASE2_ADDR + 0x24)
+#define REG_SCL_R2_CSC1 (SCL_BASE2_ADDR + 0x28)
+#define REG_SCL_R2_CSC2 (SCL_BASE2_ADDR + 0x2C)
+#define REG_SCL_R2_CSC3 (SCL_BASE2_ADDR + 0x30)
+#define REG_SCL_R2_CSC4 (SCL_BASE2_ADDR + 0x34)
+#define REG_SCL_R2_CSC5 (SCL_BASE2_ADDR + 0x38)
+#define REG_SCL_R2_CSC6 (SCL_BASE2_ADDR + 0x3C)
+#define REG_SCL_RECURSIVE_MODE (SCL_BASE2_ADDR + 0xA0)
+#define REG_SCL_FIELD_MODE (SCL_BASE2_ADDR + 0xC0)
+#define REG_SCL_DBLK_THRESHOLD (SCL_BASE2_ADDR + 0xC4)
+#define REG_SCL_FIELD_FLICKER (SCL_BASE2_ADDR + 0xC8)
+#define REG_SCL_FRAME_FLICKER (SCL_BASE2_ADDR + 0xCC)
+#define REG_SCL_READCYC_1T (SCL_BASE2_ADDR + 0xD0)
+#define REG_SCL_BASE2_END (SCL_BASE2_ADDR + 0xE0)
+
+/* REG_SCL_EN,0x00 */
+#define SCL_ALU_ENABLE REG_SCL_EN, BIT0, 0
+
+/* REG_SCL_UPD,0x04 */
+#define SCL_REG_UPDATE REG_SCL_UPD, BIT0, 0
+
+/* REG_SCL_SEL,0x08 */
+#define SCL_REG_LEVEL REG_SCL_SEL, BIT0, 0
+
+/* REG_SCL_HXWIDTH,0x3c */
+#define SCL_HXWIDTH REG_SCL_HXWIDTH, 0x1FFF, 0
+
+/* REG_SCLR2_CTL,0x40 */
+#define SCL_R2_MIF_EN REG_SCLR2_CTL, BIT0, 0
+#define SCL_R2_RGB_MODE REG_SCLR2_CTL, 0x30, 4 /* 0-YUV,1-RGB565,3-RGB32 */
+#define SCL_R2_420C_FMT REG_SCLR2_CTL, BIT8, 8 /* 0-frame,1-field */
+#define SCL_R2_VFMT REG_SCLR2_CTL, 0xE00, 9 /* 0-YUV422,1-YUV420,2-YUV444,4-RGB32 */
+#define SCL_R2_H264_FMT REG_SCLR2_CTL, BIT12, 12 /* 0-MPEG,1-H264 */
+#define SCL_R2_IOFMT REG_SCLR2_CTL, BIT16, 16 /* 0-frame,1-field */
+#define SCL_R2_COLOR_EN REG_SCLR2_CTL, BIT24, 24 /* 0-disable,1-enable */
+#define SCL_R2_COLOR_WIDE REG_SCLR2_CTL, BIT25, 25 /* 0-Normal,1-Wider */
+#define SCL_R2_COLOR_INV REG_SCLR2_CTL, BIT26, 26 /* 0-Normal,1-Opposite color */
+
+/* REG_SCLR2_YSA,0x44 */
+/* REG_SCLR2_CSA,0x48 */
+
+/* REG_SCLR2_H_SIZE,0x4C */
+#define SCL_R2_FBW REG_SCLR2_H_SIZE, 0x1FFF, 0 /* frame buffer width pixel */
+#define SCL_R2_LNSIZE REG_SCLR2_H_SIZE, 0x1FFF0000, 16 /* line width pixel */
+
+/* REG_SCLR2_CROP,0x50 */
+#define SCL_R2_HCROP REG_SCLR2_CROP, 0x1FFF, 0
+#define SCL_R2_VCROP REG_SCLR2_CROP, 0x1FFF0000, 16
+
+/* REG_ALFA_MD,0x54 */
+#define SCL_ALPHA_SRC REG_ALFA_MD, 0x3, 0 /* 0-RMIF1,1-RMIF2,2-Fixed ALPHA */
+#define SCL_ALPHA_DST REG_ALFA_MD, 0x300, 8 /* 0-RMIF1,1-RMIF2,2-Fixed ALPHA */
+#define SCL_ALPHA_SWAP REG_ALFA_MD, 0x10000, 16 /* 0-(alpha,1-alpha),1:(1-alpha,alpha) */
+
+/* REG_ALFA_FXD,0x58 */
+#define SCL_ALPHA_SRC_FIXED REG_ALFA_FXD, 0xFF, 0
+#define SCL_ALPHA_DST_FIXED REG_ALFA_FXD, 0xFF00, 8
+
+/* REG_ALFA_COLORKEY,0x5C */
+#define SCL_ALPHA_COLORKEY_ENABLE REG_ALFA_COLORKEY, BIT0, 0
+#define SCL_ALPHA_COLORKEY_FROM REG_ALFA_COLORKEY, BIT8, 8 /* 0-RMIF1,1-RMIF2 */
+#define SCL_ALPHA_COLORKEY_COMP REG_ALFA_COLORKEY, 0x30000, 16 /* 0-888,1-777,2-666,3-555 */
+#define SCL_ALPHA_COLORKEY_MODE REG_ALFA_COLORKEY, 0x7000000, 24 /* (Non-Hit,Hit):0/1-(alpha,alpha),
+ 2-(alpha,pix1),3-(pix1,alpha),4-(alpha,pix2),
+ 5-(pix2,alpha),6-(pix1,pix2),7-(pix2,pix1) */
+
+/* REG_ALFA_COLORKEY_RGB,0x60 */
+#define SCL_ALPHA_COLORKEY_R REG_ALFA_COLORKEY_RGB, 0xFF, 0
+#define SCL_ALPHA_COLORKEY_G REG_ALFA_COLORKEY_RGB, 0xFF00, 8
+#define SCL_ALPHA_COLORKEY_B REG_ALFA_COLORKEY_RGB, 0xFF0000, 16
+
+/* REG_SCL_VXWIDTH,0x70 */
+#define SCL_VXWIDTH REG_SCL_VXWIDTH, 0x1FFF, 0
+#define SCL_DST_VXWIDTH REG_SCL_VXWIDTH, 0x1FFF0000, 16
+
+/* REG_SCL_SCLUP_EN,0x74 */
+#define SCL_VSCLUP_ENABLE REG_SCL_SCLUP_EN, BIT16, 16
+#define SCL_HSCLUP_ENABLE REG_SCL_SCLUP_EN, BIT0, 0
+
+/* REG_SCL_VSCALE1,0x78 */
+#define SCL_V_SUBSTEP REG_SCL_VSCALE1, 0x1FFF0000, 16
+#define SCL_V_THR REG_SCL_VSCALE1, 0x1FFF, 0
+
+/* REG_SCL_VSCALE2,0x7c */
+#define SCL_V_STEP REG_SCL_VSCALE2, 0x1FFF0000, 16
+#define SCL_V_I_SUBSTEPCNT REG_SCL_VSCALE2, 0x1FFF, 0
+
+/* REG_SCL_VSCALE3,0x80 */
+#define SCL_V_I_STEPCNT REG_SCL_VSCALE3, 0x1FFFF, 0
+
+/* REG_SCL_HSCALE1,0x84 */
+#define SCL_H_SUBSTEP REG_SCL_HSCALE1, 0x1FFF0000, 16
+#define SCL_H_THR REG_SCL_HSCALE1, 0x1FFF, 0
+
+/* REG_SCL_HSCALE2,0x88 */
+#define SCL_H_STEP REG_SCL_HSCALE2, 0x1FFF0000, 16
+#define SCL_H_I_SUBSTEPCNT REG_SCL_HSCALE2, 0x1FFF, 0
+
+/* REG_SCL_HSCALE3,0x8c */
+#define SCL_H_I_STEPCNT REG_SCL_HSCALE3, 0x1FFFF, 0
+
+/* REG_SCLR_REQ_NUM,0x90 */
+#define SCL_R_C_REQ_NUM REG_SCLR_REQ_NUM, 0xFF, 0
+#define SCL_R_Y_REQ_NUM REG_SCLR_REQ_NUM, 0xFF00, 8
+
+/* REG_SCL_SCLDW,0x94 */
+#define SCL_SCLDW_METHOD REG_SCL_SCLDW, BIT0, 0 /* (VPU path, scale dn) 0 - bilinear mode, quality better */
+
+/* REG_SCL_426_SW,0x98 */
+#define SCL_426_SW REG_SCL_426_SW, BIT0, 0 /* 1-follow 426, 0-437 */
+
+/* REG_SCL_VBYPASS,0x9C */
+#define SCL_VBYPASS REG_SCL_VBYPASS, BIT0, 0
+
+/* SCL_TG */
+/* REG_SCL_TG_CTL,0xa0 */
+#define SCL_ONESHOT_ENABLE REG_SCL_TG_CTL, BIT24, 24 /* sacling complete will set SCL tg enable to 0 */
+#define SCL_TG_RDCYC REG_SCL_TG_CTL, 0xFF0000, 16
+#define SCL_TG_WATCHDOG_ENABLE REG_SCL_TG_CTL, BIT8, 8
+#define SCL_TG_ERR_OFF REG_SCL_TG_CTL, BIT4, 4 /* disable TG_EN whtn tg timeout */
+#define SCL_TG_ENABLE REG_SCL_TG_CTL, BIT0, 0
+
+/* REG_SCL_TG_TOTAL,0xa4 */
+#define SCL_TG_V_ALLLINE REG_SCL_TG_TOTAL, 0x1FFF0000, 16
+#define SCL_TG_H_ALLPIXEL REG_SCL_TG_TOTAL, 0x1FFF, 0
+
+/* REG_SCL_TG_V_ACTIVE,0xa8 */
+#define SCL_TG_V_ACTEND REG_SCL_TG_V_ACTIVE, 0x1FFF0000, 16
+#define SCL_TG_V_ACTBG REG_SCL_TG_V_ACTIVE, 0xFF, 0
+
+/* REG_SCL_TG_H_ACTIVE,0xac */
+#define SCL_TG_H_ACTEND REG_SCL_TG_H_ACTIVE, 0x1FFF0000, 16
+#define SCL_TG_H_ACTBG REG_SCL_TG_H_ACTIVE, 0x3FF, 0
+
+/* REG_SCL_TG_VBI,0xb0 */
+#define SCL_TG_PVBI REG_SCL_TG_VBI, 0x1F00, 8
+#define SCL_TG_VBIE REG_SCL_TG_VBI, 0x7F, 0
+
+/* REG_SCL_TG_WATCHDOG,0xb4 */
+#define SCL_TG_WATCHDOG_VALUE REG_SCL_TG_WATCHDOG, 0xFFFFFFFF, 0
+
+/* REG_SCL_TG_STS,0xb8 */
+#define SCL_INTSTS_TGERR REG_SCL_TG_STS, BIT0, 0
+
+/* REG_SCL_TG_GOVW,0xbc */
+#define SCL_TG_GOVWTG_ENABLE REG_SCL_TG_GOVW, BIT0, 0
+
+/* SCLR */
+/* REG_SCL_MIF_CTL,0xc0 */
+#define SCLR_COLBAR_INVERSION REG_SCLR_CTL, BIT26, 26
+#define SCLR_COLBAR_MODE REG_SCLR_CTL, BIT25, 25
+#define SCLR_COLBAR_ENABLE REG_SCLR_CTL, BIT24, 24
+#define SCLR_TAR_DISP_FMT REG_SCLR_CTL, BIT16, 16 /*0:Frame, 1:Field */
+#define SCLR_MEDIAFMT_H264 REG_SCLR_CTL, BIT12, 12 /*0:MPEG, 1:H264 */
+#define SCLR_COLFMT_RGB REG_SCLR_CTL, BIT11, 11 /*0:YCbCr, 1:RGB32 */
+#define SCLR_COLFMT_YUV REG_SCLR_CTL, 0x600, 9 /*0:422,1:420,2:444*/
+#define SCLR_SRC_DISP_FMT REG_SCLR_CTL, BIT8, 8 /*420C 0:Frame, 1:Field */
+#define SCLR_RGB_MODE REG_SCLR_CTL, 0x30, 4 /*0:YC,1:RGB565,3:RGB32 */
+#define SCLR_MIF_ENABLE REG_SCLR_CTL, BIT0, 0 /*0:Disable, 1:Enable */
+
+/* REG_SCLR_YSA,0xc4 */
+
+/* REG_SCLR_CSA,0xc8 */
+
+/* REG_SCLR_H_SIZE,0xcc */
+#define SCLR_YPXLWID REG_SCLR_H_SIZE, 0x1FFF0000, 16
+#define SCLR_YBUFWID REG_SCLR_H_SIZE, 0x1FFF, 0
+
+/* REG_SCLR_CROP,0xd0 */
+#define SCLR_VCROP REG_SCLR_CROP, 0x1FFF0000, 16
+#define SCLR_HCROP REG_SCLR_CROP, 0x1FFF, 0
+
+/* REG_SCLR_FIFO_CTL,0xd4 (W:0xf4) */
+#define SCLR_INTSTS_R2MIFERR REG_SCLR_FIFO_CTL, BIT9, 9
+#define SCLR_INTSTS_R1MIFERR REG_SCLR_FIFO_CTL, BIT8, 8
+#define SCLR_FIFO_THR REG_SCLR_FIFO_CTL, 0xF, 0
+
+/* SCL_W */
+/* REG_SCLW_CTL,0xe0 */
+#define SCLW_COLFMT_RGB REG_SCLW_CTL, BIT9, 9 /* 0-YC,1-RGB32 */
+#define SCLW_COLFMT_YUV REG_SCLW_CTL, BIT8, 8 /* 0-444,1-422 */
+#define SCLW_MIF_ENABLE REG_SCLW_CTL, BIT0, 0
+
+/* REG_SCLW_YSA,0xe4 */
+
+/* REG_SCLW_CSA,0xe8 */
+
+/* REG_SCLW_Y_TIME,0xec */
+#define SCLW_YPXLWID REG_SCLW_Y_TIME, 0x1FFF0000, 16
+#define SCLW_YBUFWID REG_SCLW_Y_TIME, 0x1FFF, 0
+
+/* REG_SCLW_C_TIME,0xf0 */
+#define SCLW_CPXLWID REG_SCLW_C_TIME, 0xFFF0000, 16
+#define SCLW_CBUFWID REG_SCLW_C_TIME, 0x1FFF, 0
+
+/* REG_SCLW_FF_CTL,0xf4 (R:0xd4) */
+#define SCLW_INTSTS_MIFRGBERR REG_SCLW_FF_CTL, BIT16, 16
+#define SCLW_INTSTS_MIFYERR REG_SCLW_FF_CTL, BIT8, 8
+#define SCLW_INTSTS_MIFCERR REG_SCLW_FF_CTL, BIT0, 0
+
+/* REG_SCLW_INT,0xf8 */
+#define SCLW_INT_TGERR_ENABLE REG_SCLW_INT, BIT16, 16
+#define SCLW_INT_R1MIF_ENABLE REG_SCLW_INT, BIT9, 9
+#define SCLW_INT_R2MIF_ENABLE REG_SCLW_INT, BIT8, 8
+#define SCLW_INT_WMIFRGB_ENABLE REG_SCLW_INT, BIT2, 2
+#define SCLW_INT_WMIFYERR_ENABLE REG_SCLW_INT, BIT1, 1
+#define SCLW_INT_WMIFCERR_ENABLE REG_SCLW_INT, BIT0, 0
+
+/* REG_SCL_TRUE_BILINEAR,0xfc */
+#define SCL_BILINEAR_H REG_SCL_TRUE_BILINEAR, BIT0, 0
+#define SCL_BILINEAR_V REG_SCL_TRUE_BILINEAR, BIT8, 8
+
+/* SCL Base2 */
+/* REG_SCL_CSC_CTL,0x00 */
+#define SCL_CSC_ENABLE REG_SCL_CSC_CTL, BIT16, 16
+#define SCL_CSC_CLAMP_ENABLE REG_SCL_CSC_CTL, BIT8, 8 /* clamp to 16-235 */
+#define SCL_CSC_MODE REG_SCL_CSC_CTL, BIT0, 0 /* 0-RGB2YC,1-YC2RGB */
+
+/* REG_SCL_CSC1,0x04 */
+/* REG_SCL_CSC2,0x08 */
+/* REG_SCL_CSC3,0x0c */
+/* REG_SCL_CSC4,0x10 */
+/* REG_SCL_CSC5,0x14 */
+/* REG_SCL_CSC6,0x18 */
+
+/* REG_SCL_ARGB_ALPHA,0x1C */
+#define SCL_FIXED_ALPHA_ENABLE REG_SCL_ARGB_ALPHA, BIT0, 0
+#define SCL_FIXED_ALPHA_DATA REG_SCL_ARGB_ALPHA, 0xFF00, 8
+
+/* REG_SCL_IGS,0x20 */
+#define SCL_IGS_MODE REG_SCL_IGS, 0x3, 0 /* 0-888,1-555,2-666,3-565 */
+
+/* REG_SCL_R2_CSC,0x24 */
+#define SCL_R2_CSC_MODE REG_SCL_R2_CSC, BIT0, 0 /* 0-CCIR/ITU-601 */
+#define SCL_R2_CSC_CLAMP_EN REG_SCL_R2_CSC, BIT8, 8 /* 0-direct,1-16-235 */
+#define SCL_R2_CSC_EN REG_SCL_R2_CSC, BIT16, 16
+
+/* REG_SCL_R2_CSC1,0x28 */
+/* REG_SCL_R2_CSC2,0x2C */
+/* REG_SCL_R2_CSC3,0x30 */
+/* REG_SCL_R2_CSC4,0x34 */
+/* REG_SCL_R2_CSC5,0x38 */
+/* REG_SCL_R2_CSC6,0x3C */
+
+/* REG_SCL_RECURSIVE_MODE,0xA0 */
+#define SCL_RECURSIVE_H REG_SCL_RECURSIVE_MODE, BIT0, 0
+#define SCL_RECURSIVE_V REG_SCL_RECURSIVE_MODE, BIT8, 8
+
+/* REG_SCL_FIELD_MODE,0xC0 */
+#define SCL_DEBLOCK_ENABLE REG_SCL_FIELD_MODE, BIT0, 0
+#define SCL_FIELD_DEFLICKER REG_SCL_FIELD_MODE, BIT1, 1
+#define SCL_FRAME_DEFLICKER REG_SCL_FIELD_MODE, BIT2, 2
+
+/* REG_SCL_DBLK_THRESHOLD,0xC4 */
+#define SCL_1ST_LAYER_BOUNDARY REG_SCL_DBLK_THRESHOLD, 0xFF, 0
+#define SCL_2ND_LAYER_BOUNDARY REG_SCL_DBLK_THRESHOLD, 0xFF00, 8
+
+/* REG_SCL_FIELD_FLICKER,0xC8 */
+#define SCL_FIELD_FILTER_CONDITION REG_SCL_FIELD_FLICKER, BIT0, 0 /* 0-up or down,1-up and down */
+#define SCL_FIELD_FILTER_Y_THD REG_SCL_FIELD_FLICKER, 0xFF00, 8
+#define SCL_FIELD_FILTER_C_THD REG_SCL_FIELD_FLICKER, 0xFF0000, 16
+
+/* REG_SCL_FRAME_FLICKER,0xCC */
+#define SCL_FRAME_FILTER_RGB REG_SCL_FRAME_FLICKER, BIT0, 0 /* 0-Y,1-RGB */
+#define SCL_FRAME_FILTER_SAMPLER REG_SCL_FRAME_FLICKER, 0x1F00, 8 /* 2^x */
+#define SCL_FR_FILTER_SCENE_CHG_THD REG_SCL_FRAME_FLICKER, 0xFF0000, 16
+
+/* REG_SCL_READCYC_1T,0xD0 */
+#define SCL_READCYC_1T REG_SCL_READCYC_1T,BIT0,0
+#endif /* WMT_SCL_REG_H */
diff --git a/drivers/video/wmt/hw/wmt-vpp-hw.h b/drivers/video/wmt/hw/wmt-vpp-hw.h
new file mode 100644
index 00000000..b3d3a7ad
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-vpp-hw.h
@@ -0,0 +1,135 @@
+/*++
+ * linux/drivers/video/wmt/hw/wmt-vpp-hw.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_VPP_HW_H
+#define WMT_VPP_HW_H
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+/*
+* Product ID / Project ID
+* 84xx series: 8420/3300, 8430/3357, 8435/3437
+* 85xx series: 8500/3400, 8510/3426, 8520/3429
+*/
+/* 84xx series, (1-100) with VDU & DSP */
+#define VIA_PID_8420 10 /* 3300 */
+#define VIA_PID_8430 12 /* 3357 */
+#define WMT_PID_8435 14 /* 3437 */
+#define WMT_PID_8440 16 /* 3451 */
+#define WMT_PID_8425 18 /* 3429 */
+#define WMT_PID_8710 20 /* 3445 */
+#define WMT_PID_8950 22 /* 3481 */
+#define WMT_PID_8980 24 /* 3498 */
+
+/* 85xx series, (101-200) */
+#define VIA_PID_8500 110 /* 3400 */
+#define WMT_PID_8505 111
+#define WMT_PID_8510 112 /* 3426* */
+
+#define WMT_PID_8950_A 1
+
+/* current pid */
+#define WMT_CUR_PID WMT_PID_8980
+#define WMT_SUB_PID 0
+
+/* #define WMT_SUB_PID WMT_PID_8505 */
+#ifndef WMT_SUB_PID
+ #define WMT_SUB_PID 0
+#endif
+
+/* VPP interrupt map to irq */
+#define VPP_IRQ_SCL_FINISH IRQ_VPP_IRQ0
+#define VPP_IRQ_SCL IRQ_VPP_IRQ1
+#define VPP_IRQ_SCL444_TG IRQ_VPP_IRQ2
+#define VPP_IRQ_VPPM IRQ_VPP_IRQ3
+#define VPP_IRQ_GOVW_TG IRQ_VPP_IRQ4
+#define VPP_IRQ_GOVW IRQ_VPP_IRQ5
+#define VPP_IRQ_GOVM IRQ_VPP_IRQ6
+#define VPP_IRQ_GE IRQ_VPP_IRQ7
+#define VPP_IRQ_GOVRH_TG IRQ_VPP_IRQ8 /* PVBI or VBIS or VBIE */
+#define VPP_IRQ_DVO IRQ_VPP_IRQ9
+#define VPP_IRQ_VID IRQ_VPP_IRQ10
+#define VPP_IRQ_GOVR IRQ_VPP_IRQ11 /* underrun & mif */
+#define VPP_IRQ_GOVRSD_TG IRQ_VPP_IRQ12
+#define VPP_IRQ_VPU IRQ_VPP_IRQ13
+#define VPP_IRQ_VPU_TG IRQ_VPP_IRQ14
+#define VPP_IRQ_HDMI_CP IRQ_VPP_IRQ15
+#define VPP_IRQ_HDMI_HPDH IRQ_VPP_IRQ16
+#define VPP_IRQ_HDMI_HPDL IRQ_VPP_IRQ17
+#define VPP_IRQ_GOVR_0 IRQ_VPP_IRQ18
+#define VPP_IRQ_GOVR_2 IRQ_VPP_IRQ19
+#define VPP_IRQ_CEC IRQ_VPP_IRQ20
+#define VPP_IRQ_GOVR2_0 IRQ_VPP_IRQ21
+#define VPP_IRQ_GOVR2 IRQ_VPP_IRQ22
+#define VPP_IRQ_GOVR2_2 IRQ_VPP_IRQ23
+#define VPP_IRQ_DVO2 IRQ_VPP_IRQ24
+#define VPP_IRQ_GOVR2_TG IRQ_VPP_IRQ25
+
+/* DVI I2C */
+#define VPP_DVI_I2C_BIT 0x80 /* use sw id that can vary */
+#define VPP_DVI_I2C_SW_BIT 0x10 /* hw or sw i2c */
+#define VPP_DVI_I2C_ID_MASK 0x1F
+#define VPP_DVI_I2C_ID (VPP_DVI_I2C_BIT + 0x1)
+#define VPP_DVI_EDID_ID (VPP_DVI_I2C_SW_BIT + 0x1) /* DVO EDID use sw i2c bus 1 */
+
+/* vout */
+#define VPP_VOUT_INFO_NUM 2 /* linux fb or govr number */
+
+#define VPP_VOUT_NUM 2
+#define VPP_VOUT_ALL 0xFFFFFFFF
+#define VPP_VOUT_NUM_HDMI 0
+#define VPP_VOUT_NUM_LVDS 1
+#define VPP_VOUT_NUM_DVI 1
+
+#define WMT_FTBLK_VOUT_DVI
+#define WMT_FTBLK_VOUT_HDMI
+#define WMT_FTBLK_VOUT_LVDS
+
+/* hw parameter */
+#define VPP_VOINT_NO 0 /* DVO external board interrupt use GPIOxx */
+#define VPP_UBOOT_COLFMT VDO_COL_FMT_RGB_565
+#define VPP_FB_ADDR_ALIGN 64
+#define VPP_FB_WIDTH_ALIGN 64 /* hw should 4 byte align,android
+ framework 8 byte align modify by aksenxu VPU need 64bytes alignment
+ you need modify FramebufferNativeWindow::FramebufferNativeWindow
+ in android framework together */
+#define VPP_GOVR_DVO_DELAY_24 0x4036
+#define VPP_GOVR_DVO_DELAY_12 0x120
+
+/*-------------------- DEPENDENCY -------------------------------------*/
+#ifdef __KERNEL__
+#ifndef CONFIG_WMT_HDMI
+#undef WMT_FTBLK_VOUT_HDMI
+#endif
+#endif
+
+#include "wmt-vpp-reg.h"
+#include "wmt-govrh-reg.h"
+#include "wmt-lvds-reg.h"
+#ifdef WMT_FTBLK_VOUT_HDMI
+#include "wmt-hdmi-reg.h"
+#endif
+#ifndef CFG_LOADER
+#include "wmt-scl-reg.h"
+#include "wmt-cec-reg.h"
+#endif
+#endif /* WMT_VPP_HW_H */
diff --git a/drivers/video/wmt/hw/wmt-vpp-reg.h b/drivers/video/wmt/hw/wmt-vpp-reg.h
new file mode 100644
index 00000000..ffd02b87
--- /dev/null
+++ b/drivers/video/wmt/hw/wmt-vpp-reg.h
@@ -0,0 +1,98 @@
+/*++
+ * linux/drivers/video/wmt/register/wm8710/wmt-vpp-reg.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef WMT_VPP_REG_H
+#define WMT_VPP_REG_H
+
+#define REG_VPP_BEGIN (VPP_BASE_ADDR + 0x00)
+#define REG_VPP_INTSTS (VPP_BASE_ADDR + 0x04)
+#define REG_VPP_INTEN (VPP_BASE_ADDR + 0x08)
+#define REG_VPP_WATCH_SEL (VPP_BASE_ADDR + 0x0c)
+#define REG_VPP_SWRST1_SEL (VPP_BASE_ADDR + 0x10)
+#define REG_VPP_SWRST2_SEL (VPP_BASE_ADDR + 0x14)
+#define REG_VPP_DAC_SEL (VPP_BASE_ADDR + 0x18)
+#define REG_VPP_SWRST3_SEL (VPP_BASE_ADDR + 0x1C)
+#define REG_VPP_SSCG (VPP_BASE_ADDR + 0x20)
+#define REG_VPP_END (VPP_BASE_ADDR + 0x28)
+
+/* REG_VPP_INTSTS,0x04 */
+#define VPP_GE_INTSTS_TG REG_VPP_INTSTS, BIT20, 20
+#define VPP_SCL_INTSTS_VBIE REG_VPP_INTSTS, BIT18, 18
+#define VPP_SCL_INTSTS_VBIS REG_VPP_INTSTS, BIT17, 17
+#define VPP_SCL_INTSTS_PVBI REG_VPP_INTSTS, BIT16, 16
+#define VPP_SCL_INTSTS REG_VPP_INTSTS, 0x70000, 16
+#define VPP_GOVRH2_INTSTS_VBIE REG_VPP_INTSTS, BIT14, 14
+#define VPP_GOVRH2_INTSTS_VBIS REG_VPP_INTSTS, BIT13, 13
+#define VPP_GOVRH2_INTSTS_PVBI REG_VPP_INTSTS, BIT12, 12
+#define VPP_GOVRH2_INTSTS REG_VPP_INTSTS, 0x7000, 12
+#define VPP_GOVRH_INTSTS_VBIE REG_VPP_INTSTS, BIT10, 10
+#define VPP_GOVRH_INTSTS_VBIS REG_VPP_INTSTS, BIT9, 9
+#define VPP_GOVRH_INTSTS_PVBI REG_VPP_INTSTS, BIT8, 8
+#define VPP_GOVRH_INTSTS REG_VPP_INTSTS, 0x700, 8
+
+/* REG_VPP_INTEN,0x08 */
+#define VPP_GE_INTEN_TG REG_VPP_INTEN, BIT20, 20
+#define VPP_SCL_INTEN_VBIE REG_VPP_INTEN, BIT18, 18
+#define VPP_SCL_INTEN_VBIS REG_VPP_INTEN, BIT17, 17
+#define VPP_SCL_INTEN_PVBI REG_VPP_INTEN, BIT16, 16
+#define VPP_SCL_INTEN REG_VPP_INTEN, 0x70000, 16
+#define VPP_GOVRH2_INTEN_VBIE REG_VPP_INTEN, BIT14, 14
+#define VPP_GOVRH2_INTEN_VBIS REG_VPP_INTEN, BIT13, 13
+#define VPP_GOVRH2_INTEN_PVBI REG_VPP_INTEN, BIT12, 12
+#define VPP_GOVRH2_INTEN REG_VPP_INTEN, 0x7000, 12
+#define VPP_GOVRH_INTEN_VBIE REG_VPP_INTEN, BIT10, 10
+#define VPP_GOVRH_INTEN_VBIS REG_VPP_INTEN, BIT9, 9
+#define VPP_GOVRH_INTEN_PVBI REG_VPP_INTEN, BIT8, 8
+#define VPP_GOVRH_INTEN REG_VPP_INTEN, 0x700, 8
+
+/* REG_VPP_WATCH_SEL,0x0c */
+#define VPP_WATCH_SEL REG_VPP_WATCH_SEL, 0x1F, 0
+
+/* REG_VPP_SWRST1_SEL,0x10 */
+#define VPP_GE_RST REG_VPP_SWRST1_SEL, BIT16, 16
+#define VPP_VID_RST REG_VPP_SWRST1_SEL, BIT8, 8
+#define VPP_SCL_RST REG_VPP_SWRST1_SEL, BIT0, 0
+
+/* REG_VPP_SWRST2_SEL,0x14 */
+#define VPP_CEC_RST REG_VPP_SWRST2_SEL, BIT12,12
+#define VPP_DVO2_RST REG_VPP_SWRST2_SEL, BIT9, 9
+#define VPP_DVO_RST REG_VPP_SWRST2_SEL, BIT8, 8
+#define VPP_LVDS_RST REG_VPP_SWRST2_SEL, BIT4, 4
+#define VPP_GOVRH_RST REG_VPP_SWRST2_SEL, BIT0, 0
+
+/* REG_VPP_DAC_SEL,0x18 */
+#define VPP_DAC_SEL REG_VPP_DAC_SEL, BIT0, 0
+#define VPP_DAC_SEL_TV 1
+#define VPP_DAC_SEL_VGA 0
+
+/* REG_VPP_SWRST3_SEL,0x1C */
+#define VPP_HDMI_RST REG_VPP_SWRST3_SEL, BIT0, 0
+#define VPP_DDC_RST REG_VPP_SWRST3_SEL, BIT8, 8
+#define VPP_HDMI2_RST REG_VPP_SWRST3_SEL, BIT16, 16
+
+/* REG_VPP_SSCG,0x20 */
+#define VPP_SSCG_DISABLE REG_VPP_SSCG, BIT0, 0
+#define VPP_CSI_ACT_LANE_SELECT REG_VPP_SSCG, BIT8, 8 /* 0-Active lane 0/1, 1-Active lane 2/3 */
+
+#endif /* WMT_VPP_REG_H */
+
diff --git a/drivers/video/wmt/lcd.h b/drivers/video/wmt/lcd.h
new file mode 100644
index 00000000..a2a5ecce
--- /dev/null
+++ b/drivers/video/wmt/lcd.h
@@ -0,0 +1,98 @@
+/*++
+ * linux/drivers/video/wmt/lcd.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef LCD_H
+/* To assert that only one occurrence is included */
+#define LCD_H
+/*-------------------- MODULE DEPENDENCY -------------------------------------*/
+#include "vpp.h"
+
+/* following is the C++ header */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+/* #define LCD_XXXX 1 *//*Example*/
+
+/*-------------------- EXPORTED PRIVATE TYPES---------------------------------*/
+/* typedef void lcd_xxx_t; *//*Example*/
+typedef enum {
+ LCD_WMT_OEM,
+ LCD_CHILIN_LW0700AT9003,
+ LCD_INNOLUX_AT070TN83,
+ LCD_AUO_A080SN01,
+ LCD_EKING_EK08009,
+ LCD_HANNSTAR_HSD101PFW2,
+ LCD_LVDS_1024x600,
+ LCD_B079XAN01,
+ LCD_BP080WX7,
+ LCD_PANEL_MAX
+} lcd_panel_t;
+
+#define LCD_CAP_CLK_HI BIT(0)
+#define LCD_CAP_HSYNC_HI BIT(1)
+#define LCD_CAP_VSYNC_HI BIT(2)
+#define LCD_CAP_DE_LO BIT(3)
+typedef struct {
+ int bits_per_pixel;
+ unsigned int capability;
+ struct fb_videomode vmode;
+
+ void (*initial)(void);
+ void (*uninitial)(void);
+} lcd_parm_t;
+
+/*-------------------- EXPORTED PRIVATE VARIABLES ---------------------------*/
+#ifdef LCD_C /* allocate memory for variables only in vout.c */
+#define EXTERN
+#else
+#define EXTERN extern
+#endif /* ifdef LCD_C */
+
+/* EXTERN int lcd_xxx; *//*Example*/
+
+EXTERN lcd_parm_t *p_lcd;
+#undef EXTERN
+
+/*--------------------- EXPORTED PRIVATE MACROS ------------------------------*/
+/* #define LCD_XXX_YYY xxxx *//*Example*/
+/*--------------------- EXPORTED PRIVATE FUNCTIONS --------------------------*/
+/* extern void lcd_xxx(void); *//*Example*/
+
+int lcd_panel_register(int no, void (*get_parm)(int mode));
+lcd_parm_t *lcd_get_parm(lcd_panel_t id, unsigned int arg);
+void lcd_set_parm(int id, int bpp);
+lcd_parm_t *lcd_get_oem_parm(int resx, int resy);
+void lcd_set_lvds_id(int id);
+int lcd_get_lvds_id(void);
+void lcd_set_type(int type);
+int lcd_get_type(void);
+void lcd_set_enable(int enable);
+void lcd_enable_signal(int enable);
+
+/* LCD back light */
+#ifdef __cplusplus
+}
+#endif
+#endif /* ifndef LCD_H */
diff --git a/drivers/video/wmt/lvds.c b/drivers/video/wmt/lvds.c
new file mode 100644
index 00000000..3f8f8c83
--- /dev/null
+++ b/drivers/video/wmt/lvds.c
@@ -0,0 +1,204 @@
+/*++
+ * linux/drivers/video/wmt/lvds.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define LVDS_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "lvds.h"
+
+#ifdef WMT_FTBLK_LVDS
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LVDS_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lvds_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lvds.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lvds_xxx; *//*Example*/
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lvds_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+void lvds_set_power_down(int pwrdn)
+{
+ DBG_DETAIL("(%d)\n", pwrdn);
+
+ vppif_reg32_write(LVDS_PD, pwrdn);
+ mdelay(1);
+ vppif_reg32_write(LVDS_PD_L2HA, pwrdn);
+}
+
+void lvds_set_enable(vpp_flag_t enable)
+{
+ DBG_DETAIL("(%d)\n", enable);
+ vppif_reg32_write(LVDS_TRE_EN, (enable) ? 0 : 1);
+ vppif_reg32_write(LVDS_MODE, (enable) ? 1 : 0);
+ vppif_reg32_write(LVDS_RESA_EN, (enable) ? 0 : 1);
+#ifdef CONFIG_UBOOT
+ if ((enable) && (lcd_get_lvds_id() == LCD_LVDS_1024x600)) {
+ /* GPIO10 VDD_EN->CLK delay 16->38ms */
+ REG32_VAL(GPIO_BASE_ADDR + 0x80) |= 0x400;
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) |= 0x400;
+ mdelay(16);
+ }
+#endif
+}
+
+int lvds_get_enable(void)
+{
+ return vppif_reg32_read(LVDS_MODE);
+}
+
+void lvds_set_rgb_type(int bpp)
+{
+ int mode;
+ int mode_change = 0x2;
+
+ DBG_DETAIL("(%d)\n", bpp);
+
+ /* 0:888, 1-555, 2-666, 3-565 */
+ switch (bpp) {
+ case 15:
+ mode = 1;
+ break;
+ case 16:
+ mode = 3;
+ break;
+ case 18:
+ mode = 2;
+ break;
+ case 24:
+ default:
+ mode = 0;
+ mode_change = 0x0;
+ break;
+ }
+#if 1 /* IGS default */
+ mode = 4;
+#endif
+ vppif_reg32_write(LVDS_TEST, mode_change);
+ vppif_reg32_write(LVDS_IGS_BPP_TYPE, mode);
+}
+
+vdo_color_fmt lvds_get_colfmt(void)
+{
+ return VDO_COL_FMT_ARGB;
+}
+
+void lvds_set_sync_polar(int h_lo, int v_lo)
+{
+ DBG_DETAIL("(%d,%d)\n", h_lo, v_lo);
+
+ vppif_reg32_write(LVDS_HSYNC_POLAR_LO, h_lo);
+ vppif_reg32_write(LVDS_VSYNC_POLAR_LO, v_lo);
+}
+
+void lvds_get_sync_polar(int *hsync_hi, int *vsync_hi)
+{
+ *hsync_hi = (vppif_reg32_read(LVDS_HSYNC_POLAR_LO)) ? 0 : 1;
+ *vsync_hi = (vppif_reg32_read(LVDS_VSYNC_POLAR_LO)) ? 0 : 1;
+}
+
+/*----------------------- Module API --------------------------------------*/
+void lvds_reg_dump(void)
+{
+ DPRINT("========== LVDS register dump ==========\n");
+ vpp_reg_dump(REG_LVDS_BEGIN, REG_LVDS_END-REG_LVDS_BEGIN);
+
+ DPRINT("---------- LVDS common ----------\n");
+ DPRINT("test %d,dual chan %d,inv clk %d\n", vppif_reg32_read(LVDS_TEST),
+ vppif_reg32_read(LVDS_DUAL_CHANNEL),
+ vppif_reg32_read(LVDS_INV_CLK));
+ DPRINT("ldi shift left %d,IGS bpp type %d\n",
+ vppif_reg32_read(LVDS_LDI_SHIFT_LEFT),
+ vppif_reg32_read(LVDS_IGS_BPP_TYPE));
+ DPRINT("rsen %d,pll ready %d\n", vppif_reg32_read(LVDS_RSEN),
+ vppif_reg32_read(LVDS_PLL_READY));
+ DPRINT("pwr dn %d\n", vppif_reg32_read(LVDS_PD));
+}
+
+#ifdef CONFIG_PM
+static unsigned int *lvds_pm_bk;
+static unsigned int lvds_pd_bk;
+void lvds_suspend(int sts)
+{
+ switch (sts) {
+ case 0: /* disable module */
+ break;
+ case 1: /* disable tg */
+ break;
+ case 2: /* backup register */
+ lvds_pd_bk = vppif_reg32_read(LVDS_PD);
+ lvds_set_power_down(1);
+ lvds_pm_bk = vpp_backup_reg(REG_LVDS_BEGIN,
+ (REG_LVDS_END-REG_LVDS_BEGIN));
+ break;
+ default:
+ break;
+ }
+}
+
+void lvds_resume(int sts)
+{
+ switch (sts) {
+ case 0: /* restore register */
+ vpp_restore_reg(REG_LVDS_BEGIN,
+ (REG_LVDS_END-REG_LVDS_BEGIN), lvds_pm_bk);
+ lvds_pm_bk = 0;
+ if (lcd_get_lvds_id() != LCD_LVDS_1024x600)
+ lvds_set_power_down(lvds_pd_bk);
+ break;
+ case 1: /* enable module */
+ break;
+ case 2: /* enable tg */
+ break;
+ default:
+ break;
+ }
+}
+#else
+#define lvds_suspend NULL
+#define lvds_resume NULL
+#endif
+
+void lvds_init(void)
+{
+ vppif_reg32_write(LVDS_REG_LEVEL, 1);
+ vppif_reg32_write(LVDS_REG_UPDATE, 1);
+ vppif_reg32_write(LVDS_PLL_R_F, 0x1);
+ vppif_reg32_write(LVDS_PLL_CPSET, 0x1);
+ vppif_reg32_write(LVDS_TRE_EN, 0x0);
+ vppif_reg32_write(LVDS_VBG_SEL, 2);
+ vppif_reg32_write(LVDS_DRV_PDMODE, 0);
+ vppif_reg32_write(LVDS_LDI_SHIFT_LEFT, 1);
+ vppif_reg32_out(REG_LVDS_TEST2, 0x31432);
+}
+
+#endif /* WMT_FTBLK_LVDS */
+
diff --git a/drivers/video/wmt/lvds.h b/drivers/video/wmt/lvds.h
new file mode 100644
index 00000000..7d6f3fea
--- /dev/null
+++ b/drivers/video/wmt/lvds.h
@@ -0,0 +1,73 @@
+/*++
+ * linux/drivers/video/wmt/lvds.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp.h"
+
+#ifndef LVDS_H
+/* To assert that only one occurrence is included */
+#define LVDS_H
+/*-------------------- MODULE DEPENDENCY -------------------------------------*/
+
+/* following is the C++ header */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+/* #define LVDS_XXXX 1 *//*Example*/
+
+/*-------------------- EXPORTED PRIVATE TYPES---------------------------------*/
+/* typedef void lvds_xxx_t; *//*Example*/
+
+/*-------------------- EXPORTED PRIVATE VARIABLES ----------------------------*/
+#ifdef LVDS_C
+#define EXTERN
+#else
+#define EXTERN extern
+#endif /* ifdef LVDS_C */
+
+/* EXTERN int lvds_xxx; *//*Example*/
+
+#undef EXTERN
+
+/*--------------------- EXPORTED PRIVATE MACROS ------------------------------*/
+/* #define LVDS_XXX_YYY xxxx *//*Example*/
+/*--------------------- EXPORTED PRIVATE FUNCTIONS --------------------------*/
+/* extern void lvds_xxx(void); *//*Example*/
+void lvds_set_enable(vpp_flag_t enable);
+int lvds_get_enable(void);
+void lvds_set_rgb_type(int bpp);
+vdo_color_fmt lvds_get_colfmt(void);
+void lvds_reg_dump(void);
+void lvds_suspend(int sts);
+void lvds_resume(int sts);
+void lvds_init(void);
+void lvds_set_power_down(int pwrdn);
+void lvds_set_sync_polar(int h_lo, int v_lo);
+void lvds_get_sync_polar(int *hsync_hi, int *vsync_hi);
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* ifndef LVDS_H */
+
diff --git a/drivers/video/wmt/mali.c b/drivers/video/wmt/mali.c
new file mode 100644
index 00000000..4ae9509c
--- /dev/null
+++ b/drivers/video/wmt/mali.c
@@ -0,0 +1,759 @@
+/*
+ * Copyright (c) 2008-2011 WonderMedia Technologies, Inc. All Rights Reserved.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+ */
+
+/* Written by Vincent Chen, WonderMedia Technologies, Inc., 2008-2011 */
+
+#include <asm/cacheflush.h>
+#include <linux/io.h>
+#include <linux/slab.h>
+#include <linux/mm.h>
+#include <linux/spinlock.h>
+#include <linux/module.h>
+#include <linux/platform_device.h>
+#include <linux/delay.h>
+#include "mali.h"
+#include <mach/hardware.h>
+
+/* Mali-400 Power Management Kernel Driver */
+
+#define REG_MALI_BADDR (0xD8080000 + WMT_MMAP_OFFSET)
+/*
+#define MALI_PMU_CONTROL
+#define MALI_CLK_CONTROL
+#define USE_PMC_POWER_STATUS
+*/
+
+static spinlock_t mali_spinlock;
+
+static unsigned int mali_max_freq;
+static unsigned int mali_cur_freq;
+
+static int on = 1;
+static int off;
+static int acpi;
+static int mem_size = 0x100000; /* 1 MiB */
+static int debug;
+
+#define MALI_POWER_MASK 0xf
+
+#define OPT_EQUAL(opt, name) (!strncmp(opt, name, strlen(name)))
+#define OPT_INTVAL(opt, name) kstrtoul(opt + strlen(name) + 1, 0, NULL)
+#define OPT_STRVAL(opt, name) (opt + strlen(name))
+
+static inline char *get_opt_string(const char *this_opt, const char *name)
+{
+ const char *p;
+ int i;
+ char *ret;
+
+ p = OPT_STRVAL(this_opt, name);
+ i = 0;
+ while (p[i] && p[i] != ' ' && p[i] != ',')
+ i++;
+ ret = kmalloc(i + 1, GFP_KERNEL);
+ if (ret) {
+ strncpy(ret, p, i);
+ ret[i] = '\0';
+ }
+ return ret;
+}
+
+static inline int get_opt_int(const char *this_opt, const char *name,
+ int *ret)
+{
+ if (!ret)
+ return 0;
+
+ if (!OPT_EQUAL(this_opt, name))
+ return 0;
+
+ *ret = OPT_INTVAL(this_opt, name);
+
+ return 1;
+}
+
+static inline int get_opt_bool(const char *this_opt, const char *name,
+ int *ret)
+{
+ if (!ret)
+ return 0;
+
+ if (OPT_EQUAL(this_opt, name)) {
+ if (this_opt[strlen(name)] == '=')
+ *ret = kstrtoul(this_opt + strlen(name) + 1,
+ 0, NULL);
+ else
+ *ret = 1;
+ } else {
+ if (OPT_EQUAL(this_opt, "no") && OPT_EQUAL(this_opt + 2, name))
+ *ret = 0;
+ else
+ return 0;
+ }
+ return 1;
+}
+
+static int __init malipm_setup(char *options)
+{
+ char *this_opt;
+
+ if (!options || !*options)
+ return 0;
+
+ /* The syntax is:
+ * malipm=[<param>][,<param>=<val>] ...
+ * e.g.,
+ * malipm=on:acpi,mem_size=0x100000
+ */
+
+ while ((this_opt = strsep(&options, ",")) != NULL) {
+ if (!*this_opt)
+ continue;
+ if (get_opt_bool(this_opt, "on", &on))
+ ;
+ else if (get_opt_bool(this_opt, "off", &off))
+ ;
+ else if (get_opt_bool(this_opt, "acpi", &acpi))
+ ;
+ else if (get_opt_int(this_opt, "mem_size", &mem_size))
+ ;
+ else if (get_opt_bool(this_opt, "debug", &debug))
+ ;
+ }
+
+ on = !off;
+ printk(KERN_DEBUG
+ "malipm: on %d, off %d, acpi %d, debug %d, %d MiB\n",
+ on, off, acpi, debug, mem_size >> 20);
+
+ return 0;
+}
+__setup("malipm=", malipm_setup);
+
+static int mali_suspend(u32 cores);
+static int mali_resume(u32 cores);
+static void mali_enable_clock(int enab1le);
+static void mali_enable_power(int enable);
+static void mali_set_memory_base(unsigned int val);
+static void mali_set_memory_size(unsigned int val);
+static void mali_set_mem_validation_base(unsigned int val);
+static void mali_set_mem_validation_size(unsigned int val);
+static void mali_get_memory_base(unsigned int *val);
+static void mali_get_memory_size(unsigned int *val);
+static void mali_get_mem_validation_base(unsigned int *val);
+static void mali_get_mem_validation_size(unsigned int *val);
+
+/* symbols for ARM's Mali.ko */
+unsigned int mali_memory_base;
+EXPORT_SYMBOL(mali_memory_base);
+unsigned int mali_memory_size;
+EXPORT_SYMBOL(mali_memory_size);
+unsigned int mali_mem_validation_base;
+EXPORT_SYMBOL(mali_mem_validation_base);
+unsigned int mali_mem_validation_size;
+EXPORT_SYMBOL(mali_mem_validation_size);
+unsigned int mali_ump_secure_id;
+EXPORT_SYMBOL(mali_ump_secure_id);
+unsigned int (*mali_get_ump_secure_id)(unsigned int addr, unsigned int size);
+EXPORT_SYMBOL(mali_get_ump_secure_id);
+void (*mali_put_ump_secure_id)(unsigned int ump_id);
+EXPORT_SYMBOL(mali_put_ump_secure_id);
+
+/* PMU */
+
+#define REG_MALI400_BASE (0xd8080000 + WMT_MMAP_OFFSET)
+#define REG_MALI400_GP (REG_MALI400_BASE)
+#define REG_MALI400_L2 (REG_MALI400_BASE + 0x1000)
+#define REG_MALI400_PMU (REG_MALI400_BASE + 0x2000)
+#define REG_MALI400_MMU_GP (REG_MALI400_BASE + 0x3000)
+#define REG_MALI400_MMU_PP0 (REG_MALI400_BASE + 0x4000)
+#define REG_MALI400_MMU_PP1 (REG_MALI400_BASE + 0x5000)
+#define REG_MALI400_MMU_PP2 (REG_MALI400_BASE + 0x6000)
+#define REG_MALI400_MMU_PP3 (REG_MALI400_BASE + 0x7000)
+#define REG_MALI400_PP0 (REG_MALI400_BASE + 0x8000)
+#define REG_MALI400_PP1 (REG_MALI400_BASE + 0xa000)
+#define REG_MALI400_PP2 (REG_MALI400_BASE + 0xc000)
+#define REG_MALI400_PP3 (REG_MALI400_BASE + 0xe000)
+
+#define MMU_DTE_ADDR 0x00
+#define MMU_STATUS 0x04
+#define MMU_COMMAND 0x08
+#define MMU_PAGE_FAULT_ADDR 0x0c
+#define MMU_ZAP_ONE_LINE 0x10
+#define MMU_INT_RAWSTAT 0x14
+#define MMU_INT_CLEAR 0x18
+#define MMU_INT_MASK 0x1c
+#define MMU_INT_STATUS 0x20
+
+#ifdef USE_PMC_POWER_STATUS
+#define REG_PMC_BASE (0xd8130000 + WMT_MMAP_OFFSET)
+#define REG_MALI_GP_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0600)
+#define REG_MALI_L2C_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0620)
+#define REG_MALI_PP0_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0624)
+#define REG_MALI_PP1_SHUT_OFF_CONTROL (REG_PMC_BASE + 0x0628)
+
+#define PWR_SEQ_MSK 0x6
+#define PWR_STA_MSK 0xf0
+
+#ifndef REG_VAL32
+#define REG_VAL32 REG_GET32
+#endif
+
+static int wait_powerup(unsigned int reg)
+{
+ int i = 10; /* 10 * 100 us = 1 ms */
+ unsigned int val;
+
+ while (i) {
+ val = ioread32(reg);
+ if ((val & PWR_SEQ_MSK) == 0)
+ break;
+ udelay(100);
+ i--;
+ }
+ if (i == 0)
+ printk(KERN_ERR "%s %d: *0x%08x = %08x\n",
+ __func__, __LINE__, reg, val);
+ while (i) {
+ val = ioread32(reg);
+ if (val & PWR_STA_MSK)
+ break;
+ udelay(100);
+ i--;
+ }
+ if (i == 0)
+ printk(KERN_ERR "%s %d: *0x%08x = %08x\n",
+ __func__, __LINE__, reg, val);
+
+ return i ? 0 : -1;
+}
+
+static int wait_powerdown(unsigned int reg)
+{
+ int i = 10; /* 10 * 100 us = 1 ms */
+ unsigned int val;
+
+ while (i) {
+ val = ioread32(reg);
+ if ((val & PWR_SEQ_MSK) == 0)
+ break;
+ udelay(100);
+ i--;
+ }
+ if (i == 0)
+ printk(KERN_ERR "%s %d: *0x%08x = %08x\n",
+ __func__, __LINE__, reg, val);
+ while (i) {
+ val = ioread32(reg);
+ if ((val & PWR_STA_MSK) == 0)
+ break;
+ udelay(100);
+ i--;
+ }
+ if (i == 0)
+ printk(KERN_ERR "%s %d: *0x%08x = %08x\n",
+ __func__, __LINE__, reg, val);
+
+ return i ? 0 : -1;
+}
+#endif /* USE_PMC_POWER_STATUS */
+
+int mali_platform_wait_powerup(int msk)
+{
+#ifdef USE_PMC_POWER_STATUS
+ int err;
+
+ if (debug)
+ printk(KERN_DEBUG "%s\n", __func__);
+
+ err = 0;
+
+ if (msk & BIT0)
+ err += wait_powerup(REG_MALI_GP_SHUT_OFF_CONTROL);
+ if (msk & BIT1)
+ err += wait_powerup(REG_MALI_L2C_SHUT_OFF_CONTROL);
+ if (msk & BIT2)
+ err += wait_powerup(REG_MALI_PP0_SHUT_OFF_CONTROL);
+ if (msk & BIT3)
+ err += wait_powerup(REG_MALI_PP1_SHUT_OFF_CONTROL);
+
+ if (err)
+ printk(KERN_ERR "%s error\n", __func__);
+
+ return err;
+#else
+ return 0;
+#endif /* USE_PMC_POWER_STATUS */
+}
+EXPORT_SYMBOL(mali_platform_wait_powerup);
+
+int mali_platform_wait_powerdown(int msk)
+{
+#ifdef USE_PMC_POWER_STATUS
+ int err;
+
+ if (debug)
+ printk(KERN_DEBUG "%s\n", __func__);
+
+ err = 0;
+
+ if (msk & BIT0)
+ err += wait_powerdown(REG_MALI_GP_SHUT_OFF_CONTROL);
+ if (msk & BIT1)
+ err += wait_powerdown(REG_MALI_L2C_SHUT_OFF_CONTROL);
+ if (msk & BIT2)
+ err += wait_powerdown(REG_MALI_PP0_SHUT_OFF_CONTROL);
+ if (msk & BIT3)
+ err += wait_powerdown(REG_MALI_PP1_SHUT_OFF_CONTROL);
+
+ if (err)
+ printk(KERN_ERR "%s error\n", __func__);
+
+ return err;
+#else
+ return 0;
+#endif /* USE_PMC_POWER_STATUS */
+}
+EXPORT_SYMBOL(mali_platform_wait_powerdown);
+
+int mali_pmu_power_up(unsigned int msk)
+{
+ int timeout;
+ unsigned int pmu0c;
+ unsigned int pmu08;
+ unsigned int val;
+
+ val = ioread32(REG_MALI400_PMU + 8);
+ if ((val & msk) == 0)
+ return 0;
+
+ pmu08 = ioread32(REG_MALI400_PMU + 8);
+ pmu0c = ioread32(REG_MALI400_PMU + 0xc);
+ iowrite32(0, REG_MALI400_PMU + 0xc);
+
+ asm volatile("" : : : "memory");
+
+ /* PMU_POWER_UP */
+ iowrite32(msk, REG_MALI400_PMU);
+
+ asm volatile("" : : : "memory");
+
+ timeout = 10;
+
+ do {
+ val = ioread32(REG_MALI400_PMU + 8);
+ if ((val & msk) == 0)
+ break;
+ msleep_interruptible(1);
+ timeout--;
+ } while (timeout > 0);
+
+ if (debug) {
+ val = ioread32(REG_MALI400_PMU + 8);
+ if (timeout == 0)
+ printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: fail\n",
+ __func__, msk, pmu08, val);
+ else
+ printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: pass\n",
+ __func__, msk, pmu08, val);
+ }
+
+ iowrite32(pmu0c, REG_MALI400_PMU + 0xc);
+
+ msleep_interruptible(10);
+
+ if (timeout == 0) {
+ printk(KERN_DEBUG "mali pmu power up failure\n");
+ return -1;
+ }
+
+ return 0;
+}
+
+int mali_pmu_power_down(unsigned int msk)
+{
+ unsigned int pmu08;
+ unsigned int pmu0c;
+ int timeout;
+ unsigned int val;
+
+ val = ioread32(REG_MALI400_PMU + 8);
+ if ((val & msk) == msk)
+ return 0;
+
+ pmu08 = ioread32(REG_MALI400_PMU + 8);
+ pmu0c = ioread32(REG_MALI400_PMU + 0xc);
+ iowrite32(0, REG_MALI400_PMU + 0xc);
+
+ asm volatile("" : : : "memory");
+
+ /* PMU_POWER_DOWN */
+ iowrite32(msk, REG_MALI400_PMU + 4);
+
+ asm volatile("" : : : "memory");
+
+ timeout = 10;
+
+ do {
+ val = ioread32(REG_MALI400_PMU + 8);
+ if ((val & msk) == msk)
+ break;
+ msleep_interruptible(1);
+ timeout--;
+ } while (timeout > 0);
+
+ if (debug) {
+ val = ioread32(REG_MALI400_PMU + 8);
+ if (timeout == 0)
+ printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: fail\n",
+ __func__, msk, pmu08, val);
+ else
+ printk(KERN_DEBUG "%s: 0x%x, 0x%08x -> 0x%08x: pass\n",
+ __func__, msk, pmu08, val);
+ }
+
+ iowrite32(pmu0c, REG_MALI400_PMU + 0xc);
+
+ msleep_interruptible(10);
+
+ if (timeout == 0) {
+ printk(KERN_DEBUG "mali pmu power down failure\n");
+ return -1;
+ }
+
+ return 0;
+}
+
+static void mali_show_info(void)
+{
+ /* GP_CONTR_REG_VERSION */
+ printk(KERN_INFO "maligp: version = 0x%08x\n",
+ ioread32(REG_MALI400_GP + 0x6c));
+
+ /* PP0_VERSION */
+ printk(KERN_INFO "malipp: version = 0x%08x\n",
+ ioread32(REG_MALI400_PP0 + 0x1000));
+}
+
+struct mali_device *create_mali_device(void)
+{
+ struct mali_device *dev;
+
+ dev = kcalloc(1, sizeof(struct mali_device), GFP_KERNEL);
+ dev->suspend = &mali_suspend;
+ dev->resume = &mali_resume;
+ dev->enable_clock = &mali_enable_clock;
+ dev->enable_power = &mali_enable_power;
+ dev->set_memory_base = &mali_set_memory_base;
+ dev->set_memory_size = &mali_set_memory_size;
+ dev->set_mem_validation_base = &mali_set_mem_validation_base;
+ dev->set_mem_validation_size = &mali_set_mem_validation_size;
+ dev->get_memory_base = &mali_get_memory_base;
+ dev->get_memory_size = &mali_get_memory_size;
+ dev->get_mem_validation_base = &mali_get_mem_validation_base;
+ dev->get_mem_validation_size = &mali_get_mem_validation_size;
+
+ return dev;
+}
+EXPORT_SYMBOL(create_mali_device);
+
+void release_mali_device(struct mali_device *dev)
+{
+ kfree(dev);
+}
+EXPORT_SYMBOL(release_mali_device);
+
+static int mali_suspend(u32 cores)
+{
+ if (debug)
+ printk(KERN_DEBUG "mali_suspend(%d)\n", cores);
+
+ return 0;
+}
+
+static int mali_resume(u32 cores)
+{
+ if (debug)
+ printk(KERN_DEBUG "mali_resume(%d)\n", cores);
+
+ return 0;
+}
+
+static void mali_enable_clock(int enable)
+{
+ int clk_en;
+
+ /*
+ * if your enable clock with auto_pll_divisor() twice,
+ * then you have to call it at least twice to disable clock.
+ * It is really bad.
+ */
+
+ if (enable) {
+ auto_pll_divisor(DEV_MALI, CLK_ENABLE, 0, 0);
+ if (debug)
+ printk(KERN_DEBUG "Mali clock enabled\n");
+ } else {
+ do {
+ clk_en = auto_pll_divisor(DEV_MALI, CLK_DISABLE, 0, 0);
+ } while (clk_en);
+ if (debug)
+ printk(KERN_DEBUG "Mali clock disabled\n");
+ }
+}
+
+static void mali_enable_power(int enable)
+{
+ /* Mali-400's power was always enabled on WM3481. */
+}
+
+static void mali_set_memory_base(unsigned int val)
+{
+ spin_lock(&mali_spinlock);
+ mali_memory_base = val;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_set_memory_size(unsigned int val)
+{
+ spin_lock(&mali_spinlock);
+ mali_memory_size = val;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_set_mem_validation_base(unsigned int val)
+{
+ spin_lock(&mali_spinlock);
+ mali_mem_validation_base = val;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_set_mem_validation_size(unsigned int val)
+{
+ spin_lock(&mali_spinlock);
+ mali_mem_validation_size = val;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_get_memory_base(unsigned int *val)
+{
+ spin_lock(&mali_spinlock);
+ *val = mali_memory_base;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_get_memory_size(unsigned int *val)
+{
+ spin_lock(&mali_spinlock);
+ *val = mali_memory_size;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_get_mem_validation_base(unsigned int *val)
+{
+ spin_lock(&mali_spinlock);
+ *val = mali_mem_validation_base;
+ spin_unlock(&mali_spinlock);
+}
+
+static void mali_get_mem_validation_size(unsigned int *val)
+{
+ spin_lock(&mali_spinlock);
+ *val = mali_mem_validation_size;
+ spin_unlock(&mali_spinlock);
+}
+
+/* Export functions */
+
+int mali_platform_init_impl(void *data)
+{
+ if (debug)
+ printk(KERN_DEBUG "mali_platform_init_impl\n");
+
+ return 0;
+}
+EXPORT_SYMBOL(mali_platform_init_impl);
+
+int mali_platform_deinit_impl(void *data)
+{
+ if (debug)
+ printk(KERN_DEBUG "mali_platform_deinit_impl\n");
+
+ return 0;
+}
+EXPORT_SYMBOL(mali_platform_deinit_impl);
+
+int mali_platform_powerdown_impl(u32 cores)
+{
+ unsigned int status;
+
+ if (debug)
+ printk(KERN_DEBUG "mali_platform_powerdown_impl(%d)\n", cores);
+
+ status = MALI_POWER_MASK;
+
+ if (acpi == 0)
+ return 0;
+
+#ifdef MALI_PMU_CONTROL
+ status = ioread32(REG_MALI400_PMU + 8);
+
+ if ((status & MALI_POWER_MASK) != MALI_POWER_MASK)
+ mali_pmu_power_down(MALI_POWER_MASK);
+#endif
+
+#ifdef MALI_CLK_CONTROL
+ spin_lock(&mali_spinlock);
+ mali_enable_clock(0);
+ mali_enable_power(0);
+ spin_unlock(&mali_spinlock);
+#endif
+
+ return 0;
+}
+EXPORT_SYMBOL(mali_platform_powerdown_impl);
+
+int mali_platform_powerup_impl(u32 cores)
+{
+ unsigned int status;
+
+ if (debug)
+ printk(KERN_DEBUG "mali_platform_powerup_impl(%d)\n", cores);
+
+#ifdef MALI_PMU_CONTROL
+ status = ioread32(REG_MALI400_PMU + 8);
+
+ /* printk("mali pmu: status = 0x08%x\n", status); */
+
+ if ((status & MALI_POWER_MASK) != 0)
+ mali_pmu_power_up(MALI_POWER_MASK);
+#else
+ status = 0;
+#endif
+
+#ifdef MALI_CLK_CONTROL
+ spin_lock(&mali_spinlock);
+ mali_enable_power(1);
+ mali_enable_clock(1);
+ spin_unlock(&mali_spinlock);
+#endif
+
+ return 0;
+}
+EXPORT_SYMBOL(mali_platform_powerup_impl);
+
+void mali_gpu_utilization_handler_impl(u32 utilization)
+{
+ unsigned int freq;
+ unsigned int oldfreq;
+ int ret;
+
+ if (acpi < 2 || mali_max_freq == 0)
+ return;
+
+ utilization = (utilization + 63) & ~63;
+ oldfreq = mali_cur_freq;
+ freq = ((utilization * mali_max_freq) >> 8) + 64;
+ ret = auto_pll_divisor(DEV_MALI, SET_DIV, 2, freq); /* MHz */
+ if (ret > 0)
+ mali_cur_freq = (ret >> 20) + 1; /* MHz */
+ printk(KERN_DEBUG "%s: %d, %d -> %d (%d) MHz\n",
+ __func__, utilization, oldfreq, mali_cur_freq, freq);
+}
+EXPORT_SYMBOL(mali_gpu_utilization_handler_impl);
+
+void set_mali_parent_power_domain(struct platform_device *dev)
+{
+ /* No implemented yet */
+}
+EXPORT_SYMBOL(set_mali_parent_power_domain);
+
+static int __init mali_init(void)
+{
+ unsigned long smem_start;
+ unsigned long smem_len;
+ int err = 0;
+ int ret;
+
+ spin_lock_init(&mali_spinlock);
+
+ smem_start = num_physpages << PAGE_SHIFT;
+
+ smem_len = mem_size;
+
+ mali_set_memory_base(smem_start);
+ mali_set_memory_size(smem_len);
+ mali_set_mem_validation_base(0);
+ mali_set_mem_validation_size(0);
+
+ mali_ump_secure_id = (unsigned int) -1;
+ mali_get_ump_secure_id = NULL;
+ mali_put_ump_secure_id = NULL;
+
+ if (off)
+ return -1;
+
+ mali_enable_power(1);
+ mali_enable_clock(1);
+
+ /* Wait for power stable */
+ msleep_interruptible(1);
+
+ /* Verify Mali-400 PMU */
+ err += mali_pmu_power_down(MALI_POWER_MASK);
+ if (!err)
+ err += mali_pmu_power_up(MALI_POWER_MASK);
+ if (!err)
+ mali_show_info();
+
+ if (acpi)
+ err += mali_pmu_power_down(MALI_POWER_MASK);
+
+ if (acpi > 1) {
+ ret = auto_pll_divisor(DEV_MALI, GET_FREQ, 0, 0);
+ if (ret > 0) {
+ mali_max_freq = (ret >> 20) + 1; /* MHz */
+ mali_cur_freq = mali_max_freq;
+ }
+ } else {
+ mali_max_freq = mali_cur_freq = 0;
+ }
+
+ /* Power on all Mali core at bootup, otherwise Mali driver will fail
+ * at driver/src/devicedrv/mali/common/mali_pp.c: mali_pp_reset_wait().
+ */
+ mali_pmu_power_up(MALI_POWER_MASK);
+
+ return err;
+}
+
+static void __exit mali_exit(void)
+{
+ mali_enable_clock(0);
+ mali_enable_power(0);
+}
+
+module_init(mali_init);
+module_exit(mali_exit);
+
+MODULE_AUTHOR("WonderMedia Technologies, Inc.");
+MODULE_DESCRIPTION("Mali PM Kernel Driver");
+MODULE_LICENSE("GPL");
+
diff --git a/drivers/video/wmt/mali.h b/drivers/video/wmt/mali.h
new file mode 100644
index 00000000..34605eae
--- /dev/null
+++ b/drivers/video/wmt/mali.h
@@ -0,0 +1,49 @@
+/*
+ * Copyright (c) 2008-2011 WonderMedia Technologies, Inc. All Rights Reserved.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+ */
+
+#ifndef MALI_H
+#define MALI_H
+struct mali_device {
+ int (*suspend)(u32 cores);
+ int (*resume)(u32 cores);
+ void (*enable_clock)(int enable);
+ void (*enable_power)(int enable);
+ void (*set_memory_base)(unsigned int val);
+ void (*set_memory_size)(unsigned int val);
+ void (*set_mem_validation_base)(unsigned int val);
+ void (*set_mem_validation_size)(unsigned int val);
+ void (*get_memory_base)(unsigned int *val);
+ void (*get_memory_size)(unsigned int *val);
+ void (*get_mem_validation_base)(unsigned int *val);
+ void (*get_mem_validation_size)(unsigned int *val);
+};
+
+struct mali_device *create_mali_device(void);
+void release_mali_device(struct mali_device *dev);
+
+int mali_platform_init_impl(void *data);
+int mali_platform_deinit_impl(void *data);
+int mali_platform_powerdown_impl(u32 cores);
+int mali_platform_powerup_impl(u32 cores);
+void mali_gpu_utilization_handler_impl(u32 utilization);
+void set_mali_parent_power_domain(struct platform_device *dev);
+
+extern unsigned long msleep_interruptible(unsigned int msecs);
+#endif /* MALI_H */
diff --git a/drivers/video/wmt/parse-edid.c b/drivers/video/wmt/parse-edid.c
new file mode 100644
index 00000000..c0f18115
--- /dev/null
+++ b/drivers/video/wmt/parse-edid.c
@@ -0,0 +1,1027 @@
+/*++
+ * linux/drivers/video/wmt/parse-edid.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define PARSE_EDID_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+#include "vpp.h"
+#include "edid.h"
+#include "hdmi.h"
+
+const char edid_v1_header[] = {
+ 0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00};
+
+const char edid_v1_descriptor_flag[] = { 0x00, 0x00 };
+
+#define COMBINE_HI_8LO(hi, lo) \
+ ((((unsigned)hi) << 8) | (unsigned)lo)
+
+#define COMBINE_HI_4LO(hi, lo) \
+ ((((unsigned)hi) << 4) | (unsigned)lo)
+
+#define UPPER_NIBBLE(x) \
+ (((128|64|32|16) & (x)) >> 4)
+
+#define LOWER_NIBBLE(x) \
+ ((1|2|4|8) & (x))
+
+#define MONITOR_NAME 0xfc
+#define MONITOR_LIMITS 0xfd
+#define UNKNOWN_DESCRIPTOR -1
+#define DETAILED_TIMING_BLOCK -2
+
+edid_timing_t edid_establish_timing[] = {
+ { 800, 600, 60 }, { 800, 600, 56 }, { 640, 480, 75 }, { 640, 480, 72 },
+ { 640, 480, 67 }, { 640, 480, 60 }, { 720, 400, 88 }, { 720, 400, 70 },
+ { 1280, 1024, 75 }, { 1024, 768, 75 }, { 1024, 768, 70 },
+ { 1024, 768, 60 }, { 1024, 768, 87 }, { 832, 624, 75 },
+ { 800, 600, 75 }, { 800, 600, 72 }, { 1152, 870, 75 }
+};
+int edid_msg_enable;
+int edid_disable;
+edid_parsed_t edid_parsed;
+
+#undef DBGMSG
+#define DBGMSG(fmt, args...) if (edid_msg_enable) \
+ DPRINT(fmt, ## args)
+
+static int block_type(char *block)
+{
+ if (!memcmp(edid_v1_descriptor_flag, block, 2)) {
+ /* descriptor */
+ if (block[2] != 0)
+ return UNKNOWN_DESCRIPTOR;
+ return block[3];
+ }
+ /* detailed timing block */
+ return DETAILED_TIMING_BLOCK;
+} /* End of block_type() */
+
+static char *get_vendor_sign(char *block, char *sign)
+{
+ unsigned short h;
+
+ /*
+ 08h WORD big-endian manufacturer ID (see #00136)
+ bits 14-10: first letter (01h='A', 02h='B', etc.)
+ bits 9-5: second letter
+ bits 4-0: third letter
+ */
+ h = COMBINE_HI_8LO(block[0], block[1]);
+ sign[0] = ((h >> 10) & 0x1f) + 'A' - 1;
+ sign[1] = ((h >> 5) & 0x1f) + 'A' - 1;
+ sign[2] = (h & 0x1f) + 'A' - 1;
+ sign[3] = 0;
+ return sign;
+} /* End of get_vendor_sign() */
+
+static char *get_monitor_name(char *block, char *name)
+{
+#define DESCRIPTOR_DATA 5
+
+ char *ptr = block + DESCRIPTOR_DATA;
+ unsigned i;
+
+ for (i = 0; i < 13; i++, ptr++) {
+ if (*ptr == 0xa) {
+ name[i] = 0;
+ return name;
+ }
+ name[i] = *ptr;
+ }
+ return name;
+} /* End of get_monitor_name() */
+
+static int parse_timing_description(char *dtd, edid_info_t *info)
+{
+#define PIXEL_CLOCK_LO ((unsigned)dtd[0])
+#define PIXEL_CLOCK_HI ((unsigned)dtd[1])
+#define PIXEL_CLOCK (COMBINE_HI_8LO(PIXEL_CLOCK_HI, PIXEL_CLOCK_LO) * 10000)
+#define H_ACTIVE_LO ((unsigned)dtd[2])
+#define H_BLANKING_LO ((unsigned)dtd[3])
+#define H_ACTIVE_HI UPPER_NIBBLE((unsigned)dtd[4])
+#define H_ACTIVE COMBINE_HI_8LO(H_ACTIVE_HI, H_ACTIVE_LO)
+#define H_BLANKING_HI LOWER_NIBBLE((unsigned)dtd[4])
+#define H_BLANKING COMBINE_HI_8LO(H_BLANKING_HI, H_BLANKING_LO)
+#define V_ACTIVE_LO ((unsigned)dtd[5])
+#define V_BLANKING_LO ((unsigned)dtd[6])
+#define V_ACTIVE_HI UPPER_NIBBLE((unsigned)dtd[7])
+#define V_ACTIVE COMBINE_HI_8LO(V_ACTIVE_HI, V_ACTIVE_LO)
+#define V_BLANKING_HI LOWER_NIBBLE((unsigned)dtd[7])
+#define V_BLANKING COMBINE_HI_8LO(V_BLANKING_HI, V_BLANKING_LO)
+#define H_SYNC_OFFSET_LO ((unsigned)dtd[8])
+#define H_SYNC_WIDTH_LO ((unsigned)dtd[9])
+#define V_SYNC_OFFSET_LO UPPER_NIBBLE((unsigned)dtd[10])
+#define V_SYNC_WIDTH_LO LOWER_NIBBLE((unsigned)dtd[10])
+#define V_SYNC_WIDTH_HI ((unsigned)dtd[11] & (1|2))
+#define V_SYNC_OFFSET_HI (((unsigned)dtd[11] & (4|8)) >> 2)
+#define H_SYNC_WIDTH_HI (((unsigned)dtd[11] & (16|32)) >> 4)
+#define H_SYNC_OFFSET_HI (((unsigned)dtd[11] & (64|128)) >> 6)
+#define V_SYNC_WIDTH COMBINE_HI_4LO(V_SYNC_WIDTH_HI, V_SYNC_WIDTH_LO)
+#define V_SYNC_OFFSET COMBINE_HI_4LO(V_SYNC_OFFSET_HI, V_SYNC_OFFSET_LO)
+#define H_SYNC_WIDTH COMBINE_HI_4LO(H_SYNC_WIDTH_HI, H_SYNC_WIDTH_LO)
+#define H_SYNC_OFFSET COMBINE_HI_4LO(H_SYNC_OFFSET_HI, H_SYNC_OFFSET_LO)
+#define H_SIZE_LO ((unsigned)dtd[12])
+#define V_SIZE_LO ((unsigned)dtd[13])
+#define H_SIZE_HI UPPER_NIBBLE((unsigned)dtd[14])
+#define V_SIZE_HI LOWER_NIBBLE((unsigned)dtd[14])
+#define H_SIZE COMBINE_HI_8LO(H_SIZE_HI, H_SIZE_LO)
+#define V_SIZE COMBINE_HI_8LO(V_SIZE_HI, V_SIZE_LO)
+#define H_BORDER ((unsigned)dtd[15])
+#define V_BORDER ((unsigned)dtd[16])
+#define FLAGS ((unsigned) dtd[17])
+#define INTERLACED (FLAGS & 128)
+#define SYNC_TYPE (FLAGS & 3 << 3) /* bits 4,3 */
+#define SYNC_SEPARATE (3 << 3)
+#define HSYNC_POSITIVE (FLAGS & 4)
+#define VSYNC_POSITIVE (FLAGS & 2)
+
+ int htotal, vtotal;
+ int i;
+ struct fb_videomode *t;
+ int fps;
+ int vmul;
+
+ htotal = H_ACTIVE + H_BLANKING;
+ vtotal = V_ACTIVE + V_BLANKING;
+
+ DBGMSG("Detail Timing: \"%dx%d\"\n", H_ACTIVE, V_ACTIVE);
+ DBGMSG("\tVfreq %dHz, Hfreq %dkHz\n",
+ PIXEL_CLOCK / (vtotal * htotal),
+ PIXEL_CLOCK / (htotal * 1000));
+ DBGMSG("\tDotClock\t%d\n", PIXEL_CLOCK / 1000000);
+ DBGMSG("\tHTimings\t%u %u %u %u\n", H_ACTIVE,
+ H_ACTIVE + H_SYNC_OFFSET,
+ H_ACTIVE + H_SYNC_OFFSET + H_SYNC_WIDTH,
+ htotal);
+
+ DBGMSG("\tVTimings\t%u %u %u %u\n", V_ACTIVE,
+ V_ACTIVE + V_SYNC_OFFSET,
+ V_ACTIVE + V_SYNC_OFFSET + V_SYNC_WIDTH,
+ vtotal);
+
+ if (INTERLACED || (SYNC_TYPE == SYNC_SEPARATE)) {
+ DBGMSG("\tFlags\t%s\"%sHSync\" \"%sVSync\"\n",
+ INTERLACED ? "\"Interlace\" " : "",
+ HSYNC_POSITIVE ? "+" : "-",
+ VSYNC_POSITIVE ? "+" : "-");
+ }
+
+ for (i = 0; i < 4; i++) {
+ t = &info->detail_timing[i];
+ if (t->pixclock == 0)
+ break;
+ }
+
+ if (i >= 4) {
+ DBGMSG("*W* slot full\n");
+ return 0;
+ }
+
+ t->pixclock = KHZ2PICOS(PIXEL_CLOCK / 1000);
+ t->right_margin = H_SYNC_OFFSET;
+ t->hsync_len = H_SYNC_WIDTH;
+ t->xres = H_ACTIVE;
+ t->left_margin = H_BLANKING - (H_SYNC_WIDTH + H_SYNC_OFFSET);
+ t->vmode = INTERLACED ? FB_VMODE_INTERLACED : FB_VMODE_NONINTERLACED;
+ vmul = (INTERLACED) ? 2 : 1;
+ t->lower_margin = V_SYNC_OFFSET * vmul;
+ t->vsync_len = V_SYNC_WIDTH * vmul;
+ t->yres = V_ACTIVE * vmul;
+ t->upper_margin = V_BLANKING - (V_SYNC_WIDTH + V_SYNC_OFFSET);
+
+ fps = vpp_calc_refresh(PIXEL_CLOCK, htotal, vtotal);
+ t->refresh = fps;
+ t->sync = HSYNC_POSITIVE ? (FB_SYNC_HOR_HIGH_ACT) : 0;
+ t->sync |= VSYNC_POSITIVE ? (FB_SYNC_VERT_HIGH_ACT) : 0;
+
+ if (vout_check_ratio_16_9(H_ACTIVE, V_ACTIVE))
+ info->option |= EDID_OPT_16_9;
+#if 0
+ DBGMSG("%dx%d,%d,%s\n", H_ACTIVE, V_ACTIVE, t->pixel_clock,
+ (info->option & EDID_OPT_16_9) ? "16:9" : "4:3");
+#endif
+ return 0;
+} /* End of parse_timing_description() */
+
+static int parse_dpms_capabilities(char flags)
+{
+#define DPMS_ACTIVE_OFF (1 << 5)
+#define DPMS_SUSPEND (1 << 6)
+#define DPMS_STANDBY (1 << 7)
+
+ DBGMSG("# DPMS capabilities: Active off:%s Suspend:%s Standby:%s\n\n",
+ (flags & DPMS_ACTIVE_OFF) ? "yes" : "no",
+ (flags & DPMS_SUSPEND) ? "yes" : "no",
+ (flags & DPMS_STANDBY) ? "yes" : "no");
+ return 0;
+} /* End of parse_dpms_capabilities() */
+
+static int parse_monitor_limits(char *block, edid_info_t *info)
+{
+#define V_MIN_RATE block[5]
+#define V_MAX_RATE block[6]
+#define H_MIN_RATE block[7]
+#define H_MAX_RATE block[8]
+#define MAX_PIXEL_CLOCK (((int)block[9]) * 10)
+#define GTF_SUPPORT block[10]
+
+ DBGMSG("Monitor limit\n");
+ DBGMSG("\tHorizontal Frequency: %u-%u Hz\n", H_MIN_RATE, H_MAX_RATE);
+ DBGMSG("\tVertical Frequency: %u-%u kHz\n", V_MIN_RATE, V_MAX_RATE);
+ if (MAX_PIXEL_CLOCK == 10 * 0xff) {
+ DBGMSG("\tMax dot clock not given\n");
+ } else {
+ DBGMSG("\tMax dot clock (video bandwidth) %u MHz\n",
+ (int)MAX_PIXEL_CLOCK);
+ info->pixel_clock_limit = MAX_PIXEL_CLOCK;
+ }
+
+ if (GTF_SUPPORT)
+ DBGMSG("\tEDID version 3 GTF given: contact author\n");
+ return 0;
+} /* End of parse_monitor_limits() */
+
+static int get_established_timing(char *edid, edid_info_t *info)
+{
+ char time_1, time_2;
+
+ time_1 = edid[ESTABLISHED_TIMING_I];
+ time_2 = edid[ESTABLISHED_TIMING_II];
+ info->establish_timing = time_1 + (time_2 << 8);
+
+ /*---------------------------------------------------------------------
+ 35: ESTABLISHED TIMING I
+ bit 7-0: 720x400@70 Hz, 720x400@88 Hz, 640x480@60 Hz, 640x480@67 Hz,
+ 640x480@72 Hz, 640x480@75 Hz, 800x600@56 Hz, 800x600@60 Hz
+ ---------------------------------------------------------------------*/
+ DBGMSG("Established Timimgs I: 0x%x\n", time_1);
+ if (time_1 & 0x80)
+ DBGMSG("\t%dx%d@%dHz\n", 720, 400, 70);
+ if (time_1 & 0x40)
+ DBGMSG("\t%dx%d@%dHz\n", 720, 400, 88);
+ if (time_1 & 0x20)
+ DBGMSG("\t%dx%d@%dHz\n", 640, 480, 60);
+ if (time_1 & 0x10)
+ DBGMSG("\t%dx%d@%dHz\n", 640, 480, 67);
+ if (time_1 & 0x08)
+ DBGMSG("\t%dx%d@%dHz\n", 640, 480, 72);
+ if (time_1 & 0x04)
+ DBGMSG("\t%dx%d@%dHz\n", 640, 480, 75);
+ if (time_1 & 0x02)
+ DBGMSG("\t%dx%d@%dHz\n", 800, 600, 56);
+ if (time_1 & 0x01)
+ DBGMSG("\t%dx%d@%dHz\n", 800, 600, 60);
+
+ /*---------------------------------------------------------------------
+ 36: ESTABLISHED TIMING II
+ bit 7-0: 800x600@72 Hz, 800x600@75 Hz, 832x624@75 Hz,
+ 1024x768@87 Hz (Interlaced), 1024x768@60 Hz, 1024x768@70 Hz,
+ 1024x768@75 Hz, 1280x1024@75 Hz
+ ---------------------------------------------------------------------*/
+ DBGMSG("Established Timimgs II: 0x%x\n", time_2);
+ if (time_2 & 0x80)
+ DBGMSG("\t%dx%d@%dHz\n", 800, 600, 72);
+ if (time_2 & 0x40)
+ DBGMSG("\t%dx%d@%dHz\n", 800, 600, 75);
+ if (time_2 & 0x20)
+ DBGMSG("\t%dx%d@%dHz\n", 832, 624, 75);
+ if (time_2 & 0x10)
+ DBGMSG("\t%dx%d@%dHz (Interlace)\n", 1024, 768, 87);
+ if (time_2 & 0x08)
+ DBGMSG("\t%dx%d@%dHz\n", 1024, 768, 60);
+ if (time_2 & 0x04)
+ DBGMSG("\t%dx%d@%dHz\n", 1024, 768, 70);
+ if (time_2 & 0x02)
+ DBGMSG("\t%dx%d@%dHz\n", 1024, 768, 75);
+ if (time_2 & 0x01)
+ DBGMSG("\t%dx%d@%dHz\n", 1280, 1024, 75);
+ return 0;
+} /* End of get_established_timing() */
+
+static int get_standard_timing(char *edid, edid_info_t *info)
+{
+ char *ptr = edid + STANDARD_TIMING_IDENTIFICATION_START;
+ int h_res, v_res, v_freq;
+ int byte_1, byte_2, aspect, i;
+
+ /*---------------------------------------------------------------------
+ First byte
+ Horizontal resolution. Multiply by 8, then add 248 for actual value.
+ Second byte
+ bit 7-6: Aspect ratio. Actual vertical resolution depends on horizontal
+ resolution.
+ 00=16:10, 01=4:3, 10=5:4, 11=16:9 (00=1:1 prior to v1.3)
+ bit 5-0: Vertical frequency. Add 60 to get actual value.
+ ---------------------------------------------------------------------*/
+ DBGMSG("Standard Timing Identification\n");
+ for (i = 0; i < STANDARD_TIMING_IDENTIFICATION_SIZE / 2; i++) {
+ byte_1 = *ptr++;
+ byte_2 = *ptr++;
+ if ((byte_1 == 0x01) && (byte_2 == 0x01))
+ break;
+ h_res = (byte_1 * 8) + 248;
+ aspect = byte_2 & 0xC0;
+ switch (aspect) {
+ default:
+ case 0x00:
+ v_res = h_res * 10/16;
+ break;
+ case 0x40:
+ v_res = h_res * 3/4;
+ break;
+ case 0x80:
+ v_res = h_res * 4/5;
+ break;
+ case 0xC0:
+ v_res = h_res * 9/16;
+ break;
+ }
+ v_freq = (byte_2 & 0x1F) + 60;
+ DBGMSG("\t%dx%d@%dHz\n", h_res, v_res, v_freq);
+ info->standard_timing[i].resx = h_res;
+ info->standard_timing[i].resy = v_res;
+ info->standard_timing[i].freq = v_freq;
+ }
+ return 0;
+} /* End of get_standard_timing() */
+
+static int edid_parse_v1(char *edid, edid_info_t *info)
+{
+ char *block;
+ char *monitor_name = 0;
+ char monitor_alt_name[100];
+ char vendor_sign[4];
+ int i, ret = 0;
+
+ if (edid_checksum(edid, EDID_LENGTH)) {
+ DBG_ERR("checksum failed\n");
+ ret = -1;
+ goto parse_end;
+ }
+
+ if (memcmp(edid+EDID_HEADER, edid_v1_header, EDID_HEADER_END+1)) {
+ DBGMSG("*E* first bytes don't match EDID version 1 header\n");
+ ret = -1;
+ goto parse_end;
+ }
+
+//#ifdef DEBUG
+ if(edid_msg_enable)
+ edid_dump(edid);
+//#endif
+
+ DBGMSG("[EDID] EDID version: %d.%d\n",
+ (int)edid[EDID_STRUCT_VERSION],
+ (int)edid[EDID_STRUCT_REVISION]);
+
+ get_vendor_sign(edid + ID_MANUFACTURER_NAME, (char *) &vendor_sign);
+
+ /*---------------------------------------------------------------------
+ Parse Monitor name
+ ---------------------------------------------------------------------*/
+ block = edid + DETAILED_TIMING_DESCRIPTIONS_START;
+ for (i = 0; i < NO_DETAILED_TIMING_DESCRIPTIONS; i++,
+ block += DETAILED_TIMING_DESCRIPTION_SIZE) {
+ if (block_type(block) == MONITOR_NAME) {
+ monitor_name =
+ get_monitor_name(block, monitor_alt_name);
+ break;
+ }
+ }
+
+ if (!monitor_name) {
+ /* Stupid djgpp hasn't snDBGMSG so we have to
+ hack something together */
+ if (strlen(vendor_sign) + 10 > sizeof(monitor_alt_name))
+ vendor_sign[3] = 0;
+
+ sprintf(monitor_alt_name, "%s:%02x%02x",
+ vendor_sign, edid[ID_MODEL], edid[ID_MODEL+1]);
+ monitor_name = monitor_alt_name;
+ }
+
+ DBGMSG("Identifier \"%s\"\n", monitor_name);
+ DBGMSG("VendorName \"%s\"\n", vendor_sign);
+ DBGMSG("ModelName \"%s\"\n", monitor_name);
+
+ memset(edid_parsed.tv_name.vendor_name, 0, sizeof(edid_parsed.tv_name.vendor_name));
+ strcpy(edid_parsed.tv_name.vendor_name, vendor_sign);
+
+ memset(edid_parsed.tv_name.monitor_name, 0, sizeof(edid_parsed.tv_name.monitor_name));
+ if(strlen(monitor_name) < MONITOR_NAME_LEN)
+ strcpy(edid_parsed.tv_name.monitor_name, monitor_name);
+ else
+ strncpy(edid_parsed.tv_name.monitor_name, monitor_name, MONITOR_NAME_LEN - 1);
+
+ parse_dpms_capabilities(edid[DPMS_FLAGS]);
+
+ /*---------------------------------------------------------------------
+ Parse ESTABLISHED TIMING I and II
+ ---------------------------------------------------------------------*/
+ get_established_timing(edid, info);
+
+ /*---------------------------------------------------------------------
+ Parse STANDARD TIMING IDENTIFICATION
+ ---------------------------------------------------------------------*/
+ get_standard_timing(edid, info);
+
+ block = edid + DETAILED_TIMING_DESCRIPTIONS_START;
+ for (i = 0; i < NO_DETAILED_TIMING_DESCRIPTIONS; i++,
+ block += DETAILED_TIMING_DESCRIPTION_SIZE) {
+ if (block_type(block) == MONITOR_LIMITS)
+ parse_monitor_limits(block, info);
+ }
+
+ block = edid + DETAILED_TIMING_DESCRIPTIONS_START;
+ for (i = 0; i < NO_DETAILED_TIMING_DESCRIPTIONS; i++,
+ block += DETAILED_TIMING_DESCRIPTION_SIZE) {
+ if (block_type(block) == DETAILED_TIMING_BLOCK)
+ parse_timing_description(block, info);
+ }
+parse_end:
+ return ret;
+}
+
+void edid_parse_CEA_VendorSpecDataBlock(char *block, int len,
+ edid_info_t *info)
+{
+ int index;
+ char temp;
+
+ DBGMSG("Vendor Spec Data Block\n");
+ if (len < 5) /* min size */
+ return;
+ /* IEEE Registration Identifier 0x000C03 */
+ if ((block[1] == 0x03) && (block[2] == 0x0C) &&
+ (block[3] == 0x0)) {
+ info->option |= EDID_OPT_HDMI;
+ DBGMSG("\t support HDMI\n");
+ }
+ DBGMSG("\t Source Physical Addr %d.%d.%d.%d\n",
+ (block[4] & 0xF0) >> 4, block[4] & 0x0F,
+ (block[5] & 0xF0) >> 4, block[5] & 0x0F);
+ info->hdmi_phy_addr = (block[4] << 8) + block[5];
+
+ /* extersion fields */
+ if (len < 8)
+ return;
+ DBGMSG("\t%s support AI\n",
+ (block[6] & 0x80) ? "" : "no");
+ DBGMSG("\t%s support 30 bits/pixel(10 bits/color)\n",
+ (block[6] & 0x40) ? "" : "no");
+ DBGMSG("\t%s support 36 bits/pixel(12 bits/color)\n",
+ (block[6] & 0x20) ? "" : "no");
+ DBGMSG("\t%s support 48 bits/pixel(16 bits/color)\n",
+ (block[6] & 0x10) ? "" : "no");
+ DBGMSG("\t%s support YUV444 in Deep Color mode\n",
+ (block[6] & 0x08) ? "" : "no");
+ DBGMSG("\t%s support DVI dual-link\n",
+ (block[6] & 0x01) ? "" : "no");
+ DBGMSG("\tMax TMDS Clock %d MHz\n", block[7] * 5);
+ temp = block[8];
+ index = 9;
+
+ if (temp & BIT(7)) {
+ DBGMSG("\tVideo Latency %d,Audio Latency %d\n",
+ block[index], block[index+1]);
+ index += 2;
+ }
+
+ if (temp & BIT(6)) {
+ DBGMSG("\tInterlaced Video Latency %d,Audio Latency %d\n",
+ block[index], block[index+1]);
+ index += 2;
+ }
+
+ if (temp & BIT(5)) {
+ int hdmi_xx_len, hdmi_3d_len;
+
+ DBGMSG("\tHDMI Video present\n");
+ temp = block[index];
+ if (temp & 0x80)
+ DBGMSG("\t\t3D present support Mandatory formats\n");
+ if (temp & 0x60)
+ DBGMSG("\t\t3D Multi present %d\n", (temp & 0x60) >> 5);
+ DBGMSG("\t\tImage size %d\n", (temp & 0x18) >> 3);
+ hdmi_xx_len = (block[index + 1] & 0xE0) >> 5;
+ hdmi_3d_len = block[index + 1] & 0x1F;
+ DBGMSG("\t\thdmi_xx_len %d,hdmi_3d_len %d\n",
+ hdmi_xx_len, hdmi_3d_len);
+ index += 2;
+
+ if (hdmi_xx_len) {
+ /* Refer to HDMI Spec Ver 1.4a */
+ index += hdmi_xx_len;
+ }
+
+ if (hdmi_3d_len) {
+ int struct_all_3d = 0;
+ int mask_3d = 0;
+ char vic_order_2d, struct_3d, detail_3d;
+
+ hdmi_3d_len += index;
+ switch (temp & 0x60) {
+ case 0x40:
+ struct_all_3d = (block[index] << 8) +
+ block[index + 1];
+ mask_3d = (block[index + 2] << 8) +
+ block[index + 3]; /* 3D support mask */
+ DBGMSG("\t\t3D struct 0x%x,mask 0x%x\n",
+ struct_all_3d, mask_3d);
+ index += 4;
+ break;
+ case 0x20:
+ struct_all_3d = (block[index] << 8) +
+ block[index + 1];
+ DBGMSG("\t\t3D struct 0x%x\n", struct_all_3d);
+ index += 2;
+ break;
+ default:
+ break;
+ }
+ /* 3D support type */
+ if (struct_all_3d & BIT0)
+ DBGMSG("\t\tSupport Frame packing\n");
+ if (struct_all_3d & BIT6)
+ DBGMSG("\t\tSupport Top and Bottom\n");
+ if (struct_all_3d & BIT8)
+ DBGMSG("\t\tSupport Side by Side\n");
+
+ DBGMSG("\t\t[3D Structure type 0-Frame packing");
+ DBGMSG(",6-Top and Bottom,8-Side by Side]\n");
+ while (index < hdmi_3d_len) {
+ vic_order_2d = (block[index] & 0xF0) >> 4;
+ struct_3d = block[index] & 0x0F;
+ index++;
+ if (struct_3d & 0x8) {
+ detail_3d = (block[index] & 0xF0) >> 4;
+ index++;
+ } else {
+ detail_3d = 0;
+ }
+ DBGMSG("\t\tVIC %d,3D struct %d,detail %d\n",
+ vic_order_2d, struct_3d, detail_3d);
+ }
+ }
+ }
+}
+
+static int edid_parse_CEA(char *edid, edid_info_t *info)
+{
+ char *block, *block_end;
+ char checksum = 0;
+ int i, len;
+ char audio_format;
+ int num = 0, sadcnt_in_block;
+
+ memset(edid_parsed.sad, 0, sizeof(edid_parsed.sad));
+
+ if (edid[0] != 0x2)
+ return -1;
+
+ for (i = 0; i < EDID_LENGTH; i++)
+ checksum += edid[i];
+
+ if (checksum != 0) {
+ DPRINT("*E* CEA EDID checksum (0x%02x) failed - data is corrupt\n", checksum);
+ info->option |= (EDID_OPT_HDMI + EDID_OPT_AUDIO);
+ return -1;
+ }
+
+//#ifdef DEBUG
+ if(edid_msg_enable)
+ edid_dump(edid);
+//#endif
+
+ DBGMSG("[EDID] CEA EDID Version %d.%d\n", edid[0], edid[1]);
+
+ if (edid[1] >= 2) {
+ info->option |= (edid[3] & 0xF0);
+ DBGMSG("\t%s support 422\n", (edid[3] & 0x10) ? "" : "no");
+ DBGMSG("\t%s support 444\n", (edid[3] & 0x20) ? "" : "no");
+ DBGMSG("\t%s support audio\n", (edid[3] & 0x40) ? "" : "no");
+ DBGMSG("\t%s support underscan\n",
+ (edid[3] & 0x80) ? "" : "no");
+ }
+
+ block_end = edid + edid[2];
+ block = edid + 4;
+ do {
+ len = block[0] & 0x1F;
+ switch (((block[0] & 0xE0)>>5)) {
+ case 1: /* Audio Data Block */
+ DBGMSG("Audio Data Block (0x%02X)\n", block[0]);
+ info->option |= EDID_OPT_AUDIO;
+ sadcnt_in_block = len / 3;
+
+ for(i = 0; i < sadcnt_in_block; i++){
+ if(num < AUD_SAD_NUM) {
+ edid_parsed.sad[num].flag = 1;
+ edid_parsed.sad[num].sad_byte[0] = block[i * 3 + 1];
+ edid_parsed.sad[num].sad_byte[1] = block[i * 3 + 2];
+ edid_parsed.sad[num].sad_byte[2] = block[i * 3 + 3];
+ num++;
+ } else {
+ DPRINT("Lose SAD info: 0x%02X 0x%02X 0x%02X\n",
+ block[i * 3 + 1], block[i * 3 + 2], block[i * 3 + 3]);
+ }
+
+ DBGMSG("\t ======== SDA %d ========\n", i);
+ DBGMSG("\t SDA Data: 0x%02X 0x%02X 0x%02X\n",
+ block[i * 3 + 1], block[i * 3 + 2], block[i * 3 + 3]);
+
+ audio_format = (block[i * 3 + 1] & 0x78) >> 3;
+ switch (audio_format) {
+ default:
+ case 0: /* reserved */
+ case 15:/* reserved */
+ DBGMSG("\t Reserved Audio Fmt\n");
+ break;
+ case 1: /* LPCM */
+ DBGMSG("\t Audio Fmt: LPCM\n");
+ break;
+ case 2: /* AC3 */
+ DBGMSG("\t Audio Fmt: AC3\n");
+ break;
+ case 3: /* MPEG1 */
+ DBGMSG("\t Audio Fmt: MPEG1\n");
+ break;
+ case 4: /* MP3 */
+ DBGMSG("\t Audio Fmt: MP3\n");
+ break;
+ case 5: /* MPEG2 */
+ DBGMSG("\t Audio Fmt: MPEG2\n");
+ break;
+ case 6: /* AAC */
+ DBGMSG("\t Audio Fmt: AAC\n");
+ break;
+ case 7: /* DTS */
+ DBGMSG("\t Audio Fmt: DTS\n");
+ break;
+ case 8: /* ATRAC */
+ DBGMSG("\t Audio Fmt: ATRAC\n");
+ break;
+ case 9: /* One bit audio */
+ DBGMSG("\t Audio Fmt: ONE BIT AUDIO\n");
+ break;
+ case 10:/* Dolby */
+ DBGMSG("\t Audio Fmt: DOLBY\n");
+ break;
+ case 11:/* DTS-HD */
+ DBGMSG("\t Audio Fmt: DTS-HD\n");
+ break;
+ case 12:/* MAT (MLP) */
+ DBGMSG("\t Audio Fmt: MAT\n");
+ break;
+ case 13:/* DST */
+ DBGMSG("\t Audio Fmt: DST\n");
+ break;
+ case 14:/* WMA Pro */
+ DBGMSG("\t Audio Fmt: WMA Pro\n");
+ break;
+ }
+
+ DBGMSG("\t Max channel: %d\n", (block[i * 3 + 1] & 0x7) + 1);
+ DBGMSG("\t %s support 32 KHz\n",
+ (block[i * 3 + 2] & 0x1) ? "" : "no");
+ DBGMSG("\t %s support 44 KHz\n",
+ (block[i * 3 + 2] & 0x2) ? "" : "no");
+ DBGMSG("\t %s support 48 KHz\n",
+ (block[i * 3 + 2] & 0x4) ? "" : "no");
+ DBGMSG("\t %s support 88 KHz\n",
+ (block[i * 3 + 2] & 0x8) ? "" : "no");
+ DBGMSG("\t %s support 96 KHz\n",
+ (block[i * 3 + 2] & 0x10) ? "" : "no");
+ DBGMSG("\t %s support 176 KHz\n",
+ (block[i * 3 + 2] & 0x20) ? "" : "no");
+ DBGMSG("\t %s support 192 KHz\n",
+ (block[i * 3 + 2] & 0x40) ? "" : "no");
+ if(audio_format == 1) { /* For LPCM */
+ DBGMSG("\t %s support 16 bit\n",
+ (block[i * 3 + 3] & 0x1) ? "" : "no");
+ DBGMSG("\t %s support 20 bit\n",
+ (block[i * 3 + 3] & 0x2) ? "" : "no");
+ DBGMSG("\t %s support 24 bit\n",
+ (block[i * 3 + 3] & 0x4) ? "" : "no");
+ } else if(audio_format >= 2 && audio_format <= 8) { /* From AC3 to ATRAC */
+ DBGMSG("\t Max bitrate: %d kbit/s\n", block[i * 3 + 3] * 8);
+ }
+ else if(audio_format >= 9 && audio_format <= 14) { /* From One-bit-audio to WMA Pro*/
+ DBGMSG("\t Audio Format Code dependent value: 0x%02X\n", block[i * 3 + 3]);
+ }
+ DBGMSG("\t ========================\n");
+ }
+ break;
+ case 2: /* Video Data Block */
+ DBGMSG("Video Data Block\n");
+ for (i = 0; i < len; i++) {
+ unsigned int vic;
+
+ vic = block[1 + i] & 0x7F;
+ info->cea_vic[vic / 8] |= (0x1 << (vic % 8));
+ DBGMSG("\t %2d : VIC %2d %dx%d@%d%s %s\n", i,
+ vic, hdmi_vic_info[vic].resx,
+ hdmi_vic_info[vic].resy,
+ hdmi_vic_info[vic].freq,
+ (hdmi_vic_info[vic].option
+ & HDMI_VIC_INTERLACE) ? "I" : "P",
+ (hdmi_vic_info[vic].option
+ & HDMI_VIC_4x3) ? "4:3" : "16:9");
+ }
+ break;
+ case 3: /* Vendor Spec Data Block */
+ edid_parse_CEA_VendorSpecDataBlock(block, len, info);
+ break;
+ case 4: /* Speaker Allocation Data Block */
+ DBGMSG("Speaker Allocation Data Block\n");
+ break;
+ case 5: /* VESA DTC Data Block */
+ DBGMSG("VESA DTC Data Block\n");
+ break;
+ case 7: /* Use Extended Tag */
+ DBGMSG("Use Extended Tag\n");
+ break;
+ case 0: /* Reserved */
+ default:
+ len = 0;
+ break;
+ }
+ block += (1 + len);
+ } while (block < block_end);
+
+ {
+ struct fb_videomode *p;
+ unsigned int fps;
+
+ DBGMSG("Detail Timing\n");
+ block = edid + edid[2];
+ len = (128 - edid[2]) / 18;
+ for (i = 0; i < len; i++, block += 18) {
+ p = &info->cea_timing[i];
+ p->pixclock = ((block[1] << 8) + block[0]) * 10000;
+ if (p->pixclock == 0)
+ break;
+ p->xres = ((block[4] & 0xF0) << 4) + block[2];
+ p->left_margin = ((block[4] & 0x0F) << 8) + block[3];
+ p->yres = ((block[7] & 0xF0) << 4) + block[5];
+ p->upper_margin = ((block[7] & 0x0F) << 8) + block[6];
+ fps = vpp_calc_refresh(p->pixclock, p->xres + p->left_margin,
+ p->yres + p->upper_margin);
+ p->right_margin = ((block[11] & 0xC0) << 2) + block[8];
+ p->hsync_len = ((block[11] & 0x30) << 4) + block[9];
+ p->left_margin = p->left_margin -
+ p->right_margin - p->hsync_len;
+ p->lower_margin = ((block[11] & 0x0C) << 2) +
+ ((block[10] & 0xF0) >> 4);
+ p->vsync_len = ((block[11] & 0x03) << 4) + (block[10] & 0x0F);
+ p->upper_margin = p->upper_margin -
+ p->lower_margin - p->vsync_len;
+ p->refresh = fps;
+ p->vmode = (block[17] & 0x80) ?
+ FB_VMODE_INTERLACED : FB_VMODE_NONINTERLACED;
+ p->sync = (block[17] & 0x04) ? FB_SYNC_VERT_HIGH_ACT : 0;
+ p->sync |= (block[17] & 0x02) ? FB_SYNC_HOR_HIGH_ACT : 0;
+ if (p->vmode & FB_VMODE_INTERLACED) {
+ p->yres *= 2;
+ p->upper_margin *= 2;
+ p->lower_margin *= 2;
+ p->vsync_len *= 2;
+ }
+ DBGMSG("\t%dx%d%s@%d,clk %d\n", p->xres, p->yres,
+ (block[17] & 0x80) ? "I" : "P", fps, p->pixclock);
+ DBGMSG("\t\tH bp %d,sync %d,fp %d\n",
+ p->left_margin, p->hsync_len, p->right_margin);
+ DBGMSG("\t\tV bp %d,sync %d,fp %d\n",
+ p->upper_margin, p->vsync_len, p->lower_margin);
+ DBGMSG("\t\thsync %d,vsync %d\n",
+ (p->sync & FB_SYNC_HOR_HIGH_ACT) ? 1 : 0,
+ (p->sync & FB_SYNC_VERT_HIGH_ACT) ? 1 : 0);
+ p->pixclock = KHZ2PICOS(p->pixclock / 1000);
+ }
+ }
+ return 0;
+}
+
+void edid_dump(char *edid)
+{
+ int i;
+
+ DPRINT("===================== EDID BlOCK =====================");
+ for (i = 0; i < 128; i++) {
+ if ((i % 16) == 0)
+ DPRINT("\n");
+ DPRINT("%02x ", edid[i]);
+ }
+ DPRINT("\n");
+ DPRINT("======================================================\n");
+}
+
+int edid_checksum(char *edid, int len)
+{
+ char checksum = 0;
+ int i;
+
+ for (i = 0; i < len; i++)
+ checksum += edid[i];
+
+ if (checksum) {
+#ifdef DEBUG
+ edid_dump(edid);
+#endif
+ }
+ return checksum;
+}
+
+int edid_parse(char *edid, edid_info_t *info)
+{
+ int ext_cnt = 0;
+
+ if (edid == 0)
+ return 0;
+
+ if (info->option & EDID_OPT_VALID) {
+ DBG_MSG("[EDID] parse exist\n");
+ return info->option;
+ }
+
+ DBG_MSG("[EDID] Enter\n");
+
+ memset(info, 0, sizeof(edid_info_t));
+ info->option = EDID_OPT_VALID;
+ if (edid_parse_v1(edid, info) == 0) {
+ ext_cnt = edid[0x7E];
+ if (ext_cnt >= EDID_BLOCK_MAX) {
+ DPRINT("[EDID] *W* ext block cnt %d\n", ext_cnt);
+ ext_cnt = EDID_BLOCK_MAX - 1;
+ }
+ } else {
+ info->option = 0;
+ return 0;
+ }
+
+ while (ext_cnt) {
+ edid += 128;
+ ext_cnt--;
+ if (edid_parse_CEA(edid, info) == 0)
+ continue;
+
+ DPRINT("*W* not support EDID\n");
+ edid_dump(edid);
+ }
+ return info->option;
+}
+
+int edid_find_support(edid_info_t *info, unsigned int resx,
+ unsigned int resy, int freq, struct fb_videomode **vmode)
+{
+ struct fb_videomode *p;
+ int temp;
+ int ret;
+ int i;
+
+ ret = 0;
+ *vmode = 0;
+
+ if (!info)
+ return 0;
+
+ /* find cea timing */
+ for (i = 0; i < 6; i++) {
+ p = &info->cea_timing[i];
+
+ if (p->pixclock == 0)
+ continue;
+ if (resx != p->xres)
+ continue;
+ if (resy != p->yres)
+ continue;
+
+ temp = p->refresh;
+ temp |= (p->vmode & FB_VMODE_INTERLACED) ?
+ EDID_TMR_INTERLACE : 0;
+ if (freq != temp)
+ continue;
+
+ *vmode = p;
+ ret = 4;
+ goto find_end;
+ }
+
+ /* find vic timing */
+ for (i = 0; i < 64; i++) {
+ if ((info->cea_vic[i / 8] & (0x1 << (i % 8))) == 0)
+ continue;
+
+ if (i >= HDMI_VIDEO_CODE_MAX)
+ continue;
+
+ if (resx != hdmi_vic_info[i].resx)
+ continue;
+ if (resy != hdmi_vic_info[i].resy)
+ continue;
+ temp = hdmi_vic_info[i].freq;
+ temp |= (hdmi_vic_info[i].option & HDMI_VIC_INTERLACE) ?
+ EDID_TMR_INTERLACE : 0;
+ if (freq != temp)
+ continue;
+ ret = 5;
+ goto find_end;
+ }
+
+ /* find detail timing */
+ for (i = 0; i < 4; i++) {
+ p = &info->detail_timing[i];
+ if (p->pixclock == 0)
+ continue;
+ if (resx != p->xres)
+ continue;
+ if (resy != p->yres)
+ continue;
+
+ temp = p->refresh;
+ temp |= (p->vmode & FB_VMODE_INTERLACED) ?
+ EDID_TMR_INTERLACE : 0;
+ if (freq != temp)
+ continue;
+ *vmode = p;
+ ret = 3;
+ goto find_end;
+ }
+
+ /* find established timing */
+ if (info->establish_timing) {
+ for (i = 0; i < 17; i++) {
+ if (info->establish_timing & (0x1 << i)) {
+ if ((resx == edid_establish_timing[i].resx) &&
+ (resy == edid_establish_timing[i].resy)) {
+ if (freq ==
+ edid_establish_timing[i].freq) {
+ ret = 1;
+ goto find_end;
+ }
+ }
+ }
+ }
+ }
+
+ /* find standard timing */
+ for (i = 0; i < 8; i++) {
+ if (info->standard_timing[i].resx == 0)
+ continue;
+ if ((resx == info->standard_timing[i].resx) &&
+ (resy == info->standard_timing[i].resy)) {
+ if (freq == info->standard_timing[i].freq) {
+ ret = 2;
+ goto find_end;
+ }
+ }
+ }
+
+find_end:
+#if 0
+ if ((resx == 1920) && (resy == 1080) && !(freq & EDID_TMR_INTERLACE))
+ ret = 0;
+#endif
+#if 0
+ if (!(freq & EDID_TMR_INTERLACE))
+ ret = 0;
+#endif
+#if 0
+ DPRINT("[EDID] %s support %dx%d@%d%s(ret %d)\n",
+ (ret) ? "" : "No", resx, resy, freq & EDID_TMR_FREQ,
+ (freq & EDID_TMR_INTERLACE) ? "I" : "P", ret);
+#endif
+ return ret;
+}
+
+unsigned int edid_get_hdmi_phy_addr(void)
+{
+ vout_t *vo;
+
+ vo = vout_get_entry(VPP_VOUT_NUM_HDMI);
+ return vo->edid_info.hdmi_phy_addr;
+}
+
diff --git a/drivers/video/wmt/scl.c b/drivers/video/wmt/scl.c
new file mode 100644
index 00000000..f8433a04
--- /dev/null
+++ b/drivers/video/wmt/scl.c
@@ -0,0 +1,1604 @@
+/*++
+ * linux/drivers/video/wmt/scl.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define SCL_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+#include "scl.h"
+
+#ifdef WMT_FTBLK_SCL
+void scl_reg_dump(void)
+{
+ vpp_set_clock_enable(DEV_SCL444U, 1, 0);
+
+ DPRINT("========== SCL register dump ==========\n");
+ vpp_reg_dump(REG_SCL_BASE1_BEGIN,
+ REG_SCL_BASE1_END - REG_SCL_BASE1_BEGIN);
+ vpp_reg_dump(REG_SCL_BASE2_BEGIN,
+ REG_SCL_BASE2_END - REG_SCL_BASE2_BEGIN);
+ DPRINT("---------- SCL scale ----------\n");
+ DPRINT("scale enable %d\n", vppif_reg32_read(SCL_ALU_ENABLE));
+ DPRINT("mode bilinear(H %d,V %d),recursive(H %d,V %d)\n",
+ vppif_reg32_read(SCL_BILINEAR_H),
+ vppif_reg32_read(SCL_BILINEAR_V),
+ vppif_reg32_read(SCL_RECURSIVE_H),
+ vppif_reg32_read(SCL_RECURSIVE_V));
+ DPRINT("src(%d,%d),dst(%d,%d)\n",
+ vppif_reg32_read(SCLR_YPXLWID),
+ vppif_reg32_read(SCL_VXWIDTH),
+ vppif_reg32_read(SCL_H_THR),
+ vppif_reg32_read(SCL_V_THR));
+ DPRINT("scale width H %d,V %d\n",
+ vppif_reg32_read(SCL_HXWIDTH),
+ vppif_reg32_read(SCL_VXWIDTH));
+ DPRINT("H scale up %d,V scale up %d\n",
+ vppif_reg32_read(SCL_HSCLUP_ENABLE),
+ vppif_reg32_read(SCL_VSCLUP_ENABLE));
+ DPRINT("H sub step %d,thr %d,step %d,sub step cnt %d,i step cnt %d\n",
+ vppif_reg32_read(SCL_H_SUBSTEP),
+ vppif_reg32_read(SCL_H_THR),
+ vppif_reg32_read(SCL_H_STEP),
+ vppif_reg32_read(SCL_H_I_SUBSTEPCNT),
+ vppif_reg32_read(SCL_H_I_STEPCNT));
+ DPRINT("V sub step %d,thr %d,step %d,sub step cnt %d,i step cnt %d\n",
+ vppif_reg32_read(SCL_V_SUBSTEP),
+ vppif_reg32_read(SCL_V_THR),
+ vppif_reg32_read(SCL_V_STEP),
+ vppif_reg32_read(SCL_V_I_SUBSTEPCNT),
+ vppif_reg32_read(SCL_V_I_STEPCNT));
+
+ DPRINT("---------- SCL filter ----------\n");
+ DPRINT("DEBLOCK %d,boundary 1st 0x%x,2nd 0x%x\n,",
+ vppif_reg32_read(SCL_DEBLOCK_ENABLE),
+ vppif_reg32_read(SCL_1ST_LAYER_BOUNDARY),
+ vppif_reg32_read(SCL_2ND_LAYER_BOUNDARY));
+ DPRINT("FIELD DEFLICKER %d,up %s down,thr Y %d,C %d\n",
+ vppif_reg32_read(SCL_FIELD_DEFLICKER),
+ vppif_reg32_read(SCL_FIELD_DEFLICKER) ? "&" : "or",
+ vppif_reg32_read(SCL_FIELD_FILTER_Y_THD),
+ vppif_reg32_read(SCL_FIELD_FILTER_C_THD));
+ DPRINT("FRAME DEFLICKER %d,%s,2^%d,scene chg %d\n",
+ vppif_reg32_read(SCL_FRAME_DEFLICKER),
+ vppif_reg32_read(SCL_FRAME_FILTER_RGB) ? "RGB" : "Y",
+ vppif_reg32_read(SCL_FRAME_FILTER_SAMPLER),
+ vppif_reg32_read(SCL_FR_FILTER_SCENE_CHG_THD));
+ DPRINT("CSC enable %d,CSC clamp %d\n",
+ vppif_reg32_read(SCL_CSC_ENABLE),
+ vppif_reg32_read(SCL_CSC_CLAMP_ENABLE));
+ DPRINT("---------- SCL TG ----------\n");
+ DPRINT("TG source : %s\n",
+ (vppif_reg32_read(SCL_TG_GOVWTG_ENABLE)) ? "GOVW" : "SCL");
+ DPRINT("TG enable %d, wait ready enable %d\n",
+ vppif_reg32_read(SCL_TG_ENABLE),
+ vppif_reg32_read(SCL_TG_WATCHDOG_ENABLE));
+ DPRINT("clk %d,Read cyc %d,1T %d\n",
+ vpp_get_base_clock(VPP_MOD_SCL),
+ vppif_reg32_read(SCL_TG_RDCYC),
+ vppif_reg32_read(SCL_READCYC_1T));
+ DPRINT("H total %d, beg %d, end %d\n",
+ vppif_reg32_read(SCL_TG_H_ALLPIXEL),
+ vppif_reg32_read(SCL_TG_H_ACTBG),
+ vppif_reg32_read(SCL_TG_H_ACTEND));
+ DPRINT("V total %d, beg %d, end %d\n",
+ vppif_reg32_read(SCL_TG_V_ALLLINE),
+ vppif_reg32_read(SCL_TG_V_ACTBG),
+ vppif_reg32_read(SCL_TG_V_ACTEND));
+ DPRINT("VBIE %d,PVBI %d\n",
+ vppif_reg32_read(SCL_TG_VBIE),
+ vppif_reg32_read(SCL_TG_PVBI));
+ DPRINT("Watch dog 0x%x\n",
+ vppif_reg32_read(SCL_TG_WATCHDOG_VALUE));
+ DPRINT("---------- SCLR FB ----------\n");
+ DPRINT("SCLR MIF enable %d\n",
+ vppif_reg32_read(SCLR_MIF_ENABLE));
+ DPRINT("color format %s\n", vpp_colfmt_str[sclr_get_color_format()]);
+ DPRINT("color bar enable %d,mode %d,inv %d\n",
+ vppif_reg32_read(SCLR_COLBAR_ENABLE),
+ vppif_reg32_read(SCLR_COLBAR_MODE),
+ vppif_reg32_read(SCLR_COLBAR_INVERSION));
+ DPRINT("sourc mode : %s,H264 %d\n",
+ (vppif_reg32_read(SCLR_TAR_DISP_FMT)) ? "field" : "frame",
+ vppif_reg32_read(SCLR_MEDIAFMT_H264));
+ DPRINT("Y addr 0x%x, C addr 0x%x\n",
+ vppif_reg32_in(REG_SCLR_YSA),
+ vppif_reg32_in(REG_SCLR_CSA));
+ DPRINT("width %d, fb width %d\n",
+ vppif_reg32_read(SCLR_YPXLWID),
+ vppif_reg32_read(SCLR_YBUFWID));
+ DPRINT("H crop %d, V crop %d\n",
+ vppif_reg32_read(SCLR_HCROP),
+ vppif_reg32_read(SCLR_VCROP));
+ DPRINT("---------- SCLW FB ----------\n");
+ DPRINT("SCLW MIF enable %d\n", vppif_reg32_read(SCLW_MIF_ENABLE));
+ DPRINT("color format %s\n", vpp_colfmt_str[sclw_get_color_format()]);
+ DPRINT("Y addr 0x%x, C addr 0x%x\n",
+ vppif_reg32_in(REG_SCLW_YSA), vppif_reg32_in(REG_SCLW_CSA));
+ DPRINT("Y width %d, fb width %d\n",
+ vppif_reg32_read(SCLW_YPXLWID), vppif_reg32_read(SCLW_YBUFWID));
+ DPRINT("C width %d, fb width %d\n",
+ vppif_reg32_read(SCLW_CPXLWID), vppif_reg32_read(SCLW_CBUFWID));
+ DPRINT("Y err %d, C err %d\n",
+ vppif_reg32_read(SCLW_INTSTS_MIFYERR),
+ vppif_reg32_read(SCLW_INTSTS_MIFCERR));
+ DPRINT("---------- SCLR2 FB ----------\n");
+ DPRINT("MIF enable %d\n", vppif_reg32_read(SCL_R2_MIF_EN));
+ DPRINT("color format %s\n", vpp_colfmt_str[scl_R2_get_color_format()]);
+ DPRINT("color bar enable %d,mode %d,inv %d\n",
+ vppif_reg32_read(SCL_R2_COLOR_EN),
+ vppif_reg32_read(SCL_R2_COLOR_EN),
+ vppif_reg32_read(SCL_R2_COLOR_INV));
+ DPRINT("sourc mode : %s,H264 %d\n",
+ (vppif_reg32_read(SCL_R2_IOFMT)) ? "field" : "frame",
+ vppif_reg32_read(SCL_R2_H264_FMT));
+ DPRINT("Y addr 0x%x, C addr 0x%x\n",
+ vppif_reg32_in(REG_SCLR2_YSA), vppif_reg32_in(REG_SCLR2_CSA));
+ DPRINT("width %d, fb width %d\n",
+ vppif_reg32_read(SCL_R2_LNSIZE), vppif_reg32_read(SCL_R2_FBW));
+ DPRINT("H crop %d, V crop %d\n",
+ vppif_reg32_read(SCL_R2_HCROP), vppif_reg32_read(SCL_R2_VCROP));
+ DPRINT("---------- ALPHA ----------\n");
+ DPRINT("src alpha %d,dst alpha %d,swap %d\n",
+ vppif_reg32_read(SCL_ALPHA_SRC),
+ vppif_reg32_read(SCL_ALPHA_DST),
+ vppif_reg32_read(SCL_ALPHA_SWAP));
+ DPRINT("src fix 0x%x,dst fix 0x%x\n",
+ vppif_reg32_read(SCL_ALPHA_SRC_FIXED),
+ vppif_reg32_read(SCL_ALPHA_DST_FIXED));
+ DPRINT("---------- ColorKey ----------\n");
+ DPRINT("enable %d\n", vppif_reg32_read(SCL_ALPHA_COLORKEY_ENABLE));
+ DPRINT("from %s,comp %d,mode %d\n",
+ (vppif_reg32_read(SCL_ALPHA_COLORKEY_FROM)) ? "mif2" : "mif1",
+ vppif_reg32_read(SCL_ALPHA_COLORKEY_COMP),
+ vppif_reg32_read(SCL_ALPHA_COLORKEY_MODE));
+ DPRINT("R 0x%x,G 0x%x,B 0x%x\n",
+ vppif_reg32_read(SCL_ALPHA_COLORKEY_R),
+ vppif_reg32_read(SCL_ALPHA_COLORKEY_G),
+ vppif_reg32_read(SCL_ALPHA_COLORKEY_B));
+ DPRINT("---------- sw status ----------\n");
+ DPRINT("complete %d\n", p_scl->scale_complete);
+
+ vpp_set_clock_enable(DEV_SCL444U, 0, 0);
+}
+
+void scl_set_enable(vpp_flag_t enable)
+{
+ vppif_reg32_write(SCL_ALU_ENABLE, enable);
+}
+
+void scl_set_reg_update(vpp_flag_t enable)
+{
+ vppif_reg32_write(SCL_REG_UPDATE, enable);
+}
+
+void scl_set_reg_level(vpp_reglevel_t level)
+{
+ switch (level) {
+ case VPP_REG_LEVEL_1:
+ vppif_reg32_write(SCL_REG_LEVEL, 0x0);
+ break;
+ case VPP_REG_LEVEL_2:
+ vppif_reg32_write(SCL_REG_LEVEL, 0x1);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+void scl_set_int_enable(vpp_flag_t enable, vpp_int_t int_bit)
+{
+ /* clean status first before enable/disable interrupt */
+ scl_clean_int_status(int_bit);
+
+ if (int_bit & VPP_INT_ERR_SCL_TG)
+ vppif_reg32_write(SCLW_INT_TGERR_ENABLE, enable);
+ if (int_bit & VPP_INT_ERR_SCLR1_MIF)
+ vppif_reg32_write(SCLW_INT_R1MIF_ENABLE, enable);
+ if (int_bit & VPP_INT_ERR_SCLR2_MIF)
+ vppif_reg32_write(SCLW_INT_R2MIF_ENABLE, enable);
+ if (int_bit & VPP_INT_ERR_SCLW_MIFRGB)
+ vppif_reg32_write(SCLW_INT_WMIFRGB_ENABLE, enable);
+ if (int_bit & VPP_INT_ERR_SCLW_MIFY)
+ vppif_reg32_write(SCLW_INT_WMIFYERR_ENABLE, enable);
+ if (int_bit & VPP_INT_ERR_SCLW_MIFC)
+ vppif_reg32_write(SCLW_INT_WMIFCERR_ENABLE, enable);
+}
+
+vpp_int_err_t scl_get_int_status(void)
+{
+ vpp_int_err_t int_sts;
+
+ int_sts = 0;
+ if (vppif_reg32_read(SCL_INTSTS_TGERR))
+ int_sts |= VPP_INT_ERR_SCL_TG;
+ if (vppif_reg32_read(SCLR_INTSTS_R1MIFERR))
+ int_sts |= VPP_INT_ERR_SCLR1_MIF;
+ if (vppif_reg32_read(SCLR_INTSTS_R2MIFERR))
+ int_sts |= VPP_INT_ERR_SCLR2_MIF;
+ if (vppif_reg32_read(SCLW_INTSTS_MIFRGBERR))
+ int_sts |= VPP_INT_ERR_SCLW_MIFRGB;
+ if (vppif_reg32_read(SCLW_INTSTS_MIFYERR))
+ int_sts |= VPP_INT_ERR_SCLW_MIFY;
+ if (vppif_reg32_read(SCLW_INTSTS_MIFCERR))
+ int_sts |= VPP_INT_ERR_SCLW_MIFC;
+
+ return int_sts;
+}
+
+void scl_clean_int_status(vpp_int_err_t int_sts)
+{
+ if (int_sts & VPP_INT_ERR_SCL_TG)
+ vppif_reg32_out(REG_SCL_TG_STS + 0x0, BIT0);
+ if (int_sts & VPP_INT_ERR_SCLR1_MIF)
+ vppif_reg8_out(REG_SCLR_FIFO_CTL + 0x1, BIT0);
+ if (int_sts & VPP_INT_ERR_SCLR2_MIF)
+ vppif_reg8_out(REG_SCLR_FIFO_CTL + 0x1, BIT1);
+ if (int_sts & VPP_INT_ERR_SCLW_MIFRGB)
+ vppif_reg32_out(REG_SCLW_FF_CTL, BIT16);
+ if (int_sts & VPP_INT_ERR_SCLW_MIFY)
+ vppif_reg32_out(REG_SCLW_FF_CTL, BIT8);
+ if (int_sts & VPP_INT_ERR_SCLW_MIFC)
+ vppif_reg32_out(REG_SCLW_FF_CTL, BIT0);
+}
+
+void scl_set_csc_mode(vpp_csc_t mode)
+{
+ vdo_color_fmt src_fmt, dst_fmt;
+
+ src_fmt = sclr_get_color_format();
+ dst_fmt = sclw_get_color_format();
+ mode = vpp_check_csc_mode(mode, src_fmt, dst_fmt, 0);
+ if (mode >= VPP_CSC_MAX)
+ vppif_reg32_write(SCL_CSC_ENABLE, VPP_FLAG_DISABLE);
+ else {
+ vppif_reg32_out(REG_SCL_CSC1, vpp_csc_parm[mode][0]);
+ vppif_reg32_out(REG_SCL_CSC2, vpp_csc_parm[mode][1]);
+ vppif_reg32_out(REG_SCL_CSC3, vpp_csc_parm[mode][2]);
+ vppif_reg32_out(REG_SCL_CSC4, vpp_csc_parm[mode][3]);
+ vppif_reg32_out(REG_SCL_CSC5, vpp_csc_parm[mode][4]);
+ vppif_reg32_out(REG_SCL_CSC6, vpp_csc_parm[mode][5]);
+ vppif_reg32_out(REG_SCL_CSC_CTL, vpp_csc_parm[mode][6]);
+ vppif_reg32_write(SCL_CSC_ENABLE, VPP_FLAG_ENABLE);
+ }
+}
+
+void scl_set_scale_enable(vpp_flag_t vscl_enable, vpp_flag_t hscl_enable)
+{
+ DBGMSG("V %d,H %d\n", vscl_enable, hscl_enable);
+ vppif_reg32_write(SCL_VSCLUP_ENABLE, vscl_enable);
+ vppif_reg32_write(SCL_HSCLUP_ENABLE, hscl_enable);
+}
+
+void scl_set_V_scale(int A, int B) /* A dst,B src */
+{
+ unsigned int V_STEP;
+ unsigned int V_SUB_STEP;
+ unsigned int V_THR_DIV2;
+
+ DBG_DETAIL("scl_set_V_scale(%d,%d)\r\n", A, B);
+ if (A > B) {
+ V_STEP = (B - 1) * 16 / A;
+ V_SUB_STEP = (B - 1) * 16 % A;
+ } else {
+ V_STEP = (16 * B / A);
+ V_SUB_STEP = ((16 * B) % A);
+ }
+ V_THR_DIV2 = A;
+
+ DBG_DETAIL("V step %d,sub step %d, div2 %d\r\n",
+ V_STEP, V_SUB_STEP, V_THR_DIV2);
+
+#ifdef SCL_DST_VXWIDTH
+ vppif_reg32_write(SCL_DST_VXWIDTH, (A > B) ? A : B);
+#endif
+ vppif_reg32_write(SCL_VXWIDTH, B);
+ vppif_reg32_write(SCL_V_STEP, V_STEP);
+ vppif_reg32_write(SCL_V_SUBSTEP, V_SUB_STEP);
+ vppif_reg32_write(SCL_V_THR, V_THR_DIV2);
+ vppif_reg32_write(SCL_V_I_SUBSTEPCNT, 0);
+}
+
+void scl_set_H_scale(int A, int B) /* A dst,B src */
+{
+ unsigned int H_STEP;
+ unsigned int H_SUB_STEP;
+ unsigned int H_THR_DIV2;
+
+ DBG_DETAIL("scl_set_H_scale(%d,%d)\r\n", A, B);
+ if (A > B) {
+ H_STEP = (B - 1) * 16 / A;
+ H_SUB_STEP = (B - 1) * 16 % A;
+ } else {
+ H_STEP = (16 * B / A);
+ H_SUB_STEP = ((16 * B) % A);
+ }
+ H_THR_DIV2 = A;
+ DBG_DETAIL("H step %d,sub step %d, div2 %d\r\n",
+ H_STEP, H_SUB_STEP, H_THR_DIV2);
+
+ vppif_reg32_write(SCL_HXWIDTH, ((A > B) ? A : B));
+ vppif_reg32_write(SCL_H_STEP, H_STEP);
+ vppif_reg32_write(SCL_H_SUBSTEP, H_SUB_STEP);
+ vppif_reg32_write(SCL_H_THR, H_THR_DIV2);
+ vppif_reg32_write(SCL_H_I_SUBSTEPCNT, 0);
+}
+
+void scl_set_crop(int offset_x, int offset_y)
+{
+ /* offset_x &= VPU_CROP_ALIGN_MASK; */ /* ~0x7 */
+ offset_x &= ~0xf;
+
+ vppif_reg32_write(SCL_H_I_STEPCNT, offset_x * 16);
+ vppif_reg32_write(SCL_V_I_STEPCNT, offset_y * 16);
+ /* vppif_reg32_write(VPU_SCA_THR, 0xFF); */
+ DBGMSG("[VPU] crop - x : 0x%x, y : 0x%x \r\n",
+ offset_x * 16, offset_y * 16);
+}
+
+void scl_set_tg_enable(vpp_flag_t enable)
+{
+ vppif_reg32_write(SCL_TG_ENABLE, enable);
+}
+
+unsigned int scl_set_clock(unsigned int pixel_clock)
+{
+ unsigned int rd_cyc;
+ rd_cyc = vpp_get_base_clock(VPP_MOD_SCL) / pixel_clock;
+ return rd_cyc;
+}
+
+void scl_set_timing(vpp_clock_t *timing, unsigned int pixel_clock)
+{
+#if 1
+ timing->read_cycle = WMT_SCL_RCYC_MIN;
+#else
+ timing->read_cycle = scl_set_clock(pixel_clock * 2) - 1;
+ timing->read_cycle = (timing->read_cycle < WMT_SCL_RCYC_MIN) ?
+ WMT_SCL_RCYC_MIN : timing->read_cycle;
+ timing->read_cycle = (timing->read_cycle > 255) ?
+ 0xFF : timing->read_cycle;
+#endif
+ vppif_reg32_write(SCL_TG_RDCYC, timing->read_cycle);
+ vppif_reg32_write(SCL_READCYC_1T, (timing->read_cycle) ? 0 : 1);
+ vppif_reg32_write(SCL_TG_H_ALLPIXEL, timing->total_pixel_of_line);
+ vppif_reg32_write(SCL_TG_H_ACTBG, timing->begin_pixel_of_active);
+ vppif_reg32_write(SCL_TG_H_ACTEND, timing->end_pixel_of_active);
+ vppif_reg32_write(SCL_TG_V_ALLLINE, timing->total_line_of_frame);
+ vppif_reg32_write(SCL_TG_V_ACTBG, timing->begin_line_of_active);
+ vppif_reg32_write(SCL_TG_V_ACTEND, timing->end_line_of_active);
+ vppif_reg32_write(SCL_TG_VBIE, timing->line_number_between_VBIS_VBIE);
+ vppif_reg32_write(SCL_TG_PVBI, timing->line_number_between_PVBI_VBIS);
+
+#ifdef DEBUG_DETAIL
+ vpp_show_timing("scl set timing", 0, timing);
+#endif
+}
+
+void scl_get_timing(vpp_clock_t *p_timing)
+{
+ p_timing->read_cycle = vppif_reg32_read(SCL_TG_RDCYC);
+ p_timing->total_pixel_of_line = vppif_reg32_read(SCL_TG_H_ALLPIXEL);
+ p_timing->begin_pixel_of_active = vppif_reg32_read(SCL_TG_H_ACTBG);
+ p_timing->end_pixel_of_active = vppif_reg32_read(SCL_TG_H_ACTEND);
+ p_timing->total_line_of_frame = vppif_reg32_read(SCL_TG_V_ALLLINE);
+ p_timing->begin_line_of_active = vppif_reg32_read(SCL_TG_V_ACTBG);
+ p_timing->end_line_of_active = vppif_reg32_read(SCL_TG_V_ACTEND);
+ p_timing->line_number_between_VBIS_VBIE = vppif_reg32_read(SCL_TG_VBIE);
+ p_timing->line_number_between_PVBI_VBIS = vppif_reg32_read(SCL_TG_PVBI);
+}
+
+void scl_set_watchdog(U32 count)
+{
+ if (0 != count) {
+ vppif_reg32_write(SCL_TG_WATCHDOG_VALUE, count);
+ vppif_reg32_write(SCL_TG_WATCHDOG_ENABLE, VPP_FLAG_TRUE);
+ } else
+ vppif_reg32_write(SCL_TG_WATCHDOG_ENABLE, VPP_FLAG_FALSE);
+}
+
+void scl_set_timing_master(vpp_mod_t mod_bit)
+{
+ if (VPP_MOD_SCL == mod_bit)
+ vppif_reg32_write(SCL_TG_GOVWTG_ENABLE, VPP_FLAG_DISABLE);
+ else if (VPP_MOD_GOVW == mod_bit)
+ vppif_reg32_write(SCL_TG_GOVWTG_ENABLE, VPP_FLAG_ENABLE);
+ else {
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+vpp_mod_t scl_get_timing_master(void)
+{
+ if (vppif_reg32_read(SCL_TG_GOVWTG_ENABLE))
+ return VPP_MOD_GOVW;
+ return VPP_MOD_SCL;
+}
+
+void scl_set_drop_line(vpp_flag_t enable)
+{
+ vppif_reg32_write(SCL_SCLDW_METHOD, enable);
+}
+
+/* only one feature can work, other should be disable */
+void scl_set_filter_mode(vpp_filter_mode_t mode, int enable)
+{
+ DBG_DETAIL("(%d,%d)\n", mode, enable);
+ if (mode != VPP_FILTER_SCALE) {
+ if (vppif_reg32_read(SCL_VSCLUP_ENABLE) ||
+ vppif_reg32_read(SCL_HSCLUP_ENABLE))
+ DPRINT("[SCL] *W* filter can't work w scale\n");
+ }
+ vppif_reg32_write(SCL_DEBLOCK_ENABLE, 0);
+ vppif_reg32_write(SCL_FIELD_DEFLICKER, 0);
+ vppif_reg32_write(SCL_FRAME_DEFLICKER, 0);
+ switch (mode) {
+ default:
+ case VPP_FILTER_SCALE: /* scale mode */
+ break;
+ case VPP_FILTER_DEBLOCK: /* deblock */
+ vppif_reg32_write(SCL_DEBLOCK_ENABLE, enable);
+ break;
+ case VPP_FILTER_FIELD_DEFLICKER: /* field deflicker */
+ vppif_reg32_write(SCL_FIELD_DEFLICKER, enable);
+ break;
+ case VPP_FILTER_FRAME_DEFLICKER: /* frame deflicker */
+ vppif_reg32_write(SCL_FRAME_DEFLICKER, enable);
+ break;
+ }
+}
+
+vpp_filter_mode_t scl_get_filter_mode(void)
+{
+ if (vppif_reg32_read(SCL_VSCLUP_ENABLE) ||
+ vppif_reg32_read(SCL_HSCLUP_ENABLE))
+ return VPP_FILTER_SCALE;
+ if (vppif_reg32_read(SCL_DEBLOCK_ENABLE))
+ return VPP_FILTER_DEBLOCK;
+
+ if (vppif_reg32_read(SCL_FIELD_DEFLICKER))
+ return VPP_FILTER_FIELD_DEFLICKER;
+
+ if (vppif_reg32_read(SCL_FRAME_DEFLICKER))
+ return VPP_FILTER_FRAME_DEFLICKER;
+
+ return VPP_FILTER_SCALE;
+}
+
+void sclr_set_mif_enable(vpp_flag_t enable)
+{
+ vppif_reg32_write(SCLR_MIF_ENABLE, enable);
+}
+
+void sclr_set_mif2_enable(vpp_flag_t enable)
+{
+
+}
+
+void sclr_set_colorbar(vpp_flag_t enable, int width, int inverse)
+{
+ vppif_reg32_write(SCLR_COLBAR_MODE, width);
+ vppif_reg32_write(SCLR_COLBAR_INVERSION, inverse);
+ vppif_reg32_write(SCLR_COLBAR_ENABLE, enable);
+}
+
+void sclr_set_field_mode(vpp_display_format_t fmt)
+{
+ vppif_reg32_write(SCLR_SRC_DISP_FMT, fmt);
+}
+
+void sclr_set_display_format(vpp_display_format_t source,
+ vpp_display_format_t target)
+{
+ /* source */
+ switch (source) {
+ case VPP_DISP_FMT_FRAME:
+ vppif_reg32_write(SCLR_SRC_DISP_FMT, 0x0);
+ break;
+ case VPP_DISP_FMT_FIELD:
+ vppif_reg32_write(SCLR_SRC_DISP_FMT, 0x1);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+
+ /* target */
+ switch (target) {
+ case VPP_DISP_FMT_FRAME:
+ vppif_reg32_write(SCLR_TAR_DISP_FMT, 0x0);
+ break;
+ case VPP_DISP_FMT_FIELD:
+ vppif_reg32_write(SCLR_TAR_DISP_FMT, 0x1);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+void sclr_set_color_format(vdo_color_fmt format)
+{
+ if (format >= VDO_COL_FMT_ARGB) {
+ if (format == VDO_COL_FMT_RGB_565)
+ vppif_reg32_write(SCLR_RGB_MODE, 0x1);
+ else
+ vppif_reg32_write(SCLR_RGB_MODE, 0x3);
+ return;
+ }
+ vppif_reg32_write(SCLR_RGB_MODE, 0x0);
+ switch (format) {
+ case VDO_COL_FMT_ARGB:
+ vppif_reg32_write(SCLR_COLFMT_RGB, 0x1);
+ break;
+ case VDO_COL_FMT_YUV444:
+ vppif_reg32_write(SCLR_COLFMT_RGB, 0x0);
+ vppif_reg32_write(SCLR_COLFMT_YUV, 0x2);
+ break;
+ case VDO_COL_FMT_YUV422H:
+ vppif_reg32_write(SCLR_COLFMT_RGB, 0x0);
+ vppif_reg32_write(SCLR_COLFMT_YUV, 0x0);
+ break;
+ case VDO_COL_FMT_YUV420:
+ vppif_reg32_write(SCLR_COLFMT_RGB, 0x0);
+ vppif_reg32_write(SCLR_COLFMT_YUV, 0x1);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+vdo_color_fmt sclr_get_color_format(void)
+{
+ switch (vppif_reg32_read(SCLR_RGB_MODE)) {
+ case 0x1:
+ return VDO_COL_FMT_RGB_565;
+ case 0x3:
+ return VDO_COL_FMT_ARGB;
+ default:
+ break;
+ }
+ switch (vppif_reg32_read(SCLR_COLFMT_YUV)) {
+ case 0:
+ return VDO_COL_FMT_YUV422H;
+ case 1:
+ return VDO_COL_FMT_YUV420;
+ case 2:
+ return VDO_COL_FMT_YUV444;
+ default:
+ break;
+ }
+ return VDO_COL_FMT_YUV444;
+}
+
+void sclr_set_media_format(vpp_media_format_t format)
+{
+ switch (format) {
+ case VPP_MEDIA_FMT_MPEG:
+ vppif_reg32_write(SCLR_MEDIAFMT_H264, 0x0);
+ break;
+ case VPP_MEDIA_FMT_H264:
+ vppif_reg32_write(SCLR_MEDIAFMT_H264, 0x1);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+void sclr_set_fb_addr(U32 y_addr, U32 c_addr)
+{
+ unsigned int line_y, line_c;
+ unsigned int offset_y, offset_c;
+ unsigned int pre_y, pre_c;
+
+ DBGMSG("y_addr:0x%08x, c_addr:0x%08x\n", y_addr, c_addr);
+
+ offset_y = offset_c = 0;
+ line_y = line_c = vppif_reg32_read(SCLR_YBUFWID);
+ switch (sclr_get_color_format()) {
+ case VDO_COL_FMT_YUV420:
+ offset_c /= 2;
+ line_c = 0;
+ break;
+ case VDO_COL_FMT_YUV422H:
+ break;
+ case VDO_COL_FMT_ARGB:
+ offset_y *= 4;
+ line_y *= 4;
+ break;
+ case VDO_COL_FMT_RGB_565:
+ offset_y *= 2;
+ line_y *= 2;
+ break;
+ default:
+ offset_c *= 2;
+ line_c *= 2;
+ break;
+ }
+ pre_y = vppif_reg32_in(REG_SCLR_YSA);
+ pre_c = vppif_reg32_in(REG_SCLR_CSA);
+ vppif_reg32_out(REG_SCLR_YSA, y_addr + offset_y);
+ vppif_reg32_out(REG_SCLR_CSA, c_addr + offset_c);
+}
+
+void sclr_get_fb_addr(U32 *y_addr, U32 *c_addr)
+{
+ *y_addr = vppif_reg32_in(REG_SCLR_YSA);
+ *c_addr = vppif_reg32_in(REG_SCLR_CSA);
+/* DBGMSG("y_addr:0x%08x, c_addr:0x%08x\n", *y_addr, *c_addr); */
+}
+
+void sclr_set_width(U32 y_pixel, U32 y_buffer)
+{
+ vppif_reg32_write(SCLR_YPXLWID, y_pixel);
+ vppif_reg32_write(SCLR_YBUFWID, y_buffer);
+}
+
+void sclr_get_width(U32 *p_y_pixel, U32 *p_y_buffer)
+{
+ *p_y_pixel = vppif_reg32_read(SCLR_YPXLWID);
+ *p_y_buffer = vppif_reg32_read(SCLR_YBUFWID);
+}
+
+void sclr_set_crop(U32 h_crop, U32 v_crop)
+{
+ vppif_reg32_write(SCLR_HCROP, h_crop);
+ vppif_reg32_write(SCLR_VCROP, v_crop);
+}
+
+void sclr_get_fb_info(U32 *width, U32 *act_width,
+ U32 *x_offset, U32 *y_offset)
+{
+ *width = vppif_reg32_read(SCLR_YBUFWID);
+ *act_width = vppif_reg32_read(SCLR_YPXLWID);
+ *x_offset = vppif_reg32_read(SCLR_HCROP);
+ *y_offset = vppif_reg32_read(SCLR_VCROP);
+}
+
+void sclr_set_threshold(U32 value)
+{
+ vppif_reg32_write(SCLR_FIFO_THR, value);
+}
+
+void sclw_set_mif_enable(vpp_flag_t enable)
+{
+ vppif_reg32_write(SCLW_MIF_ENABLE, enable);
+}
+
+void sclw_set_color_format(vdo_color_fmt format)
+{
+ /* 0-888(4 byte), 1-5515(2 byte), 2-666(4 byte), 3-565(2 byte) */
+ switch (format) {
+ case VDO_COL_FMT_RGB_666:
+ vppif_reg32_write(SCLW_COLFMT_RGB, 1);
+ vppif_reg32_write(SCL_IGS_MODE, 2);
+ break;
+ case VDO_COL_FMT_RGB_565:
+ vppif_reg32_write(SCLW_COLFMT_RGB, 1);
+ vppif_reg32_write(SCL_IGS_MODE, 3);
+ break;
+ case VDO_COL_FMT_RGB_1555:
+ vppif_reg32_write(SCLW_COLFMT_RGB, 1);
+ vppif_reg32_write(SCL_IGS_MODE, 1);
+ break;
+ case VDO_COL_FMT_ARGB:
+ vppif_reg32_write(SCLW_COLFMT_RGB, 1);
+ vppif_reg32_write(SCL_IGS_MODE, 0);
+ break;
+ case VDO_COL_FMT_YUV444:
+ vppif_reg32_write(SCLW_COLFMT_RGB, 0);
+ vppif_reg32_write(SCLW_COLFMT_YUV, 0);
+ vppif_reg32_write(SCL_IGS_MODE, 0);
+ break;
+ case VDO_COL_FMT_YUV422H:
+ case VDO_COL_FMT_YUV420:
+ vppif_reg32_write(SCLW_COLFMT_RGB, 0);
+ vppif_reg32_write(SCLW_COLFMT_YUV, 1);
+ vppif_reg32_write(SCL_IGS_MODE, 0);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+vdo_color_fmt sclw_get_color_format(void)
+{
+ if (vppif_reg32_read(SCLW_COLFMT_RGB)) {
+ switch (vppif_reg32_read(SCL_IGS_MODE)) {
+ case 0:
+ return VDO_COL_FMT_ARGB;
+ case 1:
+ return VDO_COL_FMT_RGB_1555;
+ case 2:
+ return VDO_COL_FMT_RGB_666;
+ case 3:
+ return VDO_COL_FMT_RGB_565;
+ }
+ }
+
+ if (vppif_reg32_read(SCLW_COLFMT_YUV))
+ return VDO_COL_FMT_YUV422H;
+ return VDO_COL_FMT_YUV444;
+}
+
+void sclw_set_alpha(int enable, char data)
+{
+ vppif_reg32_write(SCL_FIXED_ALPHA_DATA, data);
+ vppif_reg32_write(SCL_FIXED_ALPHA_ENABLE, enable);
+}
+
+void sclw_set_field_mode(vpp_display_format_t fmt)
+{
+ vppif_reg32_write(SCLR_TAR_DISP_FMT, fmt);
+}
+
+void sclw_set_fb_addr(U32 y_addr, U32 c_addr)
+{
+ DBGMSG("y_addr:0x%08x, c_addr:0x%08x\n", y_addr, c_addr);
+/* if( (y_addr & 0x3f) || (c_addr & 0x3f) ){
+ DPRINT("[SCL] *E* addr should align 64\n");
+ } */
+ vppif_reg32_out(REG_SCLW_YSA, y_addr);
+ vppif_reg32_out(REG_SCLW_CSA, c_addr);
+}
+
+void sclw_get_fb_addr(U32 *y_addr, U32 *c_addr)
+{
+ *y_addr = vppif_reg32_in(REG_SCLW_YSA);
+ *c_addr = vppif_reg32_in(REG_SCLW_CSA);
+ DBGMSG("y_addr:0x%08x, c_addr:0x%08x\n", *y_addr, *c_addr);
+}
+
+void sclw_set_fb_width(U32 width, U32 buf_width)
+{
+ vppif_reg32_write(SCLW_YPXLWID, width);
+ vppif_reg32_write(SCLW_YBUFWID, buf_width);
+ if (sclw_get_color_format() == VDO_COL_FMT_YUV444) {
+ vppif_reg32_write(SCLW_CPXLWID, width);
+ vppif_reg32_write(SCLW_CBUFWID, buf_width * 2);
+ } else {
+ vppif_reg32_write(SCLW_CPXLWID, width / 2);
+ vppif_reg32_write(SCLW_CBUFWID, buf_width);
+ }
+}
+
+void sclw_get_fb_width(U32 *width, U32 *buf_width)
+{
+ *width = vppif_reg32_read(SCLW_YPXLWID);
+ *buf_width = vppif_reg32_read(SCLW_YBUFWID);
+}
+
+void scl_R2_set_mif_enable(int enable)
+{
+ vppif_reg32_write(SCL_R2_MIF_EN, enable);
+}
+
+void scl_R2_set_colorbar(int enable, int wide, int inv)
+{
+ vppif_reg32_write(SCL_R2_COLOR_EN, enable);
+ vppif_reg32_write(SCL_R2_COLOR_WIDE, wide);
+ vppif_reg32_write(SCL_R2_COLOR_INV, inv);
+}
+
+void scl_R2_set_color_format(vdo_color_fmt colfmt)
+{
+ if (colfmt >= VDO_COL_FMT_ARGB) {
+ vppif_reg32_write(SCL_R2_RGB_MODE,
+ (colfmt == VDO_COL_FMT_RGB_565) ? 0x1 : 0x3);
+ return;
+ }
+ vppif_reg32_write(SCL_R2_RGB_MODE, 0x0);
+ switch (colfmt) {
+ case VDO_COL_FMT_YUV444:
+ vppif_reg32_write(SCL_R2_VFMT, 0x2);
+ break;
+ case VDO_COL_FMT_YUV422H:
+ vppif_reg32_write(SCL_R2_VFMT, 0x0);
+ break;
+ case VDO_COL_FMT_YUV420:
+ vppif_reg32_write(SCL_R2_VFMT, 0x1);
+ break;
+ default:
+ DBGMSG("*E* check the parameter.\n");
+ return;
+ }
+}
+
+vdo_color_fmt scl_R2_get_color_format(void)
+{
+ switch (vppif_reg32_read(SCL_R2_RGB_MODE)) {
+ case 0:
+ switch (vppif_reg32_read(SCL_R2_VFMT)) {
+ case 0:
+ return VDO_COL_FMT_YUV422H;
+ case 1:
+ return VDO_COL_FMT_YUV420;
+ case 2:
+ default:
+ return VDO_COL_FMT_YUV444;
+ }
+ break;
+ case 1:
+ return VDO_COL_FMT_RGB_565;
+ case 3:
+ default:
+ break;
+ }
+ return VDO_COL_FMT_ARGB;
+}
+
+void scl_R2_set_csc_mode(vpp_csc_t mode)
+{
+ vdo_color_fmt src_fmt, dst_fmt;
+
+ src_fmt = scl_R2_get_color_format();
+ dst_fmt = sclw_get_color_format();
+ mode = vpp_check_csc_mode(mode, src_fmt, dst_fmt, 0);
+
+ if (mode >= VPP_CSC_MAX)
+ vppif_reg32_write(SCL_R2_CSC_EN, VPP_FLAG_DISABLE);
+ else {
+ vppif_reg32_out(REG_SCL_R2_CSC1, vpp_csc_parm[mode][0]);
+ vppif_reg32_out(REG_SCL_R2_CSC2, vpp_csc_parm[mode][1]);
+ vppif_reg32_out(REG_SCL_R2_CSC3, vpp_csc_parm[mode][2]);
+ vppif_reg32_out(REG_SCL_R2_CSC4, vpp_csc_parm[mode][3]);
+ vppif_reg32_out(REG_SCL_R2_CSC5, vpp_csc_parm[mode][4]);
+ vppif_reg32_out(REG_SCL_R2_CSC6, vpp_csc_parm[mode][5]);
+ vppif_reg32_out(REG_SCL_R2_CSC, vpp_csc_parm[mode][6]);
+ vppif_reg32_write(SCL_R2_CSC_EN, VPP_FLAG_ENABLE);
+ }
+}
+
+void scl_R2_set_framebuffer(vdo_framebuf_t *fb)
+{
+ vppif_reg32_write(SCL_R2_IOFMT,
+ (fb->flag & VDO_FLAG_INTERLACE) ? 1 : 0);
+ scl_R2_set_color_format(fb->col_fmt);
+ vppif_reg32_out(REG_SCLR2_YSA, fb->y_addr);
+ vppif_reg32_out(REG_SCLR2_CSA, fb->c_addr);
+ vppif_reg32_write(SCL_R2_FBW, fb->fb_w);
+ vppif_reg32_write(SCL_R2_LNSIZE, fb->img_w);
+ vppif_reg32_write(SCL_R2_HCROP, fb->h_crop);
+ vppif_reg32_write(SCL_R2_VCROP, fb->v_crop);
+ scl_R2_set_csc_mode(p_scl->fb_p->csc_mode);
+}
+
+void scl_ALPHA_set_enable(int enable)
+{
+ vppif_reg32_write(SCL_ALPHA_COLORKEY_ENABLE, enable);
+}
+
+void scl_ALPHA_set_swap(int enable)
+{
+ /* 0-(alpha,1-alpha),1:(1-alpha,alpha) */
+ vppif_reg32_write(SCL_ALPHA_SWAP, enable);
+}
+
+void scl_ALPHA_set_src(int mode, int fixed)
+{
+ /* 0-RMIF1,1-RMIF2,2-Fixed ALPHA */
+ vppif_reg32_write(SCL_ALPHA_SRC, mode);
+ vppif_reg32_write(SCL_ALPHA_SRC_FIXED, fixed);
+}
+
+void scl_ALPHA_set_dst(int mode, int fixed)
+{
+ /* 0-RMIF1,1-RMIF2,2-Fixed ALPHA */
+ vppif_reg32_write(SCL_ALPHA_DST, mode);
+ vppif_reg32_write(SCL_ALPHA_DST_FIXED, fixed);
+}
+
+void scl_ALPHA_set_color_key(int rmif2, int comp, int mode, int colkey)
+{
+ /* 0-RMIF1,1-RMIF2 */
+ vppif_reg32_write(SCL_ALPHA_COLORKEY_FROM, rmif2);
+ /* 0-888,1-777,2-666,3-555 */
+ vppif_reg32_write(SCL_ALPHA_COLORKEY_COMP, comp);
+ /* (Non-Hit,Hit):0/1-(alpha,alpha),
+ 2-(alpha,pix1),3-(pix1,alpha),4-(alpha,pix2),
+ 5-(pix2,alpha),6-(pix1,pix2),7-(pix2,pix1) */
+ vppif_reg32_write(SCL_ALPHA_COLORKEY_MODE, mode);
+ vppif_reg32_out(REG_ALFA_COLORKEY_RGB, colkey);
+}
+
+void scl_set_overlap(vpp_overlap_t *p)
+{
+#if 0
+ DPRINT("alpha src %d,0x%x,dst %d,0x%x\n",
+ p->alpha_src_type, p->alpha_src,
+ p->alpha_dst_type, p->alpha_dst);
+ DPRINT("colkey from %d,comp %d,mode %d,0x%x\n",
+ p->color_key_from, p->color_key_comp,
+ p->color_key_mode, p->color_key);
+#endif
+ scl_ALPHA_set_src(p->alpha_src_type, p->alpha_src);
+ scl_ALPHA_set_dst(p->alpha_dst_type, p->alpha_dst);
+ scl_ALPHA_set_swap(p->alpha_swap);
+ scl_ALPHA_set_color_key(p->color_key_from, p->color_key_comp,
+ p->color_key_mode, p->color_key);
+}
+
+void scl_set_req_num(int ynum, int cnum)
+{
+ vppif_reg32_write(SCL_R_Y_REQ_NUM, ynum);
+ vppif_reg32_write(SCL_R_C_REQ_NUM, cnum);
+}
+
+static void scl_set_scale_PP(unsigned int src, unsigned int dst,
+ int horizontal)
+{
+ int gcd;
+
+/* DBGMSG("scale PP(s %d,d %d,is H %d)\n",src,dst,horizontal); */
+
+ /* gcd = scl_get_gcd(src,dst); */
+ gcd = 1;
+ src /= gcd;
+ dst /= gcd;
+
+ if (horizontal)
+ scl_set_H_scale(dst, src);
+ else
+ scl_set_V_scale(dst, src);
+}
+
+void scl_set_scale(unsigned int SRC_W, unsigned int SRC_H,
+ unsigned int DST_W, unsigned int DST_H)
+{
+ int h_scale_up;
+ int v_scale_up;
+
+ DBGMSG("[SCL] src(%dx%d),dst(%dx%d)\n", SRC_W, SRC_H, DST_W, DST_H);
+
+ h_scale_up = (DST_W > SRC_W) ? 1 : 0;
+ v_scale_up = (DST_H > SRC_H) ? 1 : 0;
+
+ if (((DST_W / SRC_W) >= 32) || ((DST_W / SRC_W) < 1/32))
+ DBGMSG("*W* SCL H scale rate invalid\n");
+
+ if (((DST_H / SRC_H) >= 32) || ((DST_H / SRC_H) < 1/32))
+ DBGMSG("*W* SCL V scale rate invalid\n");
+
+/* DBGMSG("scale H %d,V %d\n",h_scale_up,v_scale_up); */
+
+ sclr_set_mif2_enable(VPP_FLAG_DISABLE);
+ scl_set_scale_PP(SRC_W, DST_W, 1);
+ scl_set_scale_PP(SRC_H, DST_H, 0);
+ scl_set_scale_enable(v_scale_up, h_scale_up);
+
+ {
+ int rec_h, rec_v;
+ int h, v;
+
+ h = rec_h = 0;
+ if (SRC_W > DST_W) { /* scale down */
+ switch (p_scl->scale_mode) {
+ case VPP_SCALE_MODE_ADAPTIVE:
+ if ((DST_W * 2) > SRC_W) /* 1 > mode(3) > 1/2 */
+ h = 1; /* bilinear mode */
+ else
+ rec_h = 1; /* recursive mode */
+ break;
+ case VPP_SCALE_MODE_BILINEAR:
+ h = 1; /* bilinear mode */
+ break;
+ case VPP_SCALE_MODE_RECURSIVE:
+ rec_h = 1; /* recursive mode */
+ break;
+ default:
+ break;
+ }
+ }
+
+ v = rec_v = 0;
+ if (SRC_H > DST_H) { /* scale down */
+ switch (p_scl->scale_mode) {
+ case VPP_SCALE_MODE_ADAPTIVE:
+ if ((DST_H * 2) > SRC_H) /* 1 > mode(3) > 1/2 */
+ v = 1; /* bilinear mode */
+ else
+ rec_v = 1; /* recursive mode */
+ break;
+ case VPP_SCALE_MODE_BILINEAR:
+ v = 1; /* bilinear mode */
+ break;
+ case VPP_SCALE_MODE_RECURSIVE:
+ rec_v = 1; /* recursive mode */
+ break;
+ default:
+ break;
+ }
+ }
+
+ if (SRC_W == DST_W)
+ rec_h = 1;
+ if (SRC_H == DST_H)
+ rec_v = 1;
+ vppif_reg32_write(SCL_BILINEAR_H, h);
+ vppif_reg32_write(SCL_BILINEAR_V, v);
+ vppif_reg32_write(SCL_RECURSIVE_H, rec_h);
+ vppif_reg32_write(SCL_RECURSIVE_V, rec_v);
+
+ /* vertical bilinear mode */
+ if (v) {
+ vppif_reg32_write(SCL_VXWIDTH,
+ vppif_reg32_read(SCL_VXWIDTH) - 1);
+ vppif_reg32_write(SCL_DST_VXWIDTH,
+ vppif_reg32_read(SCL_VXWIDTH));
+ }
+ sclr_set_mif2_enable((v) ? VPP_FLAG_ENABLE : VPP_FLAG_DISABLE);
+ }
+}
+
+void sclr_set_framebuffer(vdo_framebuf_t *inbuf)
+{
+ sclr_set_color_format(inbuf->col_fmt);
+ sclr_set_crop(inbuf->h_crop, inbuf->v_crop);
+ sclr_set_width(inbuf->img_w, inbuf->fb_w);
+ sclr_set_fb_addr(inbuf->y_addr, inbuf->c_addr);
+ sclr_set_field_mode(vpp_get_fb_field(inbuf));
+}
+
+void scl_set_scale_timing(vdo_framebuf_t *s, vdo_framebuf_t *d)
+{
+ vpp_clock_t timing;
+ unsigned int pixel_clock;
+
+ /* scl TG */
+ timing.total_pixel_of_line = (d->img_w > s->img_w) ?
+ d->img_w : s->img_w;
+ timing.total_line_of_frame = (d->img_h > s->img_h) ?
+ d->img_h : s->img_h;
+ timing.begin_pixel_of_active = 60;
+ timing.end_pixel_of_active = timing.total_pixel_of_line + 60;
+ timing.total_pixel_of_line = timing.total_pixel_of_line + 120;
+ timing.begin_line_of_active = 8;
+ timing.end_line_of_active = timing.total_line_of_frame + 8;
+ timing.total_line_of_frame = timing.total_line_of_frame + 16;
+ timing.line_number_between_VBIS_VBIE = 4;
+ timing.line_number_between_PVBI_VBIS = 1;
+ pixel_clock = timing.total_pixel_of_line *
+ timing.total_line_of_frame * p_scl->fb_p->framerate;
+ scl_set_timing(&timing, pixel_clock);
+}
+
+void sclw_set_framebuffer(vdo_framebuf_t *fb)
+{
+ unsigned int yaddr, caddr;
+ int y_bpp, c_bpp;
+
+ vpp_get_colfmt_bpp(fb->col_fmt, &y_bpp, &c_bpp);
+ yaddr = fb->y_addr + ((fb->fb_w * fb->v_crop + fb->h_crop) * y_bpp / 8);
+ caddr = (c_bpp) ? (fb->c_addr + (((fb->fb_w * fb->v_crop +
+ fb->h_crop) / 2) * 2 * c_bpp / 8)) : 0;
+ sclw_set_fb_addr(yaddr, caddr);
+ sclw_set_color_format(fb->col_fmt);
+ sclw_set_fb_width(fb->img_w, fb->fb_w);
+ sclw_set_field_mode(vpp_get_fb_field(fb));
+ scl_set_csc_mode(p_scl->fb_p->csc_mode);
+}
+
+void scl_init(void *base)
+{
+ scl_mod_t *mod_p;
+ vpp_fb_base_t *fb_p;
+
+ mod_p = (scl_mod_t *) base;
+ fb_p = mod_p->fb_p;
+
+ scl_set_reg_level(VPP_REG_LEVEL_1);
+ scl_set_tg_enable(VPP_FLAG_DISABLE);
+ scl_set_enable(VPP_FLAG_DISABLE);
+ scl_set_int_enable(VPP_FLAG_DISABLE, VPP_INT_ALL);
+ sclr_set_mif_enable(VPP_FLAG_DISABLE);
+ sclr_set_mif2_enable(VPP_FLAG_DISABLE);
+ sclr_set_colorbar(VPP_FLAG_DISABLE, 0, 0);
+
+ scl_set_int_enable(VPP_FLAG_ENABLE, mod_p->int_catch);
+ scl_set_watchdog(fb_p->wait_ready);
+ scl_set_csc_mode(fb_p->csc_mode);
+ sclr_set_media_format(fb_p->media_fmt);
+ sclr_set_threshold(0xf);
+
+ /* filter default value */
+ vppif_reg32_write(SCL_1ST_LAYER_BOUNDARY, 48);
+ vppif_reg32_write(SCL_2ND_LAYER_BOUNDARY, 16);
+
+ vppif_reg32_write(SCL_FIELD_FILTER_Y_THD, 8);
+ vppif_reg32_write(SCL_FIELD_FILTER_C_THD, 8);
+ vppif_reg32_write(SCL_FIELD_FILTER_CONDITION, 0);
+
+ vppif_reg32_write(SCL_FRAME_FILTER_RGB, 0);
+ vppif_reg32_write(SCL_FRAME_FILTER_SAMPLER, 14);
+ vppif_reg32_write(SCL_FR_FILTER_SCENE_CHG_THD, 32);
+
+ scl_set_reg_update(VPP_FLAG_ENABLE);
+ scl_set_tg_enable(VPP_FLAG_DISABLE);
+}
+
+void sclw_init(void *base)
+{
+ sclw_set_mif_enable(VPP_FLAG_DISABLE);
+ sclw_set_fb_width(VPP_HD_DISP_RESX, VPP_HD_MAX_RESX);
+/* vppif_reg32_write(SCL_SCLDW_METHOD,0x1); */ /* drop line enable */
+}
+
+#ifdef __KERNEL__
+/* static struct work_struct scl_proc_scale_wq; */
+DECLARE_WAIT_QUEUE_HEAD(scl_proc_scale_event);
+static void scl_proc_scale_complete_work(struct work_struct *work)
+#else
+static void scl_proc_scale_complete_work(int arg)
+#endif
+{
+/* DPRINT("[SCL] scl_proc_scale_complete_work\n"); */
+ p_scl->scale_complete = 1;
+#ifdef __KERNEL__
+ wake_up_interruptible(&scl_proc_scale_event);
+#endif
+#if 0 /* avoid mutex in irq */
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_DISABLE, 0);
+#endif
+}
+
+struct timer_list scl_scale_timer;
+int scl_proc_scale_complete(void *arg)
+{
+ del_timer(&scl_scale_timer);
+
+/* DPRINT("[SCL] scl_proc_scale_complete\n"); */
+ if (vppif_reg32_read(SCL_INTSTS_TGERR)) {
+ DPRINT("[SCL] scale TG err 0x%x,0x%x\n",
+ vppif_reg32_in(REG_SCL_TG_STS),
+ vppif_reg32_in(REG_SCLW_FF_CTL));
+ vppif_reg32_out(REG_SCL_TG_STS+0x0, BIT0);
+ vppif_reg32_out(REG_SCLW_FF_CTL, 0x10101);
+ }
+ scl_set_tg_enable(VPP_FLAG_DISABLE);
+ vppm_set_int_enable(VPP_FLAG_DISABLE, SCL_COMPLETE_INT);
+ sclw_set_mif_enable(VPP_FLAG_DISABLE);
+ sclr_set_mif_enable(VPP_FLAG_DISABLE);
+ sclr_set_mif2_enable(VPP_FLAG_DISABLE);
+ scl_set_enable(VPP_FLAG_DISABLE);
+
+/* #ifdef __KERNEL__ */
+#if 0
+ INIT_WORK(&scl_proc_scale_wq, scl_proc_scale_complete_work);
+ schedule_work(&scl_proc_scale_wq);
+#else
+ scl_proc_scale_complete_work(0);
+#endif
+ return 0;
+}
+
+void scl_scale_timeout(int arg)
+{
+ DBG_ERR("scale timeout\n");
+#if 0
+ scl_reg_dump();
+ g_vpp.dbg_msg_level = VPP_DBGLVL_STREAM;
+ g_vpp.dbg_cnt = 1000;
+#endif
+ scl_proc_scale_complete(0);
+}
+
+void scl_set_scale_timer(int ms)
+{
+ if (scl_scale_timer.function)
+ del_timer(&scl_scale_timer);
+ init_timer(&scl_scale_timer);
+ scl_scale_timer.function = (void *) scl_scale_timeout;
+ scl_scale_timer.expires = jiffies + msecs_to_jiffies(ms);
+ add_timer(&scl_scale_timer);
+}
+
+int scl_proc_scale_finish(void)
+{
+#ifdef __KERNEL__
+ int ret;
+
+/* DPRINT("[SCL] scl_proc_scale_finish\n"); */
+ ret = wait_event_interruptible_timeout(scl_proc_scale_event,
+ (p_scl->scale_complete != 0), 3 * HZ);
+ if (ret == 0) { /* timeout */
+ DPRINT("[SCL] *E* wait scale timeout\n");
+ ret = -1;
+ } else {
+ ret = 0;
+ }
+#endif
+#if 1 /* avoid mutex in irq */
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_DISABLE, 0);
+#endif
+ return ret;
+}
+
+void scl_check_framebuf(vdo_framebuf_t *s,
+ vdo_framebuf_t *in, vdo_framebuf_t *out)
+{
+ if (s) {
+ if (s->img_w > WMT_SCL_H_DIV_MAX)
+ DBG_ERR("src w %d over %d\n",
+ s->img_w, WMT_SCL_H_DIV_MAX);
+ if (s->img_h > WMT_SCL_V_DIV_MAX)
+ DBG_ERR("src h %d over %d\n",
+ s->img_h, WMT_SCL_V_DIV_MAX);
+ if (s->col_fmt >= VDO_COL_FMT_ARGB) {
+ if (s->y_addr % 4)
+ DBG_ERR("src addr 0x%x not align 4\n",
+ s->y_addr);
+ }
+ if (s->fb_w > WMT_SCL_H_DIV_MAX)
+ DBG_ERR("src fb w %d over %d\n",
+ s->fb_w, WMT_SCL_H_DIV_MAX);
+ if (s->y_addr % 64)
+ DBG_ERR("src fb addr 0x%x no align 64\n", s->y_addr);
+ } else {
+ DBG_ERR("src null\n");
+ }
+
+ if (in) {
+ if (in->img_w > WMT_SCL_H_DIV_MAX)
+ DBG_ERR("in w %d over %d\n",
+ in->img_w, WMT_SCL_H_DIV_MAX);
+ if (in->img_h > WMT_SCL_V_DIV_MAX)
+ DBG_ERR("in h %d over %d\n",
+ in->img_h, WMT_SCL_V_DIV_MAX);
+ if (in->col_fmt >= VDO_COL_FMT_ARGB) {
+ if (in->y_addr % 4)
+ DBG_ERR("in addr 0x%x not align 4\n",
+ in->y_addr);
+ }
+ if (in->fb_w > WMT_SCL_H_DIV_MAX)
+ DBG_ERR("in fb w %d over %d\n",
+ in->fb_w, WMT_SCL_H_DIV_MAX);
+ if (in->y_addr % 64)
+ DBG_ERR("in fb addr 0x%x no align 64\n", in->y_addr);
+ }
+
+ if (out) {
+ if (s && (s->img_w == out->img_w)) {
+ if (out->img_w > WMT_SCL_H_DIV_MAX)
+ DBG_ERR("out w %d over %d\n", out->img_w,
+ WMT_SCL_H_DIV_MAX);
+ } else {
+ if (out->img_w > WMT_SCL_SCALE_DST_H_MAX)
+ DBG_ERR("out w %d over %d\n", out->img_w,
+ WMT_SCL_SCALE_DST_H_MAX);
+
+ }
+ if (out->col_fmt >= VDO_COL_FMT_ARGB) {
+ if (out->y_addr % 4)
+ DBG_ERR("out addr 0x%x not align 4\n",
+ out->y_addr);
+ }
+ if (out->fb_w > WMT_SCL_H_DIV_MAX)
+ DBG_ERR("out fb w %d over %d\n",
+ out->fb_w, WMT_SCL_H_DIV_MAX);
+ if (out->y_addr % 64)
+ DBG_ERR("out fb addr 0x%x no align 64\n", out->y_addr);
+ } else {
+ DBG_ERR("out null\n");
+ }
+}
+
+#define CONFIG_VPP_CHECK_SCL_STATUS
+int scl_set_scale_overlap(vdo_framebuf_t *s,
+ vdo_framebuf_t *in, vdo_framebuf_t *out)
+{
+ int ret = 0;
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_SCALE)) {
+ scl_check_framebuf(s, in, out);
+
+ if (s)
+ vpp_show_framebuf("src1", s);
+ if (in)
+ vpp_show_framebuf("src2", in);
+ if (out)
+ vpp_show_framebuf("dst", out);
+ }
+
+ if (p_scl->scale_sync)
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_ENABLE, 0);
+
+ scl_set_timing_master(VPP_MOD_SCL);
+
+ if (s) {
+ p_scl->fb_p->fb = *s;
+ sclr_set_framebuffer(s);
+ sclr_set_mif_enable(VPP_FLAG_ENABLE);
+ } else {
+ DPRINT("[SCL] *E* no source\n");
+ return -1;
+ }
+
+ if (in && (in->y_addr == 0))
+ in = 0;
+
+ scl_ALPHA_set_enable((in) ? VPP_FLAG_ENABLE : VPP_FLAG_DISABLE);
+ scl_R2_set_mif_enable((in) ? VPP_FLAG_ENABLE : VPP_FLAG_DISABLE);
+ if (in)
+ scl_R2_set_framebuffer(in);
+ if (out) {
+ p_sclw->fb_p->fb = *out;
+ sclw_set_framebuffer(out);
+ } else {
+ DPRINT("[SCL] *E* no dest\n");
+ return -1;
+ }
+
+ scl_set_scale(s->img_w, s->img_h, out->img_w, out->img_h);
+ scl_set_scale_timing(s, out);
+
+ /* scale process */
+ scl_set_enable(VPP_FLAG_ENABLE);
+ vppif_reg32_write(SCL_ONESHOT_ENABLE, 1);
+ sclw_set_mif_enable(VPP_FLAG_ENABLE);
+ scl_set_tg_enable(VPP_FLAG_ENABLE);
+#ifdef CONFIG_VPP_CHECK_SCL_STATUS
+ vppif_reg32_out(REG_SCL_TG_STS+0x0, BIT0);
+ vppif_reg32_out(REG_SCLW_FF_CTL, 0x10101);
+#endif
+ p_scl->scale_complete = 0;
+ vppm_set_int_enable(VPP_FLAG_ENABLE, SCL_COMPLETE_INT);
+
+#if 0 /* for debug scale */
+ scl_reg_dump();
+#endif
+
+ if (p_scl->scale_sync) {
+ ret = vpp_irqproc_work(SCL_COMPLETE_INT,
+ (void *)scl_proc_scale_complete, 0, 100, 1);
+ scl_proc_scale_finish();
+ } else {
+ vpp_irqproc_work(SCL_COMPLETE_INT,
+ (void *)scl_proc_scale_complete, 0, 0, 1);
+ scl_set_scale_timer(100);
+ }
+ return ret;
+}
+
+int scl_proc_scale(vdo_framebuf_t *src_fb, vdo_framebuf_t *dst_fb)
+{
+ int ret = 0;
+
+ if (dst_fb->col_fmt == VDO_COL_FMT_YUV420) {
+ int size;
+ unsigned int buf;
+ vdo_framebuf_t dfb;
+
+ dfb = *dst_fb; /* backup dst fb */
+
+ /* alloc memory */
+ size = dst_fb->img_w * dst_fb->img_h + 64;
+ buf = (unsigned int) kmalloc(size, GFP_KERNEL);
+ if (!buf) {
+ DPRINT("[SCL] *E* malloc fail %d\n", size);
+ return -1;
+ }
+
+ if (buf % 64)
+ buf = buf + (64 - (buf % 64));
+
+ /* scale for Y */
+ dst_fb->c_addr = buf;
+ ret = scl_set_scale_overlap(src_fb, 0, dst_fb);
+ if (ret == 0) {
+ /* V 1/2 scale for C */
+ dst_fb->y_addr = buf;
+ dst_fb->c_addr = dfb.c_addr;
+ dst_fb->img_h = dfb.img_h / 2;
+ ret = scl_set_scale_overlap(src_fb, 0, dst_fb);
+ }
+ kfree((void *)buf);
+ *dst_fb = dfb; /* restore dst fb */
+ } else {
+ ret = scl_set_scale_overlap(src_fb, 0, dst_fb);
+ }
+ return ret;
+}
+
+#ifdef CONFIG_PM
+static unsigned int *scl_pm_bk2;
+static unsigned int scl_pm_enable, scl_pm_tg;
+static unsigned int scl_pm_r_mif1, scl_pm_r_mif2, scl_pm_w_mif;
+void scl_suspend(int sts)
+{
+ switch (sts) {
+ case 0: /* disable module */
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_ENABLE, 1);
+ scl_pm_enable = vppif_reg32_read(SCL_ALU_ENABLE);
+ vppif_reg32_write(SCL_ALU_ENABLE, 0);
+ scl_pm_r_mif1 = vppif_reg32_read(SCLR_MIF_ENABLE);
+ scl_pm_r_mif2 = vppif_reg32_read(SCL_R2_MIF_EN);
+ vppif_reg32_write(SCL_R2_MIF_EN, 0);
+ vppif_reg32_write(SCLR_MIF_ENABLE, 0);
+ scl_pm_w_mif = vppif_reg32_read(SCLW_MIF_ENABLE);
+ vppif_reg32_write(SCLW_MIF_ENABLE, 0);
+ break;
+ case 1: /* disable tg */
+ scl_pm_tg = vppif_reg32_read(SCL_TG_ENABLE);
+ vppif_reg32_write(SCL_TG_ENABLE, 0);
+ break;
+ case 2: /* backup register */
+ p_scl->reg_bk = vpp_backup_reg(REG_SCL_BASE1_BEGIN,
+ (REG_SCL_BASE1_END - REG_SCL_BASE1_BEGIN));
+ scl_pm_bk2 = vpp_backup_reg(REG_SCL_BASE2_BEGIN,
+ (REG_SCL_BASE2_END - REG_SCL_BASE2_BEGIN));
+ break;
+ default:
+ break;
+ }
+}
+
+void scl_resume(int sts)
+{
+ switch (sts) {
+ case 0: /* restore register */
+ vpp_restore_reg(REG_SCL_BASE1_BEGIN,
+ (REG_SCL_BASE1_END - REG_SCL_BASE1_BEGIN),
+ p_scl->reg_bk);
+ vpp_restore_reg(REG_SCL_BASE2_BEGIN,
+ (REG_SCL_BASE2_END - REG_SCL_BASE2_BEGIN), scl_pm_bk2);
+ p_scl->reg_bk = 0;
+ scl_pm_bk2 = 0;
+ break;
+ case 1: /* enable module */
+ vppif_reg32_write(SCLW_MIF_ENABLE, scl_pm_w_mif);
+ vppif_reg32_write(SCLR_MIF_ENABLE, scl_pm_r_mif1);
+ vppif_reg32_write(SCL_R2_MIF_EN, scl_pm_r_mif2);
+ vppif_reg32_write(SCL_ALU_ENABLE, scl_pm_enable);
+ break;
+ case 2: /* enable tg */
+ vppif_reg32_write(SCL_TG_ENABLE, scl_pm_tg);
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_DISABLE, 1);
+ break;
+ default:
+ break;
+ }
+}
+#else
+#define scl_suspend NULL
+#define scl_resume NULL
+#endif
+
+int scl_mod_init(void)
+{
+ vpp_fb_base_t *mod_fb_p;
+ vdo_framebuf_t *fb_p;
+
+ /* -------------------- SCL module -------------------- */
+ {
+ scl_mod_t *scl_mod_p;
+
+ scl_mod_p = (scl_mod_t *) vpp_mod_register(VPP_MOD_SCL,
+ sizeof(scl_mod_t), VPP_MOD_FLAG_FRAMEBUF);
+ if (!scl_mod_p) {
+ DPRINT("*E* SCL module register fail\n");
+ return -1;
+ }
+
+ /* module member variable */
+ scl_mod_p->int_catch = VPP_INT_NULL;
+ scl_mod_p->scale_mode = VPP_SCALE_MODE_ADAPTIVE;
+ scl_mod_p->pm = DEV_SCL444U;
+ scl_mod_p->filter_mode = VPP_FILTER_SCALE;
+
+ /* module member function */
+ scl_mod_p->init = scl_init;
+ scl_mod_p->set_enable = scl_set_enable;
+ scl_mod_p->set_colorbar = sclr_set_colorbar;
+ scl_mod_p->dump_reg = scl_reg_dump;
+ scl_mod_p->get_sts = scl_get_int_status;
+ scl_mod_p->clr_sts = scl_clean_int_status;
+ scl_mod_p->scale = scl_proc_scale;
+ scl_mod_p->scale_finish = scl_proc_scale_finish;
+ scl_mod_p->suspend = scl_suspend;
+ scl_mod_p->resume = scl_resume;
+
+ /* module frame buffer variable */
+ mod_fb_p = scl_mod_p->fb_p;
+ fb_p = &mod_fb_p->fb;
+
+ fb_p->y_addr = 0;
+ fb_p->c_addr = 0;
+ fb_p->col_fmt = VDO_COL_FMT_YUV422H;
+ fb_p->img_w = VPP_HD_DISP_RESX;
+ fb_p->img_h = VPP_HD_DISP_RESY;
+ fb_p->fb_w = VPP_HD_MAX_RESX;
+ fb_p->fb_h = VPP_HD_MAX_RESY;
+ fb_p->h_crop = 0;
+ fb_p->v_crop = 0;
+
+ /* module frame buffer member function */
+ mod_fb_p->csc_mode = VPP_CSC_RGB2YUV_SDTV_0_255;
+ mod_fb_p->set_framebuf = sclr_set_framebuffer;
+ mod_fb_p->set_addr = sclr_set_fb_addr;
+ mod_fb_p->get_addr = sclr_get_fb_addr;
+ mod_fb_p->set_csc = scl_set_csc_mode;
+ mod_fb_p->framerate = 0x7fffffff;
+ mod_fb_p->wait_ready = 0xffffffff;
+ mod_fb_p->capability = BIT(VDO_COL_FMT_YUV420)
+ | BIT(VDO_COL_FMT_YUV422H) | BIT(VDO_COL_FMT_YUV444)
+ | BIT(VDO_COL_FMT_ARGB) | BIT(VDO_COL_FMT_RGB_565)
+ | VPP_FB_FLAG_CSC | VPP_FB_FLAG_FIELD;
+ p_scl = scl_mod_p;
+ p_scl->scale_complete = 1;
+ p_scl->scale_sync = 1;
+ }
+
+ /* -------------------- SCLW module -------------------- */
+ {
+ sclw_mod_t *sclw_mod_p;
+
+ sclw_mod_p = (sclw_mod_t *) vpp_mod_register(VPP_MOD_SCLW,
+ sizeof(sclw_mod_t), VPP_MOD_FLAG_FRAMEBUF);
+ if (!sclw_mod_p) {
+ DPRINT("*E* SCLW module register fail\n");
+ return -1;
+ }
+
+ /* module member variable */
+ sclw_mod_p->int_catch = VPP_INT_NULL;
+
+ /* module member function */
+ sclw_mod_p->init = sclw_init;
+ sclw_mod_p->set_enable = sclw_set_mif_enable;
+
+ /* module frame buffer */
+ mod_fb_p = sclw_mod_p->fb_p;
+ fb_p = &mod_fb_p->fb;
+
+ fb_p->y_addr = 0;
+ fb_p->c_addr = 0;
+ fb_p->col_fmt = VDO_COL_FMT_YUV422H;
+ fb_p->img_w = VPP_HD_DISP_RESX;
+ fb_p->img_h = VPP_HD_DISP_RESY;
+ fb_p->fb_w = VPP_HD_MAX_RESX;
+ fb_p->fb_h = VPP_HD_MAX_RESY;
+ fb_p->h_crop = 0;
+ fb_p->v_crop = 0;
+
+ /* module frame buffer member function */
+ mod_fb_p->csc_mode = VPP_CSC_RGB2YUV_SDTV_0_255;
+ mod_fb_p->set_framebuf = sclw_set_framebuffer;
+ mod_fb_p->set_addr = sclw_set_fb_addr;
+ mod_fb_p->get_addr = sclw_get_fb_addr;
+ mod_fb_p->set_csc = scl_set_csc_mode;
+ mod_fb_p->wait_ready = 0xffffffff;
+ mod_fb_p->capability = BIT(VDO_COL_FMT_YUV422H)
+ | BIT(VDO_COL_FMT_YUV444) | BIT(VDO_COL_FMT_ARGB)
+ | BIT(VDO_COL_FMT_RGB_565) | VPP_FB_FLAG_CSC
+ | BIT(VDO_COL_FMT_YUV420);
+ p_sclw = sclw_mod_p;
+ }
+ return 0;
+}
+module_init(scl_mod_init);
+#endif /* WMT_FTBLK_SCL */
diff --git a/drivers/video/wmt/scl.h b/drivers/video/wmt/scl.h
new file mode 100644
index 00000000..5ab9d7b6
--- /dev/null
+++ b/drivers/video/wmt/scl.h
@@ -0,0 +1,126 @@
+/*++
+ * linux/drivers/video/wmt/scl.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp.h"
+
+#ifndef SCL_H
+#define SCL_H
+
+#define SCL_COMPLETE_INT VPP_INT_SCL_PVBI
+
+typedef struct {
+ VPP_MOD_BASE;
+
+ int (*scale)(vdo_framebuf_t *src, vdo_framebuf_t *dst);
+ int (*scale_finish)(void);
+
+ vpp_scale_mode_t scale_mode;
+ vpp_scale_mode_t scale_sync;
+ int scale_complete;
+ vpp_filter_mode_t filter_mode;
+
+ vpp_dbg_timer_t overlap_timer;
+ vpp_dbg_timer_t scale_timer;
+} scl_mod_t;
+
+typedef struct {
+ VPP_MOD_BASE;
+} sclw_mod_t;
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifdef SCL_C
+#define EXTERN
+#else
+#define EXTERN extern
+#endif
+
+EXTERN scl_mod_t *p_scl;
+EXTERN sclw_mod_t *p_sclw;
+
+#ifdef WMT_FTBLK_SCL
+
+EXTERN void scl_reg_dump(void);
+EXTERN void sclr_set_framebuffer(vdo_framebuf_t *inbuf);
+EXTERN void sclw_set_framebuffer(vdo_framebuf_t *outbuf);
+
+EXTERN void scl_set_enable(vpp_flag_t enable);
+EXTERN void scl_set_reg_update(vpp_flag_t enable);
+EXTERN void scl_set_reg_level(vpp_reglevel_t level);
+EXTERN void scl_set_int_enable(vpp_flag_t enable, vpp_int_t int_bit);
+EXTERN vpp_int_err_t scl_get_int_status(void);
+EXTERN void scl_clean_int_status(vpp_int_err_t int_sts);
+EXTERN void scl_set_csc_mode(vpp_csc_t mode);
+EXTERN void scl_set_scale_enable(vpp_flag_t vscl_enable,
+ vpp_flag_t hscl_enable);
+EXTERN void scl_set_V_scale(int A, int B);
+EXTERN void scl_set_H_scale(int A, int B);
+EXTERN void scl_set_crop(int offset_x, int offset_y);
+EXTERN void scl_set_tg_enable(vpp_flag_t enable);
+EXTERN void scl_set_timing_master(vpp_mod_t mod_bit);
+EXTERN vpp_mod_t scl_get_timing_master(void);
+EXTERN void scl_set_drop_line(vpp_flag_t enable);
+EXTERN void sclr_get_fb_info(U32 *width, U32 *act_width,
+ U32 *x_offset, U32 *y_offset);
+EXTERN void sclr_set_mif_enable(vpp_flag_t enable);
+EXTERN void sclr_set_mif2_enable(vpp_flag_t enable);
+EXTERN void sclr_set_colorbar(vpp_flag_t enable, int width, int inverse);
+EXTERN void sclr_set_display_format(vpp_display_format_t source,
+ vpp_display_format_t target);
+EXTERN void sclr_set_field_mode(vpp_display_format_t fmt);
+EXTERN void sclr_set_color_format(vdo_color_fmt format);
+EXTERN vdo_color_fmt sclr_get_color_format(void);
+EXTERN void sclr_set_media_format(vpp_media_format_t format);
+EXTERN void sclr_set_fb_addr(U32 y_addr, U32 c_addr);
+EXTERN void sclr_get_fb_addr(U32 *y_addr, U32 *c_addr);
+EXTERN void sclr_set_width(U32 y_pixel, U32 y_buffer);
+EXTERN void sclr_get_width(U32 *p_y_pixel, U32 *p_y_buffer);
+EXTERN void sclr_set_crop(U32 h_crop, U32 v_crop);
+EXTERN void sclr_set_threshold(U32 value);
+EXTERN vdo_color_fmt scl_R2_get_color_format(void);
+EXTERN void sclw_set_mif_enable(vpp_flag_t enable);
+EXTERN void sclw_set_color_format(vdo_color_fmt format);
+EXTERN vdo_color_fmt sclw_get_color_format(void);
+EXTERN void sclw_set_field_mode(vpp_display_format_t fmt);
+EXTERN void sclw_set_fb_addr(U32 y_addr, U32 c_addr);
+EXTERN void sclw_get_fb_addr(U32 *y_addr, U32 *c_addr);
+EXTERN void sclw_set_fb_width(U32 y_pixel, U32 y_buffer);
+EXTERN void sclw_get_fb_width(U32 *width, U32 *buf_width);
+EXTERN void scl_set_scale(unsigned int SRC_W, unsigned int SRC_H,
+ unsigned int DST_W, unsigned int DST_H);
+EXTERN int scl_mod_init(void);
+EXTERN void scl_set_deblock_enable(int enable);
+EXTERN void scl_set_field_filter_enable(int enable);
+EXTERN void scl_set_frame_filter_enable(int enable);
+EXTERN int scl_set_scale_overlap(vdo_framebuf_t *s,
+ vdo_framebuf_t *in, vdo_framebuf_t *out);
+EXTERN void scl_set_overlap(vpp_overlap_t *p);
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* WMT_FTBLK_SCL */
+#undef EXTERN
+#endif /* SCL_H */
diff --git a/drivers/video/wmt/sw_i2c.c b/drivers/video/wmt/sw_i2c.c
new file mode 100644
index 00000000..af05e7a4
--- /dev/null
+++ b/drivers/video/wmt/sw_i2c.c
@@ -0,0 +1,426 @@
+/*++
+ * linux/drivers/video/wmt/sw_i2c.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp-osif.h"
+#include "sw_i2c.h"
+
+#define SPEED 50 /* 5000000 */
+
+#define delay_time 30
+
+struct swi2c_reg_bk_t {
+ unsigned int gpio_en;
+ unsigned int out_en;
+ unsigned int data_out;
+ unsigned int pull_en;
+};
+
+#ifdef __KERNEL__
+static DEFINE_SPINLOCK(swi2c_irqlock);
+unsigned long swi2c_irqlock_flags;
+#endif
+void wmt_swi2c_lock(int lock)
+{
+#ifdef __KERNEL__
+ if (lock)
+ spin_lock_irqsave(&swi2c_irqlock, swi2c_irqlock_flags);
+ else
+ spin_unlock_irqrestore(&swi2c_irqlock, swi2c_irqlock_flags);
+#endif
+}
+
+swi2c_reg_t *swi2c_scl, *swi2c_sda;
+
+void wmt_swi2c_delay(unsigned int time)
+{
+ udelay(time);
+}
+
+void wmt_swi2c_SetSDAInput(void)
+{
+ REG16_VAL(swi2c_sda->gpio_en) |= swi2c_sda->bit_mask;
+ REG16_VAL(swi2c_sda->out_en) &= ~swi2c_sda->bit_mask;
+}
+
+
+void wmt_swi2c_SetSDAOutput(void)
+{
+ REG16_VAL(swi2c_sda->gpio_en) |= swi2c_sda->bit_mask;
+ REG16_VAL(swi2c_sda->out_en) |= swi2c_sda->bit_mask;
+}
+
+bool wmt_swi2c_GetSDA(void) /* bit */
+{
+ if (*(volatile unsigned short *)(swi2c_sda->data_in) &
+ swi2c_sda->bit_mask)
+#ifdef CFG_LOADER
+ return 1;
+ return 0;
+#else
+ return (volatile bool) 1;
+ return (volatile bool) 0;
+#endif
+}
+
+bool wmt_swi2c_GetSCL(void) /* bit */
+{
+ if (*(volatile unsigned short *)(swi2c_scl->data_in) &
+ swi2c_scl->bit_mask)
+#ifdef CFG_LOADER
+ return 1;
+ return 0;
+#else
+ return (volatile bool) 1;
+ return (volatile bool) 0;
+#endif
+}
+
+void wmt_swi2c_SetSDA(int high)
+{
+ if (high) {
+ /* set to GPI and pull high */
+ REG16_VAL(swi2c_sda->gpio_en) |= swi2c_sda->bit_mask;
+ REG16_VAL(swi2c_sda->out_en) &= ~swi2c_sda->bit_mask;
+ if (swi2c_sda->pull_en)
+ REG16_VAL(swi2c_sda->pull_en) &=
+ ~swi2c_sda->pull_en_bit_mask;
+ } else {
+ REG16_VAL(swi2c_sda->gpio_en) |= swi2c_sda->bit_mask;
+ REG16_VAL(swi2c_sda->out_en) |= swi2c_sda->bit_mask;
+ REG16_VAL(swi2c_sda->data_out) &= ~swi2c_sda->bit_mask;
+ }
+}
+
+void wmt_swi2c_SetSCL(int high)
+{
+ if (high) {
+ REG16_VAL(swi2c_scl->gpio_en) |= swi2c_scl->bit_mask;
+ REG16_VAL(swi2c_scl->out_en) &= ~swi2c_scl->bit_mask;
+ if (swi2c_scl->pull_en)
+ REG16_VAL(swi2c_scl->pull_en) &=
+ ~swi2c_scl->pull_en_bit_mask;
+ } else {
+ REG16_VAL(swi2c_scl->gpio_en) |= swi2c_scl->bit_mask;
+ REG16_VAL(swi2c_scl->out_en) |= swi2c_scl->bit_mask;
+ REG16_VAL(swi2c_scl->data_out) &= ~swi2c_scl->bit_mask;
+ }
+}
+
+int wmt_swi2c_SetData(int high)
+{
+ unsigned int wait = 1;
+
+ wmt_swi2c_SetSDA(high);
+ do {
+ if (wmt_swi2c_GetSDA() == ((high) ? 1 : 0))
+ return 0;
+ } while (wait++ < SPEED);
+ return 1;
+}
+
+int wmt_swi2c_SetClock(int high)
+{
+ unsigned int wait = 1;
+
+ udelay(5); /* 3-100kHz, 5-80kHz */
+
+ wmt_swi2c_SetSCL(high);
+ do {
+ if (wmt_swi2c_GetSCL() == ((high) ? 1 : 0))
+ return 0;
+ } while (wait++ < SPEED);
+ return 1; /* fail */
+}
+
+int wmt_swi2c_StartI2C(void)
+{
+ if (wmt_swi2c_SetData(1))
+ return 1;
+ if (wmt_swi2c_SetClock(1))
+ return 2;
+ if (wmt_swi2c_SetData(0))
+ return 3;
+ wmt_swi2c_delay(0);
+ if (wmt_swi2c_SetClock(0))
+ return 4;
+ return 0; /* success */
+}
+
+int wmt_swi2c_StopI2C(void)
+{
+ if (wmt_swi2c_SetData(0))
+ return 1;
+ if (wmt_swi2c_SetClock(0))
+ return 1;
+ if (wmt_swi2c_SetClock(1))
+ return 1;
+ wmt_swi2c_delay(0);
+ if (wmt_swi2c_SetData(1))
+ return 1;
+ return 0; /* success */
+}
+
+int wmt_swi2c_WriteAck(unsigned char byte)
+{
+ int ret;
+ int bit;
+
+ for (bit = 7; bit >= 0; bit--) {
+ wmt_swi2c_SetData(byte & 0x80);
+ byte <<= 1;
+ if (wmt_swi2c_SetClock(1))
+ return 1;
+ if (wmt_swi2c_SetClock(0))
+ return 1;
+ }
+ ret = 0;
+ wmt_swi2c_SetSDAInput();
+ if (wmt_swi2c_SetClock(1))
+ ret = 1;
+ else if (wmt_swi2c_GetSDA())
+ ret = (wmt_swi2c_SetClock(0)) ? 1 : 0;
+ else if (wmt_swi2c_SetClock(0))
+ ret = 1;
+ wmt_swi2c_SetSDAOutput();
+ return ret;
+}
+
+int wmt_swi2c_ReadAck(unsigned char *byte, int last)
+{
+ unsigned char i;
+ unsigned char Data = 0;
+
+ wmt_swi2c_SetSDAInput();
+
+ for (i = 0; i < 8; i++) {
+ if (wmt_swi2c_SetClock(1)) {
+ wmt_swi2c_SetSDAOutput();
+ return 1;
+ }
+ Data <<= 1;
+ wmt_swi2c_delay(0);
+ Data |= wmt_swi2c_GetSDA();
+ if (wmt_swi2c_SetClock(0)) {
+ wmt_swi2c_SetSDAOutput();
+ return 1;
+ }
+ }
+
+ *byte = Data;
+ wmt_swi2c_SetSDAOutput();
+
+ wmt_swi2c_delay(0);
+ if (wmt_swi2c_SetData((last) ? 1 : 0))
+ return 1;
+ if (wmt_swi2c_SetClock(1))
+ return 1;
+ wmt_swi2c_delay(0);
+ if (wmt_swi2c_SetClock(0))
+ return 1;
+ wmt_swi2c_delay(0);
+ return 0;
+}
+
+int wmt_swi2c_tx(
+ char addr,
+ char *buf,
+ int cnt,
+ int *ret_cnt
+)
+{
+ int ret = 0;
+ unsigned char i;
+
+ wmt_swi2c_lock(1);
+
+ ret |= wmt_swi2c_StartI2C();
+ if (ret)
+ goto tx_end;
+ ret |= wmt_swi2c_WriteAck(addr);
+ if (ret)
+ goto tx_end;
+ for (i = 0; i < cnt; i++) {
+ ret |= wmt_swi2c_WriteAck(buf[i]);
+ if (ret)
+ goto tx_end;
+ }
+ ret |= wmt_swi2c_StopI2C();
+tx_end:
+ wmt_swi2c_lock(0);
+ return ret;
+}
+
+int wmt_swi2c_rx(
+ char addr,
+ char *buf,
+ int cnt,
+ int *ret_cnt
+)
+{
+ int ret = 0;
+ int i;
+
+ wmt_swi2c_lock(1);
+
+ ret |= wmt_swi2c_StartI2C();
+ if (ret)
+ goto rx_end;
+ ret |= wmt_swi2c_WriteAck(addr | 0x01);
+ if (ret)
+ goto rx_end;
+ for (i = 0; i < cnt; i++) {
+ ret |= wmt_swi2c_ReadAck((unsigned char *)&buf[i],
+ (i == (cnt - 1)));
+ if (ret)
+ goto rx_end;
+ }
+ ret |= wmt_swi2c_StopI2C();
+rx_end:
+ wmt_swi2c_lock(0);
+ return ret;
+}
+
+void wmt_swi2c_reg_bk(swi2c_reg_t *reg_p, struct swi2c_reg_bk_t *reg_bk,
+ int bk)
+{
+ if (bk) {
+ reg_bk->gpio_en = REG16_VAL(reg_p->gpio_en);
+ reg_bk->out_en = REG16_VAL(reg_p->out_en);
+ reg_bk->data_out = REG16_VAL(reg_p->data_out);
+ reg_bk->pull_en = REG16_VAL(reg_p->pull_en);
+ } else {
+ REG16_VAL(reg_p->gpio_en) = reg_bk->gpio_en;
+ REG16_VAL(reg_p->out_en) = reg_bk->out_en;
+ REG16_VAL(reg_p->data_out) = reg_bk->data_out;
+ REG16_VAL(reg_p->pull_en) = reg_bk->pull_en;
+ }
+}
+
+int wmt_swi2c_read(
+ swi2c_handle_t *handle,
+ char addr,
+ char index,
+ char *buf,
+ int cnt
+)
+{
+ int ret = 0;
+ char buffer[24];
+ int temp = 0;
+ struct swi2c_reg_bk_t scl_bk, sda_bk;
+
+ swi2c_scl = handle->scl_reg;
+ swi2c_sda = handle->sda_reg;
+
+ wmt_swi2c_reg_bk(swi2c_scl, &scl_bk, 1);
+ wmt_swi2c_reg_bk(swi2c_sda, &sda_bk, 1);
+
+ buffer[0] = index;
+ ret = wmt_swi2c_tx(addr, buffer, 1, &temp);
+ if (ret) {
+ DPRINT("[SWI2C] *E* tx fail\n");
+ goto exit;
+ }
+ ret = wmt_swi2c_rx(addr, buf, cnt, &temp);
+ if (ret) {
+ DPRINT("[SWI2C] *E* rx fail\n");
+ goto exit;
+ }
+exit:
+ wmt_swi2c_reg_bk(swi2c_scl, &scl_bk, 0);
+ wmt_swi2c_reg_bk(swi2c_sda, &sda_bk, 0);
+ return ret;
+}
+EXPORT_SYMBOL(wmt_swi2c_read);
+
+int wmt_swi2c_write(
+ swi2c_handle_t *handle,
+ char addr,
+ char index,
+ char *buf,
+ int cnt
+)
+{
+ int ret = 0;
+ char buffer[24];
+ int temp;
+ struct swi2c_reg_bk_t scl_bk, sda_bk;
+
+ swi2c_scl = handle->scl_reg;
+ swi2c_sda = handle->sda_reg;
+
+ wmt_swi2c_reg_bk(swi2c_scl, &scl_bk, 1);
+ wmt_swi2c_reg_bk(swi2c_sda, &sda_bk, 1);
+
+ buffer[0] = index;
+ buffer[1] = *buf;
+ ret = wmt_swi2c_tx(addr, buffer, cnt, &temp);
+ if (ret) {
+ DPRINT("[SWI2C] *E* tx fail\n");
+ goto exit;
+ }
+exit:
+ wmt_swi2c_reg_bk(swi2c_scl, &scl_bk, 0);
+ wmt_swi2c_reg_bk(swi2c_sda, &sda_bk, 0);
+ return ret;
+}
+EXPORT_SYMBOL(wmt_swi2c_write);
+
+int wmt_swi2c_check(swi2c_handle_t *handle)
+{
+ int ret = 0;
+#if 0
+ swi2c_reg_t *reg_p;
+ struct swi2c_reg_bk_t scl_bk, sda_bk;
+
+ swi2c_scl = handle->scl_reg;
+ swi2c_sda = handle->sda_reg;
+
+ wmt_swi2c_reg_bk(swi2c_scl, &scl_bk, 1);
+ wmt_swi2c_reg_bk(swi2c_sda, &sda_bk, 1);
+ reg_p = handle->scl_reg;
+ do {
+ REG16_VAL(reg_p->gpio_en) |= reg_p->bit_mask;
+ REG16_VAL(reg_p->out_en) |= reg_p->bit_mask;
+ REG16_VAL(reg_p->data_out) &= ~reg_p->bit_mask;
+
+ REG16_VAL(reg_p->out_en) &= ~reg_p->bit_mask;
+ if (reg_p->pull_en)
+ REG16_VAL(reg_p->pull_en) &= ~reg_p->pull_en_bit_mask;
+ if (*(volatile unsigned short *)(reg_p->data_in) &
+ reg_p->bit_mask) {
+ if (reg_p == handle->sda_reg)
+ break;
+ reg_p = handle->sda_reg;
+ } else {
+ ret = 1;
+ break;
+ }
+ } while (1);
+ wmt_swi2c_reg_bk(swi2c_scl, &scl_bk, 0);
+ wmt_swi2c_reg_bk(swi2c_sda, &sda_bk, 0);
+ DPRINT("[SWI2C] %s exist\n", (ret) ? "not" : "");
+#endif
+ return ret;
+}
+EXPORT_SYMBOL(wmt_swi2c_check);
diff --git a/drivers/video/wmt/sw_i2c.h b/drivers/video/wmt/sw_i2c.h
new file mode 100644
index 00000000..ab4b8d4c
--- /dev/null
+++ b/drivers/video/wmt/sw_i2c.h
@@ -0,0 +1,61 @@
+/*++
+ * linux/drivers/video/wmt/sw_i2c.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef _SWI2C_H_
+#define _SWI2C_H_
+
+typedef struct {
+ unsigned int bit_mask;
+ unsigned int gpio_en;
+ unsigned int out_en;
+ unsigned int data_in;
+ unsigned int data_out;
+ unsigned int pull_en_bit_mask;
+ unsigned int pull_en;
+} swi2c_reg_t;
+
+typedef struct {
+ swi2c_reg_t *scl_reg;
+ swi2c_reg_t *sda_reg;
+} swi2c_handle_t;
+
+int wmt_swi2c_read(
+ swi2c_handle_t *handle,
+ char addr,
+ char index,
+ char *buf,
+ int cnt
+);
+
+int wmt_swi2c_write(
+ swi2c_handle_t *handle,
+ char addr,
+ char index,
+ char *buf,
+ int cnt
+);
+
+int wmt_swi2c_check(swi2c_handle_t *handle);
+
+#endif
+
diff --git a/drivers/video/wmt/vout-wmt.c b/drivers/video/wmt/vout-wmt.c
new file mode 100644
index 00000000..323f9278
--- /dev/null
+++ b/drivers/video/wmt/vout-wmt.c
@@ -0,0 +1,1523 @@
+/*++
+ * linux/drivers/video/wmt/vout-wmt.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+#include "vpp.h"
+
+#ifndef CFG_LOADER
+static int vo_plug_flag;
+#endif
+int vo_plug_vout;
+int (*vo_plug_func)(int hotplug);
+vout_mode_t dvo_vout_mode;
+vout_mode_t int_vout_mode;
+struct fb_videomode vo_oem_vmode;
+int hdmi_cur_plugin;
+vout_t *vo_poll_vout;
+
+/* GPIO 10 & 11 */
+swi2c_reg_t vo_gpio_scl = {
+ .bit_mask = BIT10,
+ .gpio_en = (__GPIO_BASE + 0x40),
+ .out_en = (__GPIO_BASE + 0x80),
+ .data_in = (__GPIO_BASE + 0x00),
+ .data_out = (__GPIO_BASE + 0xC0),
+ .pull_en = (__GPIO_BASE + 0x480),
+ .pull_en_bit_mask = BIT10,
+};
+
+swi2c_reg_t vo_gpio_sda = {
+ .bit_mask = BIT11,
+ .gpio_en = (__GPIO_BASE + 0x40),
+ .out_en = (__GPIO_BASE + 0x80),
+ .data_in = (__GPIO_BASE + 0x00),
+ .data_out = (__GPIO_BASE + 0xC0),
+ .pull_en = (__GPIO_BASE + 0x480),
+ .pull_en_bit_mask = BIT11,
+};
+
+swi2c_handle_t vo_swi2c_dvi = {
+ .scl_reg = &vo_gpio_scl,
+ .sda_reg = &vo_gpio_sda,
+};
+
+struct vout_init_parm_t {
+ unsigned int virtual_display;
+ unsigned int def_resx;
+ unsigned int def_resy;
+ unsigned int def_fps;
+ unsigned int ub_resx;
+ unsigned int ub_resy;
+};
+
+#define DVI_POLL_TIME_MS 100
+
+extern void hdmi_config_audio(vout_audio_t *info);
+extern vout_dev_t *lcd_get_dev(void);
+
+/*---------------------------------- API ------------------------------------*/
+int vo_i2c_proc(int id, unsigned int addr, unsigned int index,
+ char *pdata, int len)
+{
+ swi2c_handle_t *handle = 0;
+ int ret = 0;
+
+ switch (id) {
+ case 1: /* dvi */
+ if (lcd_get_type()) /* share pin with LVDS */
+ return -1;
+ handle = &vo_swi2c_dvi;
+ break;
+ default:
+ break;
+ }
+
+ if (handle) {
+ if (wmt_swi2c_check(handle))
+ return -1;
+ if (addr & 0x1) { /* read */
+ *pdata = 0xff;
+#ifdef CONFIG_WMT_EDID
+ ret = wmt_swi2c_read(handle, addr & ~0x1,
+ index, pdata, len);
+#else
+ ret = -1;
+#endif
+ } else { /* write */
+ DBG_ERR("not support sw i2c write\n");
+ }
+ }
+ return ret;
+}
+
+#ifndef CONFIG_UBOOT
+static void vo_do_plug(struct work_struct *ptr)
+{
+ vout_t *vo;
+ int plugin;
+
+ if (vo_plug_func == 0)
+ return;
+
+ vo = vout_get_entry(vo_plug_vout);
+ govrh_set_dvo_enable(vo->govr, 1);
+ plugin = vo_plug_func(1);
+ govrh_set_dvo_enable(vo->govr, plugin);
+ vout_change_status(vo, VPP_VOUT_STS_PLUGIN, plugin);
+ vo_plug_flag = 0;
+ DBG_DETAIL("vo_do_plug %d\n", plugin);
+ /* GPIO irq enable */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x300 + VPP_VOINT_NO, 0x80, 7, 1);
+ /* GPIO input mode */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x80, 0x1 << VPP_VOINT_NO,
+ VPP_VOINT_NO, 0x0);
+#ifdef __KERNEL__
+ vpp_netlink_notify_plug(VPP_VOUT_NUM_DVI, plugin);
+#endif
+ return;
+}
+
+DECLARE_DELAYED_WORK(vo_plug_work, vo_do_plug);
+
+static irqreturn_t vo_plug_interrupt_routine
+(
+ int irq, /*!<; // irq id */
+ void *dev_id /*!<; // device id */
+)
+{
+ DBG_DETAIL("Enter\n");
+ if ((vppif_reg8_in(GPIO_BASE_ADDR + 0x360) &
+ (0x1 << VPP_VOINT_NO)) == 0)
+ return IRQ_NONE;
+
+ /* clear int status */
+ vppif_reg8_out(GPIO_BASE_ADDR + 0x360, 0x1 << VPP_VOINT_NO);
+#ifdef __KERNEL__
+ /* if (vo_plug_flag == 0) { */
+ /* GPIO irq disable */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x300 + VPP_VOINT_NO, 0x80, 7, 0);
+ schedule_delayed_work(&vo_plug_work, HZ/5);
+ vo_plug_flag = 1;
+ /* } */
+#else
+ if (vo_plug_func)
+ vo_do_plug(0);
+#endif
+ return IRQ_HANDLED;
+}
+
+#define CONFIG_VO_POLL_WORKQUEUE
+struct timer_list vo_poll_timer;
+#ifdef CONFIG_VO_POLL_WORKQUEUE
+static void vo_do_poll(struct work_struct *ptr)
+{
+ vout_t *vo;
+
+ vo = vo_poll_vout;
+ if (vo) {
+ if (vo->dev)
+ vo->dev->poll();
+ mod_timer(&vo_poll_timer,
+ jiffies + msecs_to_jiffies(vo_poll_timer.data));
+ }
+ return;
+}
+
+DECLARE_DELAYED_WORK(vo_poll_work, vo_do_poll);
+#else
+struct tasklet_struct vo_poll_tasklet;
+static void vo_do_poll_tasklet
+(
+ unsigned long data /*!<; // tasklet input data */
+)
+{
+ vout_t *vo;
+
+ vpp_lock();
+ vo = vo_poll_vout;
+ if (vo) {
+ if (vo->dev)
+ vo->dev->poll();
+ mod_timer(&vo_poll_timer,
+ jiffies + msecs_to_jiffies(vo_poll_timer.data));
+ }
+ vpp_unlock();
+}
+#endif
+
+void vo_do_poll_tmr(int ms)
+{
+#ifdef CONFIG_VO_POLL_WORKQUEUE
+ schedule_delayed_work(&vo_poll_work, msecs_to_jiffies(ms));
+#else
+ tasklet_schedule(&vo_poll_tasklet);
+#endif
+}
+
+static void vo_set_poll(vout_t *vo, int on, int ms)
+{
+ if (on) {
+ vo_poll_vout = vo;
+ if (vo_poll_timer.function) {
+ vo_poll_timer.data = ms / 2;
+ mod_timer(&vo_poll_timer,
+ jiffies + msecs_to_jiffies(vo_poll_timer.data));
+ } else {
+ init_timer(&vo_poll_timer);
+ vo_poll_timer.data = ms / 2;
+ vo_poll_timer.function = (void *) vo_do_poll_tmr;
+ vo_poll_timer.expires = jiffies +
+ msecs_to_jiffies(vo_poll_timer.data);
+ add_timer(&vo_poll_timer);
+ }
+#ifndef CONFIG_VO_POLL_WORKQUEUE
+ tasklet_init(&vo_poll_tasklet, vo_do_poll_tasklet, 0);
+#endif
+ } else {
+ del_timer(&vo_poll_timer);
+#ifndef CONFIG_VO_POLL_WORKQUEUE
+ tasklet_kill(&vo_poll_tasklet);
+#endif
+ vo_poll_vout = 0;
+ }
+}
+#endif
+
+void vout_set_int_type(int type)
+{
+ unsigned char reg;
+
+ reg = vppif_reg8_in(GPIO_BASE_ADDR+0x300 + VPP_VOINT_NO);
+ reg &= ~0x7;
+ switch (type) {
+ case 0: /* low level */
+ case 1: /* high level */
+ case 2: /* falling edge */
+ case 3: /* rising edge */
+ case 4: /* rising edge or falling */
+ reg |= type;
+ break;
+ default:
+ break;
+ }
+ vppif_reg8_out(GPIO_BASE_ADDR + 0x300 + VPP_VOINT_NO, reg);
+}
+EXPORT_SYMBOL(vout_set_int_type);
+
+void vout_set_int_enable(int enable)
+{
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x300 +
+ VPP_VOINT_NO, 0x80, 7, enable); /* GPIO irq enable/disable */
+}
+EXPORT_SYMBOL(vout_set_int_enable);
+
+int vout_get_clr_int(void)
+{
+ if ((vppif_reg8_in(GPIO_BASE_ADDR + 0x360) &
+ (0x1 << VPP_VOINT_NO)) == 0)
+ return 1;
+ /* clear int status */
+ vppif_reg8_out(GPIO_BASE_ADDR + 0x360, 0x1 << VPP_VOINT_NO);
+ return 0;
+}
+EXPORT_SYMBOL(vout_get_clr_int);
+
+static void vo_plug_enable(int enable, void *func, int no)
+{
+ vout_t *vo;
+
+ DBG_DETAIL("%d\n", enable);
+ vo_plug_vout = no;
+ vo = vout_get_entry(no);
+#ifdef CONFIG_WMT_EXT_DEV_PLUG_DISABLE
+ vo_plug_func = 0;
+ govrh_set_dvo_enable(vo->govr, enable);
+#else
+ vo_plug_func = func;
+ if (vo_plug_func == 0)
+ return;
+
+ if (enable) {
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x40, 0x1 << VPP_VOINT_NO,
+ VPP_VOINT_NO, 0x0); /* GPIO disable */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x80, 0x1 << VPP_VOINT_NO,
+ VPP_VOINT_NO, 0x0); /* GPIO input mode */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x480, 0x1 << VPP_VOINT_NO,
+ VPP_VOINT_NO, 0x1); /* GPIO pull enable */
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x4c0, 0x1 << VPP_VOINT_NO,
+ VPP_VOINT_NO, 0x1); /* GPIO pull-up */
+#ifndef CONFIG_UBOOT
+ vo_do_plug(0);
+ if (vpp_request_irq(IRQ_GPIO, vo_plug_interrupt_routine,
+ IRQF_SHARED, "vo plug", (void *) &vo_plug_vout))
+ DBG_ERR("request GPIO ISR fail\n");
+
+ vppif_reg32_write(GPIO_BASE_ADDR + 0x300 + VPP_VOINT_NO,
+ 0x80, 7, 1); /* GPIO irq enable */
+ } else {
+ vpp_free_irq(IRQ_GPIO, (void *) &vo_plug_vout);
+#endif
+ }
+#endif
+}
+
+/*--------------------------------- DVI ------------------------------------*/
+#ifdef WMT_FTBLK_VOUT_DVI
+static int vo_dvi_blank(vout_t *vo, vout_blank_t arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+ if (vo->pre_blank == VOUT_BLANK_POWERDOWN) {
+ if (vo->dev) {
+ vo->dev->init(vo);
+#ifdef __KERNEL__
+ if (vo->dev->poll)
+ vo_set_poll(vo, (vo->dev->poll) ? 1 : 0,
+ DVI_POLL_TIME_MS);
+#endif
+ }
+ }
+#ifdef __KERNEL__
+ if (arg == VOUT_BLANK_POWERDOWN)
+ vo_set_poll(vo, 0, 0);
+
+ /* patch for virtual fb,if HDMI plugin then DVI blank */
+ if (g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ if (vout_chkplug(VPP_VOUT_NUM_HDMI)) {
+ arg = VOUT_BLANK_NORMAL;
+ vout_change_status(vo, VPP_VOUT_STS_BLANK, arg);
+ }
+ }
+#endif
+ if (!lcd_get_dev()) /* enable DVO not contain LCD */
+ govrh_set_dvo_enable(vo->govr,
+ (arg == VOUT_BLANK_UNBLANK) ? 1 : 0);
+ vo->pre_blank = arg;
+ return 0;
+}
+
+static int vo_dvi_config(vout_t *vo, int arg)
+{
+ vout_info_t *vo_info;
+
+ DBG_DETAIL("Enter\n");
+
+ vo_info = (vout_info_t *) arg;
+ govrh_set_dvo_sync_polar(vo->govr,
+ (vo_info->option & VPP_DVO_SYNC_POLAR_HI) ? 0 : 1,
+ (vo_info->option & VPP_DVO_VSYNC_POLAR_HI) ? 0 : 1);
+ return 0;
+}
+
+static int vo_dvi_init(vout_t *vo, int arg)
+{
+ unsigned int clk_delay;
+
+ DBG_DETAIL("(%d)\n", arg);
+
+ lvds_set_enable(0);
+ govrh_set_dvo_color_format(vo->govr, vo->option[0]);
+ /* bit0:0-12 bit,1-24bit */
+ govrh_set_dvo_outdatw(vo->govr, vo->option[1] & BIT0);
+ govrh_IGS_set_mode(vo->govr, 0, (vo->option[1] & 0x600) >> 9,
+ (vo->option[1] & 0x800) >> 11);
+ govrh_IGS_set_RGB_swap(vo->govr, (vo->option[1] & 0x3000) >> 12);
+ clk_delay = (vo->option[1] & BIT0) ?
+ VPP_GOVR_DVO_DELAY_24 : VPP_GOVR_DVO_DELAY_12;
+ govrh_set_dvo_clock_delay(vo->govr, ((clk_delay & BIT14) != 0x0),
+ clk_delay & 0x3FFF);
+ if (vo->dev) {
+ vo->dev->set_mode(&vo->option[0]);
+ vo->dev->set_power_down(VPP_FLAG_DISABLE);
+ if (vo->dev->interrupt)
+ vo_plug_enable(VPP_FLAG_ENABLE,
+ vo->dev->interrupt, vo->num);
+#ifdef __KERNEL__
+ if (vo->dev->poll)
+ vo_set_poll(vo, (vo->dev->poll) ? 1 : 0,
+ DVI_POLL_TIME_MS);
+#endif
+ vout_change_status(vo, VPP_VOUT_STS_PLUGIN,
+ vo->dev->check_plugin(0));
+ }
+ vo->govr->fb_p->set_csc(vo->govr->fb_p->csc_mode);
+ if (!lcd_get_dev()) {
+ govrh_set_dvo_enable(vo->govr,
+ (vo->status & VPP_VOUT_STS_BLANK) ? 0 : 1);
+ }
+ return 0;
+}
+
+static int vo_dvi_uninit(vout_t *vo, int arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+
+ vo_plug_enable(VPP_FLAG_DISABLE, 0, VPP_VOUT_NUM);
+ govrh_set_dvo_enable(vo->govr, VPP_FLAG_DISABLE);
+#ifdef __KERNEL__
+ vo_set_poll(vo, 0, DVI_POLL_TIME_MS);
+#endif
+ return 0;
+}
+
+static int vo_dvi_chkplug(vout_t *vo, int arg)
+{
+ int plugin = 1;
+
+ DBG_MSG("plugin %d\n", plugin);
+ return plugin;
+}
+
+static int vo_dvi_get_edid(vout_t *vo, int arg)
+{
+ char *buf;
+ int i, cnt;
+
+ DBG_DETAIL("Enter\n");
+
+ buf = (char *) arg;
+ memset(&buf[0], 0x0, 128 * EDID_BLOCK_MAX);
+ if (vpp_i2c_read(VPP_DVI_EDID_ID, 0xA0, 0, &buf[0], 128)) {
+ DBG_ERR("read edid\n");
+ return 1;
+ }
+
+ if (edid_checksum(buf, 128)) {
+ DBG_ERR("checksum\n");
+ return 1;
+ }
+
+ cnt = buf[0x7E];
+ if (cnt >= 3)
+ cnt = 3;
+ for (i = 1; i <= cnt; i++) {
+ vpp_i2c_read(VPP_DVI_EDID_ID, 0xA0, 0x80 * i,
+ &buf[128 * i], 128);
+ }
+ return 0;
+}
+
+vout_inf_t vo_dvi_inf = {
+ .mode = VOUT_INF_DVI,
+ .init = vo_dvi_init,
+ .uninit = vo_dvi_uninit,
+ .blank = vo_dvi_blank,
+ .config = vo_dvi_config,
+ .chkplug = vo_dvi_chkplug,
+ .get_edid = vo_dvi_get_edid,
+};
+
+int vo_dvi_initial(void)
+{
+ vout_inf_register(VOUT_INF_DVI, &vo_dvi_inf);
+ return 0;
+}
+module_init(vo_dvi_initial);
+
+#endif /* WMT_FTBLK_VOUT_DVI */
+
+/*---------------------------------- HDMI -----------------------------------*/
+void vo_hdmi_set_clock(int enable)
+{
+ DBG_DETAIL("(%d)\n", enable);
+
+ enable = (enable) ? CLK_ENABLE : CLK_DISABLE;
+ vpp_set_clock_enable(DEV_HDMII2C, enable, 0);
+ vpp_set_clock_enable(DEV_HDMI, enable, 0);
+ vpp_set_clock_enable(DEV_HDCE, enable, 0);
+}
+
+#ifdef WMT_FTBLK_VOUT_HDMI
+#ifdef __KERNEL__
+struct timer_list hdmi_cp_timer;
+static struct timer_list hdmi_plug_timer;
+#endif
+
+void vo_hdmi_cp_set_enable_tmr(int sec)
+{
+ int ms = sec * 1000;
+
+ DBG_MSG("[HDMI] set enable tmr %d sec\n", sec);
+
+ if (sec == 0) {
+ hdmi_set_cp_enable(VPP_FLAG_ENABLE);
+ return ;
+ }
+#ifdef __KERNEL__
+ if (hdmi_cp_timer.function)
+ del_timer(&hdmi_cp_timer);
+ init_timer(&hdmi_cp_timer);
+ hdmi_cp_timer.data = VPP_FLAG_ENABLE;
+ hdmi_cp_timer.function = (void *) hdmi_set_cp_enable;
+ hdmi_cp_timer.expires = jiffies + msecs_to_jiffies(ms);
+ add_timer(&hdmi_cp_timer);
+#else
+ hdmi_set_cp_enable(VPP_FLAG_ENABLE);
+#endif
+}
+
+static int vo_hdmi_blank(vout_t *vo, vout_blank_t arg)
+{
+ int enable;
+
+ DBG_DETAIL("(%d)\n", arg);
+
+ enable = (arg == VOUT_BLANK_UNBLANK) ? 1 : 0;
+ if (g_vpp.hdmi_cp_enable && enable)
+ vo_hdmi_cp_set_enable_tmr(2);
+ else
+ hdmi_set_cp_enable(VPP_FLAG_DISABLE);
+ hdmi_set_enable(enable);
+ hdmi_set_power_down((enable) ? 0 : 1);
+ return 0;
+}
+
+#ifndef CFG_LOADER
+static irqreturn_t vo_hdmi_cp_interrupt
+(
+ int irq, /*!<; // irq id */
+ void *dev_id /*!<; // device id */
+)
+{
+ vout_t *vo;
+
+ DBG_DETAIL("%d\n", irq);
+ vo = vout_get_entry(VPP_VOUT_NUM_HDMI);
+ switch (hdmi_check_cp_int()) {
+ case 1:
+ if (hdmi_cp)
+ hdmi_cp->enable(VPP_FLAG_DISABLE);
+ vo_hdmi_cp_set_enable_tmr(HDMI_CP_TIME);
+ vout_change_status(vo, VPP_VOUT_STS_CONTENT_PROTECT, 0);
+ vpp_netlink_notify_cp(0);
+ break;
+ case 0:
+ vout_change_status(vo, VPP_VOUT_STS_CONTENT_PROTECT, 1);
+ vpp_netlink_notify_cp(1);
+ break;
+ case 2:
+ hdmi_ri_tm_cnt = 3 * 30;
+ break;
+ default:
+ break;
+ }
+ return IRQ_HANDLED;
+}
+
+#ifdef __KERNEL__
+static void vo_hdmi_do_plug(struct work_struct *ptr)
+#else
+static void vo_hdmi_do_plug(void)
+#endif
+{
+ vout_t *vo;
+ int plugin;
+ int option = 0;
+
+ plugin = hdmi_check_plugin(1);
+ vo = vout_get_entry(VPP_VOUT_NUM_HDMI);
+ vout_change_status(vo, VPP_VOUT_STS_PLUGIN, plugin);
+ if (plugin) {
+ option = vout_get_edid_option(VPP_VOUT_NUM_HDMI);
+#ifdef CONFIG_VPP_DEMO
+ option |= (EDID_OPT_HDMI + EDID_OPT_AUDIO);
+#endif
+ hdmi_set_option(option);
+ } else {
+ g_vpp.hdmi_bksv[0] = g_vpp.hdmi_bksv[1] = 0;
+ }
+ vo_hdmi_blank(vo, (vo->status & VPP_VOUT_STS_BLANK) ? 1 : !(plugin));
+ if (!g_vpp.hdmi_certify_flag)
+ hdmi_hotplug_notify(plugin);
+ DBG_MSG("%d\n", plugin);
+ return;
+}
+DECLARE_WORK(vo_hdmi_plug_work, vo_hdmi_do_plug);
+
+static void hdmi_handle_plug(vpp_flag_t enable)
+{
+ schedule_work(&vo_hdmi_plug_work);
+}
+
+static void vo_hdmi_handle_plug_tmr(int ms)
+{
+ static int timer_init;
+
+ if(timer_init == 0) {
+ init_timer(&hdmi_plug_timer);
+ hdmi_plug_timer.data = VPP_FLAG_ENABLE;
+ hdmi_plug_timer.function = (void *) hdmi_handle_plug;
+ timer_init = 1;
+ }
+ hdmi_plug_timer.expires = jiffies + msecs_to_jiffies(ms);
+ mod_timer(&hdmi_plug_timer, hdmi_plug_timer.expires);
+}
+
+static irqreturn_t vo_hdmi_plug_interrupt
+(
+ int irq, /*!<; // irq id */
+ void *dev_id /*!<; // device id */
+)
+{
+ DBG_MSG("vo_hdmi_plug_interrupt %d\n", irq);
+ hdmi_clear_plug_status();
+ if (g_vpp.hdmi_certify_flag)
+ vo_hdmi_do_plug(0);
+ else
+ vo_hdmi_handle_plug_tmr(HDMI_PLUG_DELAY);
+ return IRQ_HANDLED;
+}
+#endif
+
+static int vo_hdmi_init(vout_t *vo, int arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+
+ vo_hdmi_set_clock(1);
+ vout_change_status(vout_get_entry(VPP_VOUT_NUM_HDMI),
+ VPP_VOUT_STS_PLUGIN, hdmi_check_plugin(0));
+ hdmi_enable_plugin(1);
+
+ if (g_vpp.hdmi_disable)
+ return 0;
+#ifndef CONFIG_UBOOT
+ if (vpp_request_irq(VPP_IRQ_HDMI_CP, vo_hdmi_cp_interrupt,
+ SA_INTERRUPT, "hdmi cp", (void *) 0)) {
+ DBG_ERR("*E* request HDMI ISR fail\n");
+ }
+ if (vpp_request_irq(VPP_IRQ_HDMI_HPDH, vo_hdmi_plug_interrupt,
+ SA_INTERRUPT, "hdmi plug", (void *) 0)) {
+ DBG_ERR("*E* request HDMI ISR fail\n");
+ }
+ if (vpp_request_irq(VPP_IRQ_HDMI_HPDL, vo_hdmi_plug_interrupt,
+ SA_INTERRUPT, "hdmi plug", (void *) 0)) {
+ DBG_ERR("*E* request HDMI ISR fail\n");
+ }
+#endif
+ hdmi_set_enable((vo->status & VPP_VOUT_STS_BLANK) ?
+ VPP_FLAG_DISABLE : VPP_FLAG_ENABLE);
+ return 0;
+}
+
+static int vo_hdmi_uninit(vout_t *vo, int arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+ hdmi_enable_plugin(0);
+ hdmi_set_cp_enable(VPP_FLAG_DISABLE);
+ hdmi_set_enable(VPP_FLAG_DISABLE);
+#ifndef CONFIG_UBOOT
+ vpp_free_irq(VPP_IRQ_HDMI_CP, (void *) 0);
+ vpp_free_irq(VPP_IRQ_HDMI_HPDH, (void *) 0);
+ vpp_free_irq(VPP_IRQ_HDMI_HPDL, (void *) 0);
+#endif
+ vo_hdmi_set_clock(0);
+ return 0;
+}
+
+static int vo_hdmi_config(vout_t *vo, int arg)
+{
+ vout_info_t *vo_info;
+ vdo_color_fmt colfmt;
+
+ hdmi_set_enable(0);
+ vo_info = (vout_info_t *) arg;
+
+ DBG_DETAIL("(%dx%d@%d)\n", vo_info->resx, vo_info->resy, vo_info->fps);
+ DPRINT("hdmi config (%dx%d@%d)\n", vo_info->resx, vo_info->resy, vo_info->fps);
+
+ /* 1280x720@60, HDMI pixel clock 74250060 not 74500000 */
+ if ((vo_info->resx == 1280)
+ && (vo_info->resy == 720) && (vo_info->pixclk == 74500000))
+ vo_info->pixclk = 74250060;
+ colfmt = (vo->option[0] == VDO_COL_FMT_YUV422V) ?
+ VDO_COL_FMT_YUV422H : vo->option[0];
+ hdmi_cur_plugin = hdmi_check_plugin(0);
+ hdmi_info.option = (hdmi_cur_plugin) ?
+ vout_get_edid_option(VPP_VOUT_NUM_HDMI) : 0;
+ hdmi_info.outfmt = colfmt;
+ hdmi_info.vic = hdmi_get_vic(vo_info->resx, vo_info->resy,
+ vo_info->fps, (vo_info->option & VPP_OPT_INTERLACE) ? 1 : 0);
+
+ govrh_set_csc_mode(vo->govr, vo->govr->fb_p->csc_mode);
+ hdmi_set_sync_low_active((vo_info->option & VPP_DVO_SYNC_POLAR_HI) ?
+ 0 : 1, (vo_info->option & VPP_DVO_VSYNC_POLAR_HI) ? 0 : 1);
+ hdmi_config(&hdmi_info);
+#ifdef __KERNEL__
+ mdelay(200); /* patch for VIZIO change resolution issue */
+#endif
+ hdmi_cur_plugin = hdmi_check_plugin(0);
+ vo_hdmi_blank(vo, (vo->status & VPP_VOUT_STS_BLANK) ?
+ 1 : !(hdmi_cur_plugin));
+ return 0;
+}
+
+static int vo_hdmi_chkplug(vout_t *vo, int arg)
+{
+ int plugin;
+
+ if (g_vpp.hdmi_disable)
+ return 0;
+ plugin = hdmi_get_plugin();
+ DBG_DETAIL("%d\n", plugin);
+ return plugin;
+}
+
+static int vo_hdmi_get_edid(vout_t *vo, int arg)
+{
+ char *buf;
+#ifdef CONFIG_WMT_EDID
+ int i, cnt;
+#endif
+ DBG_DETAIL("Enter\n");
+ buf = (char *) arg;
+#ifdef CONFIG_WMT_EDID
+ memset(&buf[0], 0x0, 128*EDID_BLOCK_MAX);
+ if (!hdmi_get_plugin())
+ return 1;
+
+ if (hdmi_DDC_read(0xA0, 0x0, &buf[0], 128)) {
+ DBG_ERR("read edid\n");
+ return 1;
+ }
+
+ if (edid_checksum(buf, 128)) {
+ DBG_ERR("hdmi checksum\n");
+/* g_vpp.dbg_hdmi_ddc_crc_err++; */
+ return 1;
+ }
+
+ cnt = buf[0x7E];
+ if (cnt >= 3)
+ cnt = 3;
+ for (i = 1; i <= cnt; i++)
+ hdmi_DDC_read(0xA0, 0x80 * i, &buf[128 * i], 128);
+#endif
+ return 0;
+}
+
+vout_inf_t vo_hdmi_inf = {
+ .mode = VOUT_INF_HDMI,
+ .init = vo_hdmi_init,
+ .uninit = vo_hdmi_uninit,
+ .blank = vo_hdmi_blank,
+ .config = vo_hdmi_config,
+ .chkplug = vo_hdmi_chkplug,
+ .get_edid = vo_hdmi_get_edid,
+};
+
+int vo_hdmi_initial(void)
+{
+ vout_inf_register(VOUT_INF_HDMI, &vo_hdmi_inf);
+ return 0;
+}
+module_init(vo_hdmi_initial);
+
+#endif /* WMT_FTBLK_VOUT_HDMI */
+
+/*--------------------------------- LVDS ------------------------------------*/
+#ifdef WMT_FTBLK_VOUT_LVDS
+int vo_lvds_init_flag;
+static int vo_lvds_blank(vout_t *vo, vout_blank_t arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+ if (arg == VOUT_BLANK_POWERDOWN) {
+ vppif_reg32_write(LVDS_TRE_EN, 0);
+ } else { /* avoid suspend signal not clear */
+ lvds_set_enable((arg == VOUT_BLANK_UNBLANK) ? 1 : 0);
+ }
+
+ if (vo_lvds_init_flag)
+ lvds_set_power_down(arg);
+ return 0;
+}
+
+static int vo_lvds_config(vout_t *vo, int arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+ lvds_set_power_down(VPP_FLAG_DISABLE);
+ vo_lvds_init_flag = 1;
+ return 0;
+}
+
+static int vo_lvds_init(vout_t *vo, int arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+
+ vpp_set_clock_enable(DEV_LVDS, 1, 0);
+ if (vo->dev)
+ vo->dev->set_mode(&vo->option[0]);
+ govrh_set_dvo_enable(p_govrh2, 0);
+ govrh_set_csc_mode(vo->govr, vo->govr->fb_p->csc_mode);
+ lvds_set_enable((vo->status & VPP_VOUT_STS_BLANK) ?
+ VPP_FLAG_DISABLE : VPP_FLAG_ENABLE);
+ return 0;
+}
+
+static int vo_lvds_uninit(vout_t *vo, int arg)
+{
+ DBG_DETAIL("(%d)\n", arg);
+ lvds_set_enable(VPP_FLAG_DISABLE);
+ if (vo->dev)
+ vo->dev->set_mode(0);
+ lvds_set_power_down(VPP_FLAG_ENABLE);
+ vpp_set_clock_enable(DEV_LVDS, 0, 0);
+ vo_lvds_init_flag = 0;
+ return 0;
+}
+
+static int vo_lvds_chkplug(vout_t *vo, int arg)
+{
+ DBG_DETAIL("\n");
+#if 0
+ vo = vout_get_info(VOUT_LVDS);
+ if (vo->dev)
+ return vo->dev->check_plugin(0);
+#endif
+ return 1;
+}
+
+vout_inf_t vo_lvds_inf = {
+ .mode = VOUT_INF_LVDS,
+ .capability = VOUT_INF_CAP_FIX_PLUG,
+ .init = vo_lvds_init,
+ .uninit = vo_lvds_uninit,
+ .blank = vo_lvds_blank,
+ .config = vo_lvds_config,
+ .chkplug = vo_lvds_chkplug,
+#ifdef WMT_FTBLK_VOUT_HDMI
+ .get_edid = vo_hdmi_get_edid,
+#endif
+};
+
+int vo_lvds_initial(void)
+{
+ vout_inf_register(VOUT_INF_LVDS, &vo_lvds_inf);
+ return 0;
+}
+module_init(vo_lvds_initial);
+
+#endif /* WMT_FTBLK_VOUT_LVDS */
+/*---------------------------------- API ------------------------------------*/
+#ifndef CFG_LOADER
+int vout_set_audio(vout_audio_t *arg)
+{
+ vout_t *vout;
+ int ret = 0;
+
+#if 0
+ vout = vout_get_info(VPP_VOUT_DVO2HDMI);
+ if (vout && (vout->status & VPP_VOUT_STS_PLUGIN)) {
+ if (vout->dev->set_audio)
+ vout->dev->set_audio(arg);
+ }
+#endif
+
+#ifdef WMT_FTBLK_VOUT_HDMI
+ vout = vout_get_entry(VPP_VOUT_NUM_HDMI);
+ if (vout) {
+ hdmi_config_audio(arg);
+ ret = 1;
+ }
+#endif
+ return ret;
+}
+#endif
+
+/* 3445 port1 : DVI/SDD, port2 : VGA/SDA, port3 : HDMI/LVDS */
+/* 3481 port1 : HDMI/LVDS, port2 : DVI */
+/* 3498 port1 : HDMI, port2 : DVI/LVDS */
+vout_t vout_entry_0 = {
+ .fix_cap = BIT(VOUT_INF_HDMI),
+ .option[0] = VDO_COL_FMT_ARGB,
+ .option[1] = VPP_DATAWIDHT_24,
+ .option[2] = 0,
+};
+
+vout_t vout_entry_1 = {
+ .fix_cap = BIT(VOUT_INF_DVI) + BIT(VOUT_INF_LVDS) +
+ VOUT_CAP_EXT_DEV + 0x100, /* i2c bus 1,ext dev */
+ .option[0] = VDO_COL_FMT_ARGB,
+ .option[1] = VPP_DATAWIDHT_24,
+ .option[2] = 0,
+};
+
+void vout_init_param(struct vout_init_parm_t *init_parm)
+{
+ char buf[100];
+ int varlen = 100;
+ unsigned int parm[10];
+
+ memset(init_parm, 0, sizeof(struct vout_init_parm_t));
+
+ /* register vout & set default */
+ vout_register(0, &vout_entry_0);
+ vout_entry_0.inf = vout_inf_get_entry(VOUT_INF_HDMI);
+ vout_register(1, &vout_entry_1);
+ vout_entry_1.inf = vout_inf_get_entry(VOUT_INF_DVI);
+
+ /* [uboot parameter] up bound : resx:resy */
+ if (wmt_getsyspara("wmt.display.upbound", buf, &varlen) == 0) {
+ vpp_parse_param(buf, parm, 2, 0);
+ init_parm->ub_resx = parm[0];
+ init_parm->ub_resy = parm[1];
+ MSG("up bound(%d,%d)\n", init_parm->ub_resx,
+ init_parm->ub_resy);
+ }
+
+ /* [uboot parameter] default resolution : resx:resy:fps */
+ init_parm->def_resx = VOUT_INFO_DEFAULT_RESX;
+ init_parm->def_resy = VOUT_INFO_DEFAULT_RESY;
+ init_parm->def_fps = VOUT_INFO_DEFAULT_FPS;
+ if (wmt_getsyspara("wmt.display.default.res", buf, &varlen) == 0) {
+ vpp_parse_param(buf, parm, 3, 0);
+ init_parm->def_resx = parm[0];
+ init_parm->def_resy = parm[1];
+ init_parm->def_fps = parm[2];
+ MSG("default res(%d,%d,%d)\n", init_parm->def_resx,
+ init_parm->def_resy, init_parm->def_fps);
+ } else if (wmt_getsyspara("wmt.display.param", buf, &varlen) == 0) {
+ vpp_parse_param(buf, parm, 6, 0);
+ init_parm->def_resx = parm[3];
+ init_parm->def_resy = parm[4];
+ init_parm->def_fps = parm[5];
+ MSG("param2 res(%d,%d,%d)\n", init_parm->def_resx,
+ init_parm->def_resy, init_parm->def_fps);
+ }
+}
+
+const char *vout_sys_parm_str[] = {"wmt.display.param", "wmt.display.param2"};
+int vout_check_display_info(struct vout_init_parm_t *init_parm)
+{
+ char buf[100];
+ const char *display_param[] = { "2:0:24:1024:600:60", "4:0:24:1024:600:60" };
+ int varlen = 100;
+ unsigned int parm[10];
+ vout_t *vo = 0;
+ int ret = 1;
+ int i;
+ int info_no;
+
+ DBG_DETAIL("Enter\n");
+
+ /* [uboot parameter] display param : type:op1:op2:resx:resy:fps */
+ for (i = 0; i < 2; i++) {
+ strncpy(buf, display_param[i], 19);
+ DBG_MSG("display param %d : %s\n", i, buf);
+ vpp_parse_param(buf, (unsigned int *)parm, 6, 0);
+ MSG("boot parm vo %d opt %d,%d, %dx%d@%d\n", parm[0],
+ parm[1], parm[2], parm[3], parm[4], parm[5]);
+ vo = vout_get_entry_adapter(parm[0]);
+ if (vo == 0) {
+ vout_t *vo_boot;
+
+ if (parm[0] != VOUT_BOOT) {
+ ret = 1;
+ DBG_ERR("uboot param invalid\n");
+ break;
+ }
+
+ g_vpp.virtual_display_mode = parm[1];
+ if (parm[1] == 1) {
+ init_parm->def_resx = parm[3];
+ init_parm->def_resy = parm[4];
+ init_parm->def_fps = parm[5];
+ parm[3] = 1920;
+ parm[4] = 1080;
+ parm[5] = 60;
+ } else if (parm[1] == 2) {
+ init_parm->def_resx = parm[3];
+ init_parm->def_resy = parm[4];
+ init_parm->def_fps = parm[5];
+ g_vpp.dual_display = 0;
+ break;
+ }
+
+ MSG("[VOUT] virtual display\n");
+ init_parm->virtual_display = 1;
+ g_vpp.virtual_display = 1;
+ g_vpp.fb0_bitblit = 1;
+ vo_boot = kmalloc(sizeof(vout_t), GFP_KERNEL);
+ if (vo_boot == 0) {
+ ret = 1;
+ break;
+ }
+ vo_boot->resx = parm[3];
+ vo_boot->resy = parm[4];
+ vo_boot->pixclk = parm[5];
+ vo_boot->inf = 0;
+ vo_boot->num = VPP_VOUT_INFO_NUM;
+ vo_boot->govr = 0;
+ info_no = vout_info_add_entry(vo_boot);
+ kfree(vo_boot);
+ } else {
+ vo->inf = vout_get_inf_entry_adapter(parm[0]);
+ vo->option[0] = parm[1];
+ vo->option[1] = parm[2];
+ vo->resx = parm[3];
+ vo->resy = parm[4];
+ vo->pixclk = parm[5];
+ vout_change_status(vo, VPP_VOUT_STS_BLANK,
+ (parm[2] & VOUT_OPT_BLANK) ?
+ VOUT_BLANK_NORMAL : VOUT_BLANK_UNBLANK);
+ info_no = vout_info_add_entry(vo);
+ }
+
+ switch (parm[0]) {
+ case VOUT_LVDS:
+ {
+ struct fb_videomode *vmode;
+ vout_info_t *info;
+
+ info = vout_info_get_entry(info_no);
+ vmode = 0;
+ if ((parm[1] == 0) && (parm[3] == 0) &&
+ (parm[4] == 0)) { /* auto detect */
+ if (vout_get_edid_option(vo->num)) {
+ vmode = &vo->edid_info.detail_timing[0];
+ if (vmode->pixclock == 0) {
+ vmode = 0;
+ DBG_ERR("LVDS timing\n");
+ }
+ }
+
+ if (vo->inf->get_edid(vo,
+ (int)vo->edid) == 0) {
+ if (edid_parse(vo->edid,
+ &vo->edid_info) == 0)
+ DBG_ERR("LVDS edid parse\n");
+ } else {
+ DBG_ERR("LVDS edid read\n");
+ }
+ }
+
+ if (vmode == 0) { /* use internal timing */
+ lcd_parm_t *p = 0;
+
+ if (parm[1]) {
+ p = lcd_get_parm(parm[1], parm[2]);
+ if (p)
+ lcd_set_lvds_id(parm[1]);
+ }
+
+ if (p == 0)
+ p = lcd_get_oem_parm(parm[3], parm[4]);
+ vmode = &p->vmode;
+ }
+ memcpy(&vo_oem_vmode, vmode,
+ sizeof(struct fb_videomode));
+ vo->option[2] = vmode->vmode;
+ info->resx = vmode->xres;
+ info->resy = vmode->yres;
+ info->fps = vmode->refresh;
+ vout_info_set_fixed_timing(info_no, &vo_oem_vmode);
+ lcd_set_type(1);
+ }
+ case VOUT_LCD:
+ {
+ vout_dev_t *dev;
+
+ lcd_set_parm(parm[1], parm[2] & 0xFF);
+ dev = lcd_get_dev();
+ vo->ext_dev = dev;
+ vo->dev = dev;
+ dev->vout = vo;
+ vo->option[0] = VDO_COL_FMT_ARGB;
+ vo->option[1] &= ~0xFF;
+ vo->option[1] |= VPP_DATAWIDHT_24;
+ }
+ break;
+ case VOUT_DVI:
+ case VOUT_HDMI:
+ vout_info[info_no].option = (parm[2] & 0x2) ?
+ VPP_OPT_INTERLACE : 0;
+ break;
+ default:
+ break;
+ }
+ ret = 0;
+ }
+
+ if (g_vpp.virtual_display)
+ ret = 1;
+
+ /* add vout entry to info */
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo) {
+ if (vo->resx == 0) {
+ vo->resx = init_parm->def_resx;
+ vo->resy = init_parm->def_resy;
+ vo->pixclk = init_parm->def_fps;
+ }
+ vout_info_add_entry(vo);
+ }
+ }
+ DBG_DETAIL("Leave\n");
+ return ret;
+}
+
+void vout_check_ext_device(void)
+{
+ char buf[100];
+ int varlen = 100;
+ vout_t *vo = 0;
+ vout_dev_t *dev = 0;
+ int i;
+ vout_info_t *info;
+
+ DBG_DETAIL("Enter\n");
+ vpp_i2c_init(1, 0xA0);
+
+ /* [uboot parameter] reg operation : addr op val */
+ if (wmt_getsyspara("wmt.display.regop", buf, &varlen) == 0) {
+ unsigned int addr;
+ unsigned int val;
+ char op;
+ char *p, *endp;
+
+ p = buf;
+ while (1) {
+ addr = simple_strtoul(p, &endp, 16);
+ if (*endp == '\0')
+ break;
+
+ op = *endp;
+ if (endp[1] == '~') {
+ val = simple_strtoul(endp + 2, &endp, 16);
+ val = ~val;
+ } else {
+ val = simple_strtoul(endp + 1, &endp, 16);
+ }
+
+ DBG_DETAIL(" reg op: 0x%X %c 0x%X\n", addr, op, val);
+ switch (op) {
+ case '|':
+ REG32_VAL(addr) |= val;
+ break;
+ case '=':
+ REG32_VAL(addr) = val;
+ break;
+ case '&':
+ REG32_VAL(addr) &= val;
+ break;
+ default:
+ DBG_ERR("Error, Unknown operator %c\n", op);
+ }
+
+ if (*endp == '\0')
+ break;
+ p = endp + 1;
+ }
+ }
+
+ /* [uboot parameter] dvi device : name:i2c id */
+ vo = vout_get_entry(VPP_VOUT_NUM_DVI);
+ info = vout_get_info_entry(VPP_VOUT_NUM_DVI);
+ if (lcd_get_dev()) {
+ vo->dev->init(vo);
+ vout_info_set_fixed_timing(info->num, &p_lcd->vmode);
+ } else if (vo->ext_dev == 0) {
+ if (wmt_getsyspara("wmt.display.dvi.dev", buf, &varlen) == 0) {
+ unsigned int param[1];
+ char *p;
+
+ p = strchr(buf, ':');
+ *p = 0;
+ vpp_parse_param(p + 1, param, 1, 0x1);
+ vpp_i2c_release();
+ vpp_i2c_init(param[0], 0xA0);
+ MSG("dvi dev %s : %x\n", buf, param[0]);
+ do {
+ dev = vout_get_device(dev);
+ if (dev == 0)
+ break;
+ if (strcmp(buf, dev->name) == 0) {
+ vo->ext_dev = dev;
+ vo->dev = dev;
+ dev->vout = vo;
+
+ /* probe & init external device */
+ if (dev->init(vo)) {
+ DBG_ERR("%s not exist\n",
+ dev->name);
+ vo->dev = vo->ext_dev = 0;
+ break;
+ } else {
+ MSG("[VOUT] dvi dev %s\n", buf);
+ }
+
+ /* if LCD then set fixed timing */
+ if (vo->dev == lcd_get_dev())
+ vout_info_set_fixed_timing(
+ info->num, &p_lcd->vmode);
+ }
+ } while (1);
+ }
+ }
+
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo == 0)
+ continue;
+
+ if ((vo->fix_cap & VOUT_CAP_EXT_DEV) && !(vo->ext_dev)) {
+ dev = 0;
+ do {
+ dev = vout_get_device(dev);
+ if (dev == 0)
+ break;
+ if (vo->fix_cap & BIT(dev->mode)) {
+ vo->inf = vout_inf_get_entry(dev->mode);
+ if (dev->init(vo) == 0) {
+ vo->ext_dev = dev;
+ vo->dev = dev;
+ dev->vout = vo;
+ break;
+ }
+ }
+ } while (1);
+ }
+ DBG_MSG("vout %d ext dev : %s\n", i,
+ (vo->dev) ? vo->dev->name : "NO");
+ }
+
+ /* [uboot parameter] oem timing :
+ pixclk:option:hsync:hbp:hpixel:hfp:vsync:vbp:vpixel:vfp */
+ if (wmt_getsyspara("wmt.display.tmr", buf, &varlen) == 0) {
+ unsigned int param[10];
+ struct fb_videomode *p;
+ int xres, yres;
+
+ p = &vo_oem_vmode;
+ DBG_MSG("tmr %s\n", buf);
+ vpp_parse_param(buf, param, 10, 0);
+ p->pixclock = param[0];
+ p->vmode = param[1];
+ p->hsync_len = param[2];
+ p->left_margin = param[3];
+ p->xres = param[4];
+ p->right_margin = param[5];
+ p->vsync_len = param[6];
+ p->upper_margin = param[7];
+ p->yres = param[8];
+ p->lower_margin = param[9];
+ p->pixclock *= 1000;
+ xres = p->hsync_len + p->left_margin + p->xres +
+ p->right_margin;
+ yres = p->vsync_len + p->upper_margin + p->yres +
+ p->lower_margin;
+ p->refresh = vpp_calc_refresh(p->pixclock, xres, yres);
+ if (p->refresh == 59)
+ p->refresh = 60;
+ DBG_MSG("tmr pixclk %d,option 0x%x\n",
+ p->pixclock, p->vmode);
+ DBG_MSG("H sync %d,bp %d,pixel %d,fp %d\n", p->hsync_len,
+ p->left_margin, p->xres, p->right_margin);
+ DBG_MSG("V sync %d,bp %d,pixel %d,fp %d\n", p->vsync_len,
+ p->upper_margin, p->yres, p->lower_margin);
+ p->pixclock = KHZ2PICOS(p->pixclock / 1000);
+ vout_info_set_fixed_timing(0, &vo_oem_vmode);
+ vout_info[0].resx = p->xres;
+ vout_info[0].resy = p->yres;
+ vout_info[0].fps = p->refresh;
+ }
+ DBG_DETAIL("Leave\n");
+}
+
+void vout_check_monitor_resolution(struct vout_init_parm_t *init_parm)
+{
+ vout_info_t *p;
+ vout_t *vo;
+ int i;
+
+ DBG_DETAIL("Check monitor support\n");
+
+ if (g_vpp.virtual_display) {
+ p = &vout_info[0];
+ if ((p->resx == 0) || (p->resy == 0)) {
+ struct fb_videomode vmode;
+ int edid_option;
+
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if (vout_chkplug(i) == 0) {
+ DBG_MSG("no plugin %d\n", i);
+ continue;
+ }
+
+ edid_option = vout_get_edid_option(i);
+ if ((edid_option & EDID_OPT_VALID) == 0) {
+ DBG_MSG("no EDID %d\n", i);
+ continue;
+ }
+ break;
+ }
+ vmode.xres = (i == VPP_VOUT_NUM) ? 1280 : 1920;
+ vmode.yres = (i == VPP_VOUT_NUM) ? 720 : 1080;
+ vmode.refresh = 60;
+ vmode.vmode = 0;
+ vout_find_match_mode(1, &vmode, 1);
+ init_parm->def_resx = vmode.xres;
+ init_parm->def_resy = vmode.yres;
+ init_parm->def_fps = vmode.refresh;
+
+ for (i = 0; i < VPP_VOUT_INFO_NUM; i++) {
+ p = &vout_info[i];
+ p->resx = init_parm->def_resx;
+ p->resy = init_parm->def_resy;
+ p->fps = init_parm->def_fps;
+ }
+ DBG_MSG("virtual display %dx%d@%d\n",
+ vmode.xres, vmode.yres, vmode.refresh);
+ }
+ }
+
+ for (i = 0; i < VPP_VOUT_INFO_NUM; i++) {
+ int support;
+ int vo_num;
+
+ p = &vout_info[i];
+
+ DBG_MSG("info %d (%d,%d,%d)\n", i, p->resx, p->resy, p->fps);
+
+ if (p->vo_mask == 0)
+ break;
+ if (p->govr == 0)
+ continue;
+ if (p->fixed_vmode)
+ continue;
+ if (init_parm->ub_resx)
+ p->resx = init_parm->ub_resx;
+ if (init_parm->ub_resy)
+ p->resy = init_parm->ub_resy;
+ if (!p->resx)
+ p->resx = init_parm->def_resx;
+ if (!p->resy)
+ p->resy = init_parm->def_resy;
+ if (!p->fps)
+ p->fps = init_parm->def_fps;
+
+ DBG_DETAIL("info %d (%d,%d,%d)\n", i, p->resx, p->resy, p->fps);
+
+ support = 0;
+ for (vo_num = 0; vo_num < VPP_VOUT_NUM; vo_num++) {
+ int edid_option;
+
+ if ((p->vo_mask & (0x1 << vo_num)) == 0)
+ continue;
+ vo = vout_get_entry(vo_num);
+ if (vo == 0)
+ continue;
+#ifdef CONFIG_WMT_EDID
+ if (vout_chkplug(vo_num) == 0) {
+ DBG_MSG("no plugin %d\n", vo_num);
+ continue;
+ }
+
+ vout_change_status(vo, VPP_VOUT_STS_BLANK, 0);
+ edid_option = vout_get_edid_option(vo_num);
+ if (edid_option & EDID_OPT_VALID) {
+ struct fb_videomode *vmode;
+
+ if (edid_find_support(&vo->edid_info, p->resx,
+ p->resy, p->fps, &vmode)) {
+ support = 1;
+ break;
+ } else {
+ if (vout_find_edid_support_mode(
+ &vo->edid_info,
+ (unsigned int *)&p->resx,
+ (unsigned int *)&p->resy,
+ (unsigned int *)&p->fps,
+ (edid_option & EDID_OPT_16_9) ?
+ 1 : 0)) {
+ support = 1;
+ break;
+ }
+ }
+ }
+#else
+ break;
+#endif
+ }
+
+ if (support == 0) {
+ if (vout_chkplug(VPP_VOUT_NUM_HDMI)) {
+ init_parm->def_resx = 1280;
+ init_parm->def_resy = 720;
+ init_parm->def_fps = 60;
+ } else {
+ char buf[40];
+ int varlen = 40;
+
+ if (wmt_getsyspara("wmt.display.tvformat",
+ buf, &varlen) == 0) {
+ if (strnicmp(buf, "PAL", 3) == 0) {
+ init_parm->def_resx = 720;
+ init_parm->def_resy = 576;
+ init_parm->def_fps = 50;
+ } else if (strnicmp(buf,"NTSC",4) == 0) {
+ init_parm->def_resx = 720;
+ init_parm->def_resy = 480;
+ init_parm->def_fps = 60;
+ }
+ } else {
+ init_parm->def_resx = 1024;
+ init_parm->def_resy = 768;
+ init_parm->def_fps = 60;
+ }
+ }
+ p->resx = init_parm->def_resx;
+ p->resy = init_parm->def_resy;
+ p->fps = init_parm->def_fps;
+ DBG_MSG("use default(%dx%d@%d)\n", p->resx,
+ p->resy, p->fps);
+ }
+ DBG_MSG("fb%d(%dx%d@%d)\n", i, p->resx, p->resy, p->fps);
+ }
+ DBG_DETAIL("Leave\n");
+}
+
+int vout_init(void)
+{
+ vout_t *vo;
+ int flag;
+ int i;
+ struct vout_init_parm_t init_parm;
+
+ DBG_DETAIL("Enter\n");
+
+ vout_init_param(&init_parm);
+
+ /* check vout info */
+ memset(vout_info, 0, sizeof(vout_info_t) * VPP_VOUT_INFO_NUM);
+ flag = vout_check_display_info(&init_parm);
+
+ /* probe external device */
+ vout_check_ext_device();
+
+ /* check plug & EDID for resolution */
+ if (flag)
+ vout_check_monitor_resolution(&init_parm);
+
+ if ((init_parm.virtual_display) || (g_vpp.dual_display == 0)) {
+ int plugin;
+
+ plugin = vout_chkplug(VPP_VOUT_NUM_HDMI);
+ vout_change_status(vout_get_entry(VPP_VOUT_NUM_HDMI),
+ VPP_VOUT_STS_BLANK, (plugin) ? 0 : 1);
+ vout_change_status(vout_get_entry(VPP_VOUT_NUM_DVI),
+ VPP_VOUT_STS_BLANK, (plugin) ? 1 : 0);
+ }
+
+ DBG_DETAIL("Set mode\n");
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo && (vo->inf)) {
+ vout_inf_mode_t mode;
+
+ mode = vo->inf->mode;
+ vo->inf = 0; /* force interface init */
+ vout_set_mode(i, mode);
+ DBG_DETAIL("vout %d : inf %s, ext dev %s,status 0x%x\n",
+ i,
+ (vo->inf) ? vout_inf_str[vo->inf->mode] : "No",
+ (vo->dev) ? vo->dev->name : "No", vo->status);
+ }
+ }
+
+#ifndef CONFIG_UBOOT
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo)
+ vout_set_blank((0x1 << i),
+ (vo->status & VPP_VOUT_STS_BLANK) ? 1 : 0);
+ }
+#endif
+
+ /* show vout info */
+ for (i = 0; i < VPP_VOUT_INFO_NUM; i++) {
+ vout_info_t *info;
+
+ info = &vout_info[i];
+ if (info->vo_mask == 0)
+ break;
+ if (i == 0) {
+ if (info->govr) {
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ p_cursor->mmio = info->govr->mmio;
+#endif
+ }
+ }
+ MSG("[VOUT] info %d,mask 0x%x,%s,%dx%d@%d\n", i, info->vo_mask,
+ (info->govr) ? vpp_mod_str[info->govr_mod] : "VIR",
+ info->resx, info->resy, info->fps);
+ }
+ DBG_DETAIL("Leave\n");
+ return 0;
+}
+
+int vout_exit(void)
+{
+ return 0;
+}
diff --git a/drivers/video/wmt/vout.c b/drivers/video/wmt/vout.c
new file mode 100644
index 00000000..683fc97e
--- /dev/null
+++ b/drivers/video/wmt/vout.c
@@ -0,0 +1,936 @@
+/*++
+ * linux/drivers/video/wmt/vout.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2012 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define VOUT_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "vout.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+/* #define VO_XXXX xxxx *//*Example*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define VO_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx vo_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in vout.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int vo_xxx; *//*Example*/
+vout_t *vout_array[VPP_VOUT_NUM];
+vout_inf_t *vout_inf_array[VOUT_INF_MODE_MAX];
+vout_dev_t *vout_dev_list;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void vo_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+/*----------------------- vout API --------------------------------------*/
+void vout_print_entry(vout_t *vo)
+{
+ MSG(" ===== vout entry =====\n");
+ MSG("0x%x\n", (int)vo);
+ if (vo == 0)
+ return;
+
+ MSG("num %d,fix 0x%x,var 0x%x\n", vo->num, vo->fix_cap, vo->var_cap);
+ MSG("inf 0x%x,dev 0x%x,ext_dev 0x%x\n", (int)vo->inf,
+ (int)vo->dev, (int)vo->ext_dev);
+ MSG("resx %d,resy %d,pixclk %d\n", vo->resx, vo->resy, vo->pixclk);
+ MSG("status 0x%x,option %d,%d,%d\n", vo->status,
+ vo->option[0], vo->option[1], vo->option[2]);
+
+ if (vo->inf) {
+ MSG(" ===== inf entry =====\n");
+ MSG("mode %d, %s\n",
+ vo->inf->mode, vout_inf_str[vo->inf->mode]);
+ }
+
+ if (vo->dev) {
+ MSG(" ===== dev entry =====\n");
+ MSG("name %s,inf %d,%s\n", vo->dev->name,
+ vo->dev->mode, vout_inf_str[vo->dev->mode]);
+ MSG("vout 0x%x,capability 0x%x\n",
+ (int)vo->dev->vout, vo->dev->capability);
+ }
+}
+
+void vout_register(int no, vout_t *vo)
+{
+ if (no >= VPP_VOUT_NUM)
+ return;
+
+ vo->num = no;
+ vo->govr = (void *) p_govrh;
+ vo->status = VPP_VOUT_STS_REGISTER + VPP_VOUT_STS_BLANK;
+ vout_array[no] = vo;
+}
+
+vout_t *vout_get_entry(int no)
+{
+ if (no >= VPP_VOUT_NUM)
+ return 0;
+ return vout_array[no];
+}
+
+vout_info_t *vout_get_info_entry(int no)
+{
+ int i;
+
+ if (no >= VPP_VOUT_NUM)
+ return 0;
+
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if (vout_info[i].vo_mask & (0x1 << no))
+ return &vout_info[i];
+ }
+ return 0;
+}
+
+void vout_change_status(vout_t *vo, int mask, int sts)
+{
+ DBG_DETAIL("(0x%x,%d)\n", mask, sts);
+ if (sts)
+ vo->status |= mask;
+ else
+ vo->status &= ~mask;
+
+ switch (mask) {
+ case VPP_VOUT_STS_PLUGIN:
+ if (sts == 0) {
+ vo->status &= ~(VPP_VOUT_STS_EDID +
+ VPP_VOUT_STS_CONTENT_PROTECT);
+ vo->edid_info.option = 0;
+ vpp_netlink_notify_cp(0);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+int vout_query_inf_support(int no, vout_inf_mode_t mode)
+{
+ vout_t *vo;
+
+ if (no >= VPP_VOUT_NUM)
+ return 0;
+
+ if (mode >= VOUT_INF_MODE_MAX)
+ return 0;
+
+ vo = vout_get_entry(no);
+ return (vo->fix_cap & BIT(mode)) ? 1 : 0;
+}
+
+/*----------------------- vout interface API --------------------------------*/
+int vout_inf_register(vout_inf_mode_t mode, vout_inf_t *inf)
+{
+ if (mode >= VOUT_INF_MODE_MAX) {
+ DBG_ERR("vout interface mode invalid %d\n", mode);
+ return -1;
+ }
+
+ if (vout_inf_array[mode])
+ DBG_ERR("vout interface register again %d\n", mode);
+
+ vout_inf_array[mode] = inf;
+ return 0;
+} /* End of vout_register */
+
+vout_inf_t *vout_inf_get_entry(vout_inf_mode_t mode)
+{
+ if (mode >= VOUT_INF_MODE_MAX) {
+ DBG_ERR("vout interface mode invalid %d\n", mode);
+ return 0;
+ }
+ return vout_inf_array[mode];
+}
+
+/*----------------------- vout device API -----------------------------------*/
+int vout_device_register(vout_dev_t *ops)
+{
+ vout_dev_t *list;
+
+ if (vout_dev_list == 0) {
+ vout_dev_list = ops;
+ list = ops;
+ } else {
+ list = vout_dev_list;
+ while (list->next != 0)
+ list = list->next;
+ list->next = ops;
+ }
+ ops->next = 0;
+ return 0;
+}
+
+vout_dev_t *vout_get_device(vout_dev_t *ops)
+{
+ if (ops == 0)
+ return vout_dev_list;
+ return ops->next;
+}
+
+vout_t *vout_get_entry_adapter(vout_mode_t mode)
+{
+ int no;
+
+ switch (mode) {
+ case VOUT_SD_DIGITAL:
+ case VOUT_DVI:
+ case VOUT_LCD:
+ case VOUT_DVO2HDMI:
+ case VOUT_SD_ANALOG:
+ case VOUT_VGA:
+ no = VPP_VOUT_NUM_DVI;
+ break;
+ case VOUT_HDMI:
+ no = VPP_VOUT_NUM_HDMI;
+ break;
+ case VOUT_LVDS:
+ no = VPP_VOUT_NUM_LVDS;
+ break;
+ default:
+ no = VPP_VOUT_NUM;
+ break;
+ }
+ return vout_get_entry(no);
+}
+
+vout_inf_t *vout_get_inf_entry_adapter(vout_mode_t mode)
+{
+ int no;
+
+ switch (mode) {
+ case VOUT_SD_DIGITAL:
+ case VOUT_SD_ANALOG:
+ case VOUT_DVI:
+ case VOUT_LCD:
+ case VOUT_DVO2HDMI:
+ case VOUT_VGA:
+ no = VOUT_INF_DVI;
+ break;
+ case VOUT_HDMI:
+ no = VOUT_INF_HDMI;
+ break;
+ case VOUT_LVDS:
+ no = VOUT_INF_LVDS;
+ break;
+ default:
+ no = VOUT_INF_MODE_MAX;
+ return 0;
+ }
+ return vout_inf_get_entry(no);
+}
+
+int vout_info_add_entry(vout_t *vo)
+{
+ vout_info_t *p;
+ int i = 0;
+
+ if (vo == 0) /* invalid */
+ return 0;
+
+ if (vo->inf) {
+ switch (vo->inf->mode) {
+ case VOUT_INF_DVI:
+ case VOUT_INF_LVDS:
+ vo->govr = p_govrh2;
+ break;
+ default:
+ vo->govr = p_govrh;
+ break;
+ }
+ }
+
+ if (g_vpp.virtual_display) {
+ if (vo->inf)
+ i = 1;
+ } else {
+ if (vo->inf == 0) /* invalid */
+ return 0;
+
+ if (g_vpp.dual_display) {
+ for (i = 0; i < VPP_VOUT_INFO_NUM; i++) {
+ p = &vout_info[i];
+ if (p->vo_mask == 0) /* no use */
+ break;
+
+ if (p->vo_mask & (0x1 << vo->num)) /* exist */
+ return i;
+
+ if (vo->govr && (p->govr_mod ==
+ ((vpp_mod_base_t *)(vo->govr))->mod))
+ break;
+ }
+ }
+ }
+
+ if (i >= VPP_VOUT_INFO_NUM) {
+ DBG_ERR("full\n");
+ return 0;
+ }
+ p = &vout_info[i];
+ p->num = i;
+ if (p->vo_mask == 0) {
+ p->resx = vo->resx;
+ p->resy = vo->resy;
+ p->resx_virtual = vpp_calc_align(p->resx, 4);
+ p->resy_virtual = p->resy;
+ p->fps = (int) vo->pixclk;
+ p->govr_mod = (vo->govr) ?
+ ((vpp_mod_base_t *)vo->govr)->mod : VPP_MOD_MAX;
+ p->govr = vo->govr;
+ }
+ p->vo_mask |= (0x1 << vo->num);
+ DBG_MSG("info %d,vo mask 0x%x,%s,%dx%d@%d\n", i, p->vo_mask,
+ vpp_mod_str[p->govr_mod], p->resx, p->resy, p->fps);
+ return i;
+}
+
+vout_info_t *vout_info_get_entry(int no)
+{
+ if (g_vpp.dual_display == 0)
+ return &vout_info[0];
+
+ if ((no >= VPP_VOUT_INFO_NUM) || (vout_info[no].vo_mask == 0))
+ no = 0;
+
+ return &vout_info[no];
+}
+
+void vout_info_set_fixed_timing(int no, struct fb_videomode *vmode)
+{
+ vout_info_t *info;
+
+ DBG_MSG("(%d)\n", no);
+
+ info = vout_info_get_entry(no);
+ info->fixed_vmode = vmode;
+}
+
+govrh_mod_t *vout_info_get_govr(int no)
+{
+ vout_info_t *info;
+ govrh_mod_t *govr;
+
+ info = vout_info_get_entry(no);
+ govr = info->govr;
+ if (g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ govr = (hdmi_get_plugin()) ? p_govrh : p_govrh2;
+ }
+ return govr;
+}
+
+int vout_check_ratio_16_9(unsigned int resx, unsigned int resy)
+{
+ int val;
+
+ val = (resx * 10) / resy;
+ return (val >= 15) ? 1 : 0;
+}
+
+struct fb_videomode *vout_get_video_mode(int vout_num,
+ struct fb_videomode *vmode, int option)
+{
+ int i;
+ struct fb_videomode *p, *best = NULL;
+ unsigned int diff = -1, diff_refresh = -1;
+ int d;
+ int resx, resy, fps;
+ unsigned int pixel_clock, diff_pixel_clock = -1;
+ vout_t *vo = 0;
+ char *edid = 0;
+
+ resx = vmode->xres;
+ resy = vmode->yres;
+ fps = vmode->refresh;
+ pixel_clock = vmode->pixclock;
+ DBG_MSG("%d,%dx%d@%d,%d,0x%x\n", vout_num, resx, resy, fps,
+ (int) PICOS2KHZ(vmode->pixclock) * 1000, option);
+
+ /* EDID detail timing */
+ if (option & VOUT_MODE_OPTION_EDID) {
+ unsigned int opt;
+ struct fb_videomode *edid_vmode;
+
+ vo = vout_get_entry(vout_num);
+ if (vo == 0)
+ return 0;
+
+ edid = vout_get_edid(vout_num);
+ if (edid_parse(edid, &vo->edid_info)) {
+ opt = fps | ((option &
+ VOUT_MODE_OPTION_INTERLACE) ?
+ EDID_TMR_INTERLACE : 0);
+ if (edid_find_support(&vo->edid_info,
+ resx, resy, opt, &edid_vmode)) {
+ if (edid_vmode) {
+ DBG_MSG("EDID detail timing\n");
+ return edid_vmode;
+ }
+ }
+ }
+ }
+
+ /* video mode table */
+ for (i = 0; ; i++) {
+ p = (struct fb_videomode *) &vpp_videomode[i];
+ if (p->pixclock == 0)
+ break;
+
+ if (option & VOUT_MODE_OPTION_LESS) {
+ if ((p->xres > resx) || (p->yres > resy))
+ continue;
+ }
+ if (option & VOUT_MODE_OPTION_GREATER) {
+ if ((p->xres < resx) || (p->yres < resy))
+ continue;
+ }
+ if (option & VOUT_MODE_OPTION_INTERLACE) {
+ if (!(p->vmode & FB_VMODE_INTERLACED))
+ continue;
+ }
+ if (option & VOUT_MODE_OPTION_PROGRESS) {
+ if ((p->vmode & FB_VMODE_INTERLACED))
+ continue;
+ }
+ if ((option & VOUT_MODE_OPTION_EDID) &&
+ (edid_parse(edid, &vo->edid_info))) {
+ unsigned int opt;
+ struct fb_videomode *edid_vmode;
+
+ opt = p->refresh | ((option &
+ VOUT_MODE_OPTION_INTERLACE) ?
+ EDID_TMR_INTERLACE : 0);
+ if (edid_find_support(&vo->edid_info,
+ p->xres, p->yres, opt, &edid_vmode)) {
+ if (edid_vmode)
+ p = edid_vmode;
+ } else {
+ continue;
+ }
+ }
+ d = abs(p->xres - resx) + abs(p->yres - resy);
+ d = abs(d);
+ if (diff > d) {
+ diff = d;
+ diff_refresh = abs(p->refresh - fps);
+ diff_pixel_clock = abs(p->pixclock - pixel_clock);
+ best = p;
+ } else if (diff == d) {
+ d = abs(p->refresh - fps);
+ if (diff_refresh > d) {
+ diff_refresh = d;
+ diff_pixel_clock =
+ abs(p->pixclock - pixel_clock);
+ best = p;
+ } else if (diff_refresh == d) {
+ d = abs(p->pixclock - pixel_clock);
+ if (diff_pixel_clock > d) {
+ diff_pixel_clock = d;
+ best = p;
+ }
+ }
+ }
+ }
+ if (best)
+ DBG_MSG("%dx%d@%d\n", best->xres, best->yres, best->refresh);
+ return best;
+}
+
+#ifndef CONFIG_VPOST
+int vout_find_match_mode(int fbnum,
+ struct fb_videomode *vmode, int match)
+{
+ vout_info_t *info;
+ struct fb_videomode *p;
+ int no = VPP_VOUT_NUM;
+ unsigned int option;
+ int i;
+ unsigned int vo_mask;
+
+ DBG_DETAIL("(%d)\n", fbnum);
+
+ info = vout_info_get_entry(fbnum);
+ if (vmode->refresh == 59)
+ vmode->refresh = 60;
+
+ /* fixed timing */
+ if (info->fixed_vmode) {
+ if (info->fixed_vmode->xres != vmode->xres)
+ return -1;
+ if (info->fixed_vmode->yres != vmode->yres)
+ return -1;
+ if (info->fixed_vmode->refresh != vmode->refresh)
+ return -1;
+ p = info->fixed_vmode;
+ goto label_find_match;
+ }
+
+ vo_mask = vout_get_mask(info);
+ if (vo_mask == 0 && vmode->pixclock)
+ return 0;
+
+ /* find plugin or first vout */
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if ((vo_mask & (0x1 << i)) == 0)
+ continue;
+
+ if (no == VPP_VOUT_NUM)
+ no = i; /* get first vo */
+
+ if (vout_chkplug(i)) {
+ no = i;
+ break;
+ }
+ }
+
+ /* resolution match and interlace match */
+ option = VOUT_MODE_OPTION_GREATER + VOUT_MODE_OPTION_LESS +
+ VOUT_MODE_OPTION_EDID;
+ option |= (vmode->vmode & FB_VMODE_INTERLACED) ?
+ VOUT_MODE_OPTION_INTERLACE : VOUT_MODE_OPTION_PROGRESS;
+ p = vout_get_video_mode(no, vmode, option);
+ if (p)
+ goto label_find_match;
+
+ /* resolution match but interlace not match */
+ option = VOUT_MODE_OPTION_GREATER + VOUT_MODE_OPTION_LESS +
+ VOUT_MODE_OPTION_EDID;
+ option |= (vmode->vmode & FB_VMODE_INTERLACED) ?
+ VOUT_MODE_OPTION_PROGRESS : VOUT_MODE_OPTION_INTERLACE;
+ p = vout_get_video_mode(no, vmode, option);
+ if (p)
+ goto label_find_match;
+
+/* if( !match ){ */
+ /* resolution less best mode */
+ option = VOUT_MODE_OPTION_LESS + VOUT_MODE_OPTION_EDID;
+ p = vout_get_video_mode(no, vmode, option);
+ if (p)
+ goto label_find_match;
+/* } */
+ DBG_ERR("no support mode\n");
+ return -1;
+label_find_match:
+ *vmode = *p;
+#ifdef CONFIG_UBOOT
+ info->p_vmode = p;
+#endif
+ return 0;
+}
+#endif
+
+int vout_find_edid_support_mode(
+ edid_info_t *info,
+ unsigned int *resx,
+ unsigned int *resy,
+ unsigned int *fps,
+ int r_16_9
+)
+{
+ /* check the EDID to find one that not large and same ratio mode*/
+#ifdef CONFIG_WMT_EDID
+ int i, cnt;
+ struct fb_videomode *p;
+ unsigned int w, h, f, option;
+
+ if ((*resx == 720) && (*resy == 480) && (*fps == 50))
+ *fps = 60;
+
+ for (i = 0, cnt = 0; ; i++) {
+ if (vpp_videomode[i].pixclock == 0)
+ break;
+ cnt++;
+ }
+
+ for (i = cnt - 1; i >= 0; i--) {
+ p = (struct fb_videomode *) &vpp_videomode[i];
+ h = p->yres;
+ if (h > *resy)
+ continue;
+
+ w = p->xres;
+ if (w > *resx)
+ continue;
+
+ f = p->refresh;
+ if (f > *fps)
+ continue;
+
+ if (r_16_9 != vout_check_ratio_16_9(w, h))
+ continue;
+
+ option = f & EDID_TMR_FREQ;
+ option |= (p->vmode & FB_VMODE_INTERLACED) ?
+ EDID_TMR_INTERLACE : 0;
+
+ if (edid_find_support(info, w, h, option, &p)) {
+ *resx = w;
+ *resy = h;
+ *fps = f;
+ DBG_MSG("(%dx%d@%d)\n", w, h, f);
+ return 1;
+ }
+ }
+#endif
+ return 0;
+}
+
+/*----------------------- vout control API ----------------------------------*/
+void vout_set_framebuffer(unsigned int mask, vdo_framebuf_t *fb)
+{
+ int i;
+ vout_t *vo;
+
+ if (mask == 0)
+ return;
+
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if ((mask & (0x1 << i)) == 0)
+ continue;
+ vo = vout_get_entry(i);
+ if (vo && (vo->govr))
+ vo->govr->fb_p->set_framebuf(fb);
+ }
+}
+
+void vout_set_tg_enable(unsigned int mask, int enable)
+{
+ int i;
+ vout_t *vo;
+
+ if (mask == 0)
+ return;
+
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if ((mask & (0x1 << i)) == 0)
+ continue;
+ vo = vout_get_entry(i);
+ if (vo && (vo->govr))
+ govrh_set_tg_enable(vo->govr, enable);
+ }
+}
+
+int vout_set_blank(unsigned int mask, vout_blank_t arg)
+{
+ int i;
+ vout_t *vo;
+
+ DBG_DETAIL("(0x%x,%d)\n", mask, arg);
+
+ if (mask == 0)
+ return 0;
+
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if ((mask & (0x1 << i)) == 0)
+ continue;
+
+ vo = vout_get_entry(i);
+ if (vo && (vo->inf)) {
+ vout_change_status(vo, VPP_VOUT_STS_BLANK, arg);
+ vo->inf->blank(vo, arg);
+ if (vo->dev && vo->dev->set_power_down)
+ vo->dev->set_power_down(
+ (arg == VOUT_BLANK_POWERDOWN) ? 1 : 0);
+ }
+ }
+
+ if (1) {
+ int govr_enable_mask, govr_mask;
+ govrh_mod_t *govr;
+
+ govr_enable_mask = 0;
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo && (vo->govr)) {
+ govr_mask = (vo->govr->mod == VPP_MOD_GOVRH) ?
+ 0x1 : 0x2;
+ govr_enable_mask |= (vo->status &
+ VPP_VOUT_STS_BLANK) ? 0 : govr_mask;
+ }
+ }
+ for (i = 0; i < VPP_VOUT_INFO_NUM; i++) {
+ govr = (i == 0) ? p_govrh : p_govrh2;
+ govr_mask = 0x1 << i;
+ govrh_set_MIF_enable(govr,
+ (govr_enable_mask & govr_mask) ? 1 : 0);
+ }
+ }
+ return 0;
+}
+
+int vout_set_mode(int no, vout_inf_mode_t mode)
+{
+ vout_t *vo;
+
+ DBG_DETAIL("(%d,%d)\n", no, mode);
+
+ if (vout_query_inf_support(no, mode) == 0) {
+ DBG_ERR("not support this interface(%d,%d)\n", no, mode);
+ return -1;
+ }
+
+ vo = vout_get_entry(no);
+ if (vo->inf) {
+ if (vo->inf->mode == mode)
+ return 0;
+ vo->inf->uninit(vo, 0);
+ vout_change_status(vo, VPP_VOUT_STS_ACTIVE, 0);
+ if (vo->dev)
+ vo->dev->set_power_down(1);
+ }
+
+ vo->inf = vout_inf_get_entry(mode);
+ vo->inf->init(vo, 0);
+ vout_change_status(vo, VPP_VOUT_STS_ACTIVE, 1);
+ return 0;
+}
+
+int vout_config(unsigned int mask,
+ vout_info_t *info, struct fb_videomode *vmode)
+{
+ vout_t *vo;
+ unsigned int govr_mask = 0;
+ int i;
+
+ DBG_DETAIL("(0x%x)\n", mask);
+
+ if (mask == 0)
+ return 0;
+
+ /* option for interface & device config */
+ info->resx = vmode->xres;
+ info->resy = vmode->yres;
+ info->fps = (vmode->refresh == 59) ? 60 : vmode->refresh;
+ info->option = (vmode->vmode & FB_VMODE_INTERLACED) ?
+ VPP_OPT_INTERLACE : 0;
+ info->option |= (vmode->sync & FB_SYNC_HOR_HIGH_ACT) ?
+ VPP_DVO_SYNC_POLAR_HI : 0;
+ info->option |= (vmode->sync & FB_SYNC_VERT_HIGH_ACT) ?
+ VPP_DVO_VSYNC_POLAR_HI : 0;
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if ((mask & (0x1 << i)) == 0)
+ continue;
+
+ vo = vout_get_entry(i);
+ if (vo == 0)
+ continue;
+
+ if (vo->govr == 0)
+ continue;
+
+ if ((govr_mask & (0x1 << vo->govr->mod)) == 0) {
+ govrh_set_videomode(vo->govr, vmode);
+ govr_mask |= (0x1 << vo->govr->mod);
+ }
+
+ if (vo->inf) {
+ vo->inf->config(vo, (int)info);
+ if (vo->dev)
+ vo->dev->config(info);
+ }
+ }
+ return 0;
+}
+
+int vout_chkplug(int no)
+{
+ vout_t *vo;
+ vout_inf_t *inf;
+ int ret = 0;
+
+ DBG_DETAIL("(%d)\n", no);
+
+ vo = vout_get_entry(no);
+ if (vo == 0)
+ return 0;
+
+ if (vo->inf == 0)
+ return 0;
+
+ inf = vout_inf_get_entry(vo->inf->mode);
+ if (inf == 0)
+ return 0;
+
+ if (vo->dev && vo->dev->check_plugin)
+ ret = vo->dev->check_plugin(0);
+ else
+ ret = inf->chkplug(vo, 0);
+ vout_change_status(vo, VPP_VOUT_STS_PLUGIN, ret);
+ return ret;
+}
+
+int vout_inf_chkplug(int no, vout_inf_mode_t mode)
+{
+ vout_t *vo;
+ vout_inf_t *inf;
+ int plugin = 0;
+
+ DBG_MSG("(%d,%d)\n", no, mode);
+ if (vout_query_inf_support(no, mode) == 0)
+ return 0;
+
+ vo = vout_get_entry(no);
+ inf = vout_inf_get_entry(mode);
+ if (inf) {
+ if (inf->chkplug)
+ plugin = inf->chkplug(vo, 0);
+ }
+ return plugin;
+}
+
+char *vout_get_edid(int no)
+{
+ vout_t *vo;
+ int ret;
+
+ DBG_DETAIL("(%d)\n", no);
+
+ if (edid_disable)
+ return 0;
+ vo = vout_get_entry(no);
+ if (vo == 0)
+ return 0;
+
+ if (vo->status & VPP_VOUT_STS_EDID) {
+ DBG_MSG("edid exist\n");
+ return vo->edid;
+ }
+
+ vout_change_status(vo, VPP_VOUT_STS_EDID, 0);
+#ifdef CONFIG_VOUT_EDID_ALLOC
+ if (vo->edid == 0) {
+ vo->edid = kmalloc(128 * EDID_BLOCK_MAX, GFP_KERNEL);
+ if (!vo->edid) {
+ DBG_ERR("edid buf alloc\n");
+ return 0;
+ }
+ }
+#endif
+
+ ret = 1;
+ if (vo->dev && vo->dev->get_edid) {
+ ret = vo->dev->get_edid(vo->edid);
+ } else {
+ if (vo->inf->get_edid)
+ ret = vo->inf->get_edid(vo, (int)vo->edid);
+ }
+
+ if (ret == 0) {
+ DBG_DETAIL("edid read\n");
+ vout_change_status(vo, VPP_VOUT_STS_EDID, 1);
+ return vo->edid;
+ } else {
+ DBG_MSG("read edid fail\n");
+ }
+
+#ifdef CONFIG_VOUT_EDID_ALLOC
+ kfree(vo->edid);
+ vo->edid = 0;
+#endif
+ return 0;
+}
+
+int vout_get_edid_option(int no)
+{
+ vout_t *vo;
+
+ DBG_DETAIL("(%d)\n", no);
+
+ vo = vout_get_entry(no);
+ if (vo == 0)
+ return 0;
+
+ if (vo->edid_info.option)
+ return vo->edid_info.option;
+
+ if (vout_get_edid(no) == 0) {
+ if (no == VPP_VOUT_NUM_HDMI) { /* HDMI wo EDID still can work */
+ vo->edid_info.option = (EDID_OPT_HDMI + EDID_OPT_AUDIO);
+ return vo->edid_info.option;
+ }
+ return 0;
+ }
+
+ edid_parse(vo->edid, &vo->edid_info);
+ return vo->edid_info.option;
+}
+
+unsigned int vout_get_mask(vout_info_t *vo_info)
+{
+ if (g_vpp.dual_display == 0)
+ return VPP_VOUT_ALL;
+
+ if (g_vpp.virtual_display) {
+ if (vo_info->num == 0)
+ return 0;
+ return VPP_VOUT_ALL;
+ }
+ return vo_info->vo_mask;
+}
+
+int vout_check_plugin(int clr_sts)
+{
+ vout_t *vo;
+ int i;
+ int plugin = 0;
+
+ for (i = 0; i <= VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo == 0)
+ continue;
+ if (vo->inf == 0)
+ continue;
+
+ if (vo->dev) {
+ if (!(vo->dev->capability & VOUT_DEV_CAP_FIX_PLUG)) {
+ if (vout_chkplug(i)) {
+ plugin = 1;
+ if (clr_sts)
+ vout_change_status(vo,
+ VPP_VOUT_STS_PLUGIN, 0);
+ DBG_MSG("[VPP] ext dev plugin\n");
+ }
+ }
+ } else {
+ if (!(vo->inf->capability & VOUT_INF_CAP_FIX_PLUG)) {
+ if (vout_chkplug(i)) {
+ plugin = 1;
+ if (clr_sts)
+ vout_change_status(vo,
+ VPP_VOUT_STS_PLUGIN, 0);
+ DBG_MSG("[VPP] inf dev plugin\n");
+ }
+ }
+ }
+ }
+ return plugin;
+}
+/*--------------------End of Function Body -----------------------------------*/
+#undef VOUT_C
diff --git a/drivers/video/wmt/vout.h b/drivers/video/wmt/vout.h
new file mode 100644
index 00000000..069cc32e
--- /dev/null
+++ b/drivers/video/wmt/vout.h
@@ -0,0 +1,283 @@
+/*++
+ * linux/drivers/video/wmt/vout.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef VOUT_H
+/* To assert that only one occurrence is included */
+#define VOUT_H
+/*-------------------- MODULE DEPENDENCY -------------------------------------*/
+#include "vpp.h"
+#include "sw_i2c.h"
+#include "edid.h"
+
+/* following is the C++ header */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+/* #define VO_XXXX 1 *//*Example*/
+/* #define CONFIG_VOUT_EDID_ALLOC */
+#define CONFIG_VOUT_REFACTORY
+
+#define VOUT_INFO_DEFAULT_RESX 1024
+#define VOUT_INFO_DEFAULT_RESY 768
+#define VOUT_INFO_DEFAULT_FPS 60
+
+/*-------------------- EXPORTED PRIVATE TYPES---------------------------------*/
+/* typedef void vo_xxx_t; *//*Example*/
+typedef enum {
+ VOUT_SD_ANALOG,
+ VOUT_SD_DIGITAL,
+ VOUT_LCD,
+ VOUT_DVI,
+ VOUT_HDMI,
+ VOUT_DVO2HDMI,
+ VOUT_LVDS,
+ VOUT_VGA,
+ VOUT_BOOT,
+ VOUT_MODE_MAX,
+ VOUT_MODE_ALL = VOUT_MODE_MAX
+} vout_mode_t;
+
+typedef enum {
+ VOUT_DEV_VGA,
+ VOUT_DEV_DVI,
+ VOUT_DEV_LCD,
+ VOUT_DEV_HDMI,
+ VOUT_DEV_SDD,
+ VOUT_DEV_LVDS,
+ VOUT_DEV_MODE_MAX
+} vout_dev_mode_t;
+
+typedef enum {
+ VOUT_INF_DVI,
+ VOUT_INF_HDMI,
+ VOUT_INF_LVDS,
+ VOUT_INF_MODE_MAX
+} vout_inf_mode_t;
+
+typedef enum {
+ TV_PAL,
+ TV_NTSC,
+ TV_UNDEFINED,
+ TV_MAX
+} vout_tvformat_t;
+
+typedef struct {
+ int num;
+ unsigned int vo_mask; /* vo bit mask for multi vout */
+ int govr_mod; /* govr module type */
+ govrh_mod_t *govr; /* govr pointer */
+
+ int resx;
+ int resy;
+ int resx_virtual;
+ int resy_virtual;
+ int bpp;
+ int fps;
+ unsigned int pixclk;
+ unsigned int option;
+
+ struct fb_videomode *fixed_vmode;
+ vdo_framebuf_t fb;
+#ifdef CONFIG_UBOOT
+ struct fb_videomode *p_vmode;
+#endif
+ vpp_dbg_timer_t pandisp_timer;
+} vout_info_t;
+
+typedef struct {
+ int fmt; /* sample bits */
+ int sample_rate; /* sample rate */
+ int channel; /* channel count */
+} vout_audio_t;
+
+struct vout_s;
+struct vout_inf_s;
+
+#define VOUT_DEV_CAP_FIX_RES 0x1
+#define VOUT_DEV_CAP_EDID 0x2
+#define VOUT_DEV_CAP_AUDIO 0x4
+#define VOUT_DEV_CAP_FIX_PLUG 0x8
+
+typedef struct vout_dev_s {
+ struct vout_dev_s *next;
+ char name[10];
+ vout_inf_mode_t mode;
+ struct vout_s *vout;
+ unsigned int capability;
+
+ int (*init)(struct vout_s *vo);
+ void (*set_power_down)(int enable);
+ int (*set_mode)(unsigned int *option);
+ int (*config)(vout_info_t *info);
+ int (*check_plugin)(int hotplug);
+ int (*get_edid)(char *buf);
+ int (*set_audio)(vout_audio_t *arg);
+ int (*interrupt)(void);
+ void (*poll)(void);
+} vout_dev_t;
+
+typedef enum {
+ VOUT_BLANK_UNBLANK, /* screen: unblanked, hsync: on, vsync: on */
+ VOUT_BLANK_NORMAL, /* screen: blanked, hsync: on, vsync: on */
+ VOUT_BLANK_VSYNC_SUSPEND,/* screen: blanked, hsync: on, vsync: off */
+ VOUT_BLANK_HSYNC_SUSPEND,/* screen: blanked, hsync: off, vsync: on */
+ VOUT_BLANK_POWERDOWN /* screen: blanked, hsync: off, vsync: off */
+} vout_blank_t;
+
+#define VOUT_CAP_INTERFACE 0x000000FF
+#define VOUT_CAP_BUS 0x00000F00
+#define VOUT_CAP_GOVR 0x0000F000
+#define VOUT_CAP_EXT_DEV 0x00010000
+#define VOUT_CAP_FIX_PLUG 0x00020000
+#define VOUT_CAP_AUDIO 0x00040000
+#define VOUT_CAP_EDID 0x00080000
+
+typedef struct vout_s {
+ int num;
+ unsigned int fix_cap;
+ unsigned int var_cap;
+ struct vout_inf_s *inf; /* interface ops */
+ vout_dev_t *dev; /* device ops */
+ vout_dev_t *ext_dev;
+ govrh_mod_t *govr;
+ int resx;
+ int resy;
+ int pixclk;
+ unsigned int status;
+#ifdef CONFIG_VOUT_EDID_ALLOC
+ char *edid;
+#else
+ char edid[128*EDID_BLOCK_MAX];
+#endif
+ edid_info_t edid_info;
+ unsigned int option[3];
+ vout_blank_t pre_blank;
+} vout_t;
+
+#define VOUT_INF_CAP_FIX_PLUG BIT(0)
+typedef struct vout_inf_s {
+ vout_inf_mode_t mode;
+ unsigned int capability;
+
+ /* function */
+ int (*init)(vout_t *vo, int arg);
+ int (*uninit)(vout_t *vo, int arg);
+ int (*blank)(vout_t *vo, vout_blank_t arg);
+ int (*config)(vout_t *vo, int arg);
+ int (*chkplug)(vout_t *vo, int arg);
+ int (*get_edid)(vout_t *vo, int arg);
+/* int (*ioctl)(vout_t *vo,int arg); */
+} vout_inf_t;
+
+/*-------------------- EXPORTED PRIVATE VARIABLES ---------------------------*/
+#ifdef VOUT_C /* allocate memory for variables only in vout.c */
+#define EXTERN
+
+const char *vout_inf_str[] = {"DVI", "HDMI", "LVDS", "VGA", "SDA", "SDD"};
+const char *vout_adpt_str[] = {"SD_DIGITAL", "SD_DIGITAL", "LCD", "DVI",
+ "HDMI", "DVO2HDMI", "LVDS", "VGA", "BOOT"};
+
+#else
+#define EXTERN extern
+
+extern const char *vout_inf_str[];
+extern const char *vout_adpt_str[];
+
+#endif /* ifdef VOUT_C */
+
+EXTERN vout_info_t vout_info[VPP_VOUT_INFO_NUM];
+
+/* EXTERN int vo_xxx; *//*Example*/
+EXTERN int (*vout_board_info)(int arg);
+
+#undef EXTERN
+
+/*--------------------- EXPORTED PRIVATE MACROS -----------------------------*/
+/* #define VO_XXX_YYY xxxx *//*Example*/
+/*--------------------- EXPORTED PRIVATE FUNCTIONS -------------------------*/
+/* extern void vo_xxx(void); *//*Example*/
+
+void vout_register(int no, vout_t *vo);
+vout_t *vout_get_entry(int no);
+vout_info_t *vout_get_info_entry(int no);
+void vout_change_status(vout_t *vo, int mask, int sts);
+int vout_query_inf_support(int no, vout_inf_mode_t mode);
+
+int vout_inf_register(vout_inf_mode_t mode, vout_inf_t *inf);
+vout_inf_t *vout_inf_get_entry(vout_inf_mode_t mode);
+
+int vout_device_register(vout_dev_t *ops);
+vout_dev_t *vout_get_device(vout_dev_t *ops);
+
+vout_t *vout_get_entry_adapter(vout_mode_t mode);
+vout_inf_t *vout_get_inf_entry_adapter(vout_mode_t mode);
+int vout_info_add_entry(vout_t *vo);
+vout_info_t *vout_info_get_entry(int no);
+void vout_info_set_fixed_timing(int no, struct fb_videomode *vmode);
+govrh_mod_t *vout_info_get_govr(int no);
+
+int vout_config(unsigned int mask,
+ vout_info_t *info, struct fb_videomode *vmode);
+int vout_set_mode(int no, vout_inf_mode_t mode);
+int vout_set_blank(unsigned int mask, vout_blank_t blank);
+void vout_set_tg_enable(unsigned int mask, int enable);
+void vout_set_framebuffer(unsigned int mask, vdo_framebuf_t *fb);
+int vout_chkplug(int no);
+void vout_set_int_type(int type);
+char *vout_get_edid(int no);
+int vout_get_edid_option(int no);
+int vout_check_plugin(int clr_sts);
+
+int vout_init(void);
+int vout_exit(void);
+int vo_i2c_proc(int id, unsigned int addr, unsigned int index,
+ char *pdata, int len);
+int vout_set_audio(vout_audio_t *arg);
+int vout_find_edid_support_mode(edid_info_t *info,
+ unsigned int *resx, unsigned int *resy, unsigned int *fps, int r_16_9);
+int vout_check_ratio_16_9(unsigned int resx, unsigned int resy);
+unsigned int vout_get_mask(vout_info_t *vo_info);
+void vout_set_int_enable(int enable);
+int vout_get_clr_int(void);
+void vo_hdmi_set_clock(int enable);
+#ifndef CONFIG_VPOST
+int vout_find_match_mode(int fbnum,
+ struct fb_videomode *vmode, int match);
+#endif
+#define VOUT_MODE_OPTION_LESS BIT0
+#define VOUT_MODE_OPTION_GREATER BIT1
+#define VOUT_MODE_OPTION_EDID BIT2
+#define VOUT_MODE_OPTION_INTERLACE BIT3
+#define VOUT_MODE_OPTION_PROGRESS BIT4
+struct fb_videomode *vout_get_video_mode(int vout_num,
+ struct fb_videomode *vmode, int option);
+void vo_hdmi_cp_set_enable_tmr(int sec);
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* ifndef VOUT_H */
+
+/*=== END vout.h ==========================================================*/
diff --git a/drivers/video/wmt/vpp-osif.c b/drivers/video/wmt/vpp-osif.c
new file mode 100644
index 00000000..a8505a59
--- /dev/null
+++ b/drivers/video/wmt/vpp-osif.c
@@ -0,0 +1,891 @@
+/*++
+ * linux/drivers/video/wmt/osif.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define VPP_OSIF_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include "vpp.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define LVDS_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx lvds_xxx_t; *//*Example*/
+
+/*----------EXPORTED PRIVATE VARIABLES are defined in lvds.h -------------*/
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int lvds_xxx; *//*Example*/
+#ifdef __KERNEL__
+static DEFINE_SPINLOCK(vpp_irqlock);
+static unsigned long vpp_lock_flags;
+#endif
+int vpp_dvi_i2c_id;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void lvds_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+#ifdef __KERNEL__
+#include <asm/io.h>
+#include <linux/proc_fs.h>
+#else
+__inline__ U32 inl(U32 offset)
+{
+ return REG32_VAL(offset);
+}
+
+__inline__ void outl(U32 val, U32 offset)
+{
+ REG32_VAL(offset) = val;
+}
+
+static __inline__ U16 inw(U32 offset)
+{
+ return REG16_VAL(offset);
+}
+
+static __inline__ void outw(U16 val, U32 offset)
+{
+ REG16_VAL(offset) = val;
+}
+
+static __inline__ U8 inb(U32 offset)
+{
+ return REG8_VAL(offset);
+}
+
+static __inline__ void outb(U8 val, U32 offset)
+{
+ REG8_VAL(offset) = val;
+}
+#ifndef CFG_LOADER
+int get_key(void)
+{
+ int key;
+
+ key = get_num(0, 256, "Input:", 5);
+ DPRINT("\n");
+ return key;
+}
+
+void udelay(int us)
+{
+ vpp_post_delay(us);
+}
+
+void mdelay(int ms)
+{
+ udelay(ms * 1000);
+}
+#endif
+#endif
+
+extern unsigned int wmt_read_oscr(void);
+void vpp_udelay(unsigned int us)
+{
+#if 1
+ udelay(us);
+#else
+ unsigned int cur;
+ unsigned int cross;
+
+ if (us == 0)
+ return;
+
+ cur = wmt_read_oscr();
+ us = cur + us * 3;
+ cross = (us < cur) ? cur : 0; /* check overflow */
+ while (1) {
+ if (cross) {
+ if (cur < cross)
+ cross = 0;
+ } else {
+ if (us < cur)
+ break;
+ }
+ cur = wmt_read_oscr();
+ }
+#endif
+}
+
+/* Internal functions */
+U8 vppif_reg8_in(U32 offset)
+{
+ return inb(offset);
+}
+
+U8 vppif_reg8_out(U32 offset, U8 val)
+{
+ outb(val, offset);
+ return val;
+}
+
+U16 vppif_reg16_in(U32 offset)
+{
+ return inw(offset);
+}
+
+U16 vppif_reg16_out(U32 offset, U16 val)
+{
+ outw(val, offset);
+ return val;
+}
+
+U32 vppif_reg32_in(U32 offset)
+{
+ return inl(offset);
+}
+
+U32 vppif_reg32_out(U32 offset, U32 val)
+{
+ outl(val, offset);
+ return val;
+}
+
+U32 vppif_reg32_write(U32 offset, U32 mask, U32 shift, U32 val)
+{
+ U32 new_val;
+
+#if 0
+ if (val > (mask >> shift))
+ MSG("*E* check the parameter 0x%x 0x%x 0x%x 0x%x\n",
+ offset, mask, shift, val);
+#endif
+ new_val = (inl(offset) & ~(mask)) | (((val) << (shift)) & mask);
+ outl(new_val, offset);
+ return new_val;
+}
+
+U32 vppif_reg32_read(U32 offset, U32 mask, U32 shift)
+{
+ return (inl(offset) & mask) >> shift;
+}
+
+U32 vppif_reg32_mask(U32 offset, U32 mask, U32 shift)
+{
+ return mask;
+}
+
+int vpp_request_irq(unsigned int irq_no, void *routine,
+ unsigned int flags, char *name, void *arg)
+{
+#if 0 /* disable irq */
+ return 0;
+#endif
+
+#ifdef __KERNEL__
+ if (request_irq(irq_no, routine, flags, name, arg)) {
+ DPRINT("[VPP] *E* request irq %s fail\n", name);
+ return -1;
+ }
+#endif
+ return 0;
+}
+
+void vpp_free_irq(unsigned int irq_no, void *arg)
+{
+#ifdef __KERNEL__
+ free_irq(irq_no, arg);
+#endif
+}
+
+#ifndef __KERNEL__
+int wmt_getsyspara(char *varname, char *varval, int *varlen)
+{
+ int i = 0;
+ char *p;
+
+ p = getenv(varname);
+ if (!p) {
+ printf("## Warning: %s not defined\n", varname);
+ return -1;
+ }
+ while (p[i] != '\0') {
+ varval[i] = p[i];
+ i++;
+ }
+ varval[i] = '\0';
+ *varlen = i;
+ /* printf("getsyspara: %s,len %d\n", p, *varlen); */
+ return 0;
+}
+#endif
+
+#ifndef CONFIG_VPOST
+int vpp_parse_param(char *buf, unsigned int *param,
+ int cnt, unsigned int hex_mask)
+{
+ char *p;
+ char *endp;
+ int i = 0;
+
+ if (*buf == '\0')
+ return 0;
+
+ for (i = 0; i < cnt; i++)
+ param[i] = 0;
+
+ p = (char *)buf;
+ for (i = 0; i < cnt; i++) {
+ param[i] = simple_strtoul(p, &endp,
+ (hex_mask & (0x1 << i)) ? 16 : 0);
+ if (*endp == '\0')
+ break;
+ p = endp + 1;
+
+ if (*p == '\0')
+ break;
+ }
+ return i + 1;
+}
+#endif
+
+unsigned int vpp_lock_cnt;
+void vpp_lock_l(void)
+{
+#ifdef __KERNEL__
+#if 0
+ vpp_lock_cnt++;
+ DPRINT("vpp_lock(%d)\n", vpp_lock_cnt);
+#endif
+ spin_lock_irqsave(&vpp_irqlock, vpp_lock_flags);
+#else
+#endif
+}
+
+void vpp_unlock(void)
+{
+#ifdef __KERNEL__
+#if 0
+ vpp_lock_cnt--;
+ DPRINT("vpp_unlock(%d)\n", vpp_lock_cnt);
+#endif
+ spin_unlock_irqrestore(&vpp_irqlock, vpp_lock_flags);
+#else
+#endif
+}
+
+#ifdef __KERNEL__
+struct i2c_adapter *vpp_i2c_adapter;
+struct i2c_client *vpp_i2c_client;
+struct vpp_i2c_priv {
+ unsigned int var;
+};
+static int __devinit vpp_i2c_probe(struct i2c_client *i2c,
+ const struct i2c_device_id *id)
+{
+ struct vpp_i2c_priv *vpp_i2c;
+ int ret = 0;
+
+ DBGMSG("\n");
+ if (vpp_i2c_client == 0)
+ return -ENODEV;
+
+ if (!i2c_check_functionality(vpp_i2c_client->adapter, I2C_FUNC_I2C)) {
+ DBG_ERR("need I2C_FUNC_I2C\n");
+ return -ENODEV;
+ }
+
+ vpp_i2c = kzalloc(sizeof(struct vpp_i2c_priv), GFP_KERNEL);
+ if (vpp_i2c == NULL) {
+ DBG_ERR("kzalloc fail\n");
+ return -ENOMEM;
+ }
+ i2c_set_clientdata(i2c, vpp_i2c);
+ return ret;
+}
+
+static int vpp_i2c_remove(struct i2c_client *client)
+{
+ kfree(i2c_get_clientdata(client));
+ return 0;
+}
+
+static const struct i2c_device_id vpp_i2c_id[] = {
+ { "vpp_i2c", 0},
+ { }
+};
+MODULE_DEVICE_TABLE(i2c, vpp_i2c_id);
+
+static struct i2c_driver vpp_i2c_driver = {
+ .probe = vpp_i2c_probe,
+ .remove = vpp_i2c_remove,
+ .id_table = vpp_i2c_id,
+ .driver = { .name = "vpp_i2c", },
+};
+
+static struct i2c_board_info vpp_i2c_board_info = {
+ .type = "vpp_i2c",
+ .flags = 0x00,
+ .platform_data = NULL,
+ .archdata = NULL,
+ .irq = -1,
+};
+#endif
+#ifdef __KERNEL__
+int vpp_i2c_xfer(struct i2c_msg *msg, unsigned int num, int bus_id)
+#else
+int vpp_i2c_xfer(struct i2c_msg_s *msg, unsigned int num, int bus_id)
+#endif
+{
+ int ret = 1;
+
+#ifdef __KERNEL__
+ int i = 0;
+#if 0
+ if (bus_id == 1) {
+ ret = wmt_i2c_xfer_continue_if_4(msg, num, bus_id);
+ return ret;
+ }
+#endif
+
+ if (num > 1) {
+ for (i = 0; i < num - 1; ++i)
+ msg[i].flags |= I2C_M_NOSTART;
+ }
+ ret = i2c_transfer(vpp_i2c_adapter, msg, num);
+ if (ret <= 0) {
+ DBG_ERR("i2c fail\n");
+ return ret;
+ }
+#elif defined(CONFIG_VPOST)
+ ret = i2c_transfer(msg, 1);
+#else
+ ret = wmt_i2c_transfer(msg, num, bus_id);
+#endif
+ return ret;
+}
+
+int vpp_i2c_init(int i2c_id, unsigned short addr)
+{
+#ifdef __KERNEL__
+ struct i2c_board_info *vpp_i2c_bi = &vpp_i2c_board_info;
+ int ret = 0;
+
+ DBGMSG("id %d,addr 0x%x\n", i2c_id, addr);
+
+ vpp_dvi_i2c_id = i2c_id;
+ if (i2c_id & VPP_DVI_I2C_SW_BIT)
+ return 0;
+
+ i2c_id &= VPP_DVI_I2C_ID_MASK;
+ vpp_i2c_adapter = i2c_get_adapter(i2c_id);
+ if (vpp_i2c_adapter == NULL) {
+ DBG_ERR("can not get i2c adapter, client address error\n");
+ return -ENODEV;
+ }
+
+ vpp_i2c_bi->addr = addr >> 1;
+
+ vpp_i2c_client = i2c_new_device(vpp_i2c_adapter, vpp_i2c_bi);
+ if (vpp_i2c_client == NULL) {
+ DBG_ERR("allocate i2c client failed\n");
+ return -ENOMEM;
+ }
+ i2c_put_adapter(vpp_i2c_adapter);
+
+ ret = i2c_add_driver(&vpp_i2c_driver);
+ if (ret)
+ DBG_ERR("i2c_add_driver fail\n");
+ return ret;
+#else
+ vpp_dvi_i2c_id = i2c_id & 0xF;
+ return 0;
+#endif
+}
+
+int vpp_i2c_release(void)
+{
+#ifdef __KERNEL__
+ if (vpp_i2c_client) {
+ i2c_del_driver(&vpp_i2c_driver);
+ i2c_unregister_device(vpp_i2c_client);
+ vpp_i2c_remove(vpp_i2c_client);
+ vpp_i2c_client = 0;
+ }
+#else
+
+#endif
+ return 0;
+}
+
+#ifdef __KERNEL__
+static DEFINE_SEMAPHORE(vpp_i2c_sem);
+#endif
+void vpp_i2c_set_lock(int lock)
+{
+#ifdef __KERNEL__
+ if (lock)
+ down(&vpp_i2c_sem);
+ else
+ up(&vpp_i2c_sem);
+#endif
+}
+
+int vpp_i2c_lock;
+int vpp_i2c_enhanced_ddc_read(int id, unsigned int addr,
+ unsigned int index, char *pdata, int len)
+{
+#ifdef __KERNEL__
+ struct i2c_msg msg[3];
+#else
+ struct i2c_msg_s msg[3];
+#endif
+ unsigned char buf[len + 1];
+ unsigned char buf2[2];
+ int ret = 0;
+
+ vpp_i2c_set_lock(1);
+
+ if (vpp_i2c_lock)
+ DBG_ERR("in lock\n");
+
+ vpp_i2c_lock = 1;
+
+ if (id & VPP_DVI_I2C_BIT)
+ id = vpp_dvi_i2c_id;
+ id = id & VPP_DVI_I2C_ID_MASK;
+ buf2[0] = 0x1;
+ buf2[1] = 0x0;
+ msg[0].addr = (0x60 >> 1);
+ msg[0].flags = 0 ;
+ msg[0].flags &= ~(I2C_M_RD);
+ msg[0].len = 1;
+ msg[0].buf = buf2;
+
+ addr = (addr >> 1);
+ memset(buf, 0x55, len + 1);
+ buf[0] = index;
+ buf[1] = 0x0;
+
+ msg[1].addr = addr;
+ msg[1].flags = 0 ;
+ msg[1].flags &= ~(I2C_M_RD);
+ msg[1].len = 1;
+ msg[1].buf = buf;
+
+ msg[2].addr = addr;
+ msg[2].flags = 0 ;
+ msg[2].flags |= (I2C_M_RD);
+ msg[2].len = len;
+ msg[2].buf = buf;
+
+ ret = vpp_i2c_xfer(msg, 3, id);
+ memcpy(pdata, buf, len);
+ vpp_i2c_lock = 0;
+ vpp_i2c_set_lock(0);
+ return ret;
+} /* End of vpp_i2c_enhanced_ddc_read */
+EXPORT_SYMBOL(vpp_i2c_enhanced_ddc_read);
+
+int vpp_i2c_read(int id, unsigned int addr,
+ unsigned int index, char *pdata, int len)
+{
+ int ret = 0;
+
+ DBG_DETAIL("(%d,0x%x,%d,%d)\n", id, addr, index, len);
+ vpp_i2c_set_lock(1);
+ if (vpp_i2c_lock)
+ DBG_ERR("in lock\n");
+
+ vpp_i2c_lock = 1;
+
+ if (id & VPP_DVI_I2C_BIT)
+ id = vpp_dvi_i2c_id;
+ id = id & VPP_DVI_I2C_ID_MASK;
+ switch (id) {
+ case 0 ... 0xF: /* hw i2c */
+ {
+#ifdef CONFIG_KERNEL
+ struct i2c_msg msg[2];
+#else
+ struct i2c_msg_s msg[2] ;
+#endif
+ unsigned char buf[len + 1];
+
+ addr = (addr >> 1);
+
+ buf[0] = index;
+ buf[1] = 0x0;
+
+ msg[0].addr = addr; /* slave address */
+ msg[0].flags = 0 ;
+ msg[0].flags &= ~(I2C_M_RD);
+ msg[0].len = 1;
+ msg[0].buf = buf;
+
+ msg[1].addr = addr;
+ msg[1].flags = I2C_M_RD;
+ msg[1].len = len;
+ msg[1].buf = buf;
+ ret = vpp_i2c_xfer(msg, 2, id);
+ memcpy(pdata, buf, len);
+ }
+ break;
+ default:
+#ifdef CONFIG_KERNEL
+ ret = vo_i2c_proc((id & 0xF), (addr | BIT0), index, pdata, len);
+#endif
+ break;
+ }
+#ifdef DEBUG
+ {
+ int i;
+
+ DBGMSG("vpp_i2c_read(addr 0x%x,index 0x%x,len %d\n", addr, index, len);
+ for (i = 0; i < len; i += 8) {
+ DBGMSG("%d : 0x%02x 0x%02x 0x%02x 0x%02x",
+ i, pdata[i], pdata[i + 1], pdata[i + 2], pdata[i + 3]);
+ DBGMSG(" 0x%02x 0x%02x 0x%02x 0x%02x\n",
+ pdata[i + 4], pdata[i + 5], pdata[i + 6], pdata[i + 7]);
+ }
+ }
+#endif
+ vpp_i2c_lock = 0;
+ vpp_i2c_set_lock(0);
+ return ret;
+}
+EXPORT_SYMBOL(vpp_i2c_read);
+
+int vpp_i2c_write(int id, unsigned int addr, unsigned int index,
+ char *pdata, int len)
+{
+ int ret = 0;
+
+ DBG_DETAIL("(%d,0x%x,%d,%d)\n", id, addr, index, len);
+ vpp_i2c_set_lock(1);
+ if (vpp_i2c_lock)
+ DBG_ERR("in lock\n");
+
+ vpp_i2c_lock = 1;
+
+ if (id & VPP_DVI_I2C_BIT)
+ id = vpp_dvi_i2c_id;
+ id = id & VPP_DVI_I2C_ID_MASK;
+ switch (id) {
+ case 0 ... 0xF: /* hw i2c */
+ {
+#ifdef CONFIG_KERNEL
+ struct i2c_msg msg[1];
+#else
+ struct i2c_msg_s msg[1] ;
+#endif
+ unsigned char buf[len + 1];
+
+ addr = (addr >> 1);
+ buf[0] = index;
+ memcpy(&buf[1], pdata, len);
+ msg[0].addr = addr; /* slave address */
+ msg[0].flags = 0 ;
+ msg[0].flags &= ~(I2C_M_RD);
+ msg[0].len = len + 1;
+ msg[0].buf = buf;
+ ret = vpp_i2c_xfer(msg, 1, id);
+ }
+ break;
+ default:
+#ifdef CONFIG_KERNEL
+ vo_i2c_proc((id & 0xF), (addr & ~BIT0), index, pdata, len);
+#endif
+ break;
+ }
+
+#ifdef DEBUG
+ {
+ int i;
+
+ DBGMSG("vpp_i2c_write(addr 0x%x,index 0x%x,len %d\n", addr, index, len);
+ for (i = 0; i < len; i += 8) {
+ DBGMSG("%d : 0x%02x 0x%02x 0x%02x 0x%02x",
+ i, pdata[i], pdata[i + 1], pdata[i + 2], pdata[i + 3]);
+ DBGMSG(" 0x%02x 0x%02x 0x%02x 0x%02x\n",
+ pdata[i + 4], pdata[i + 5], pdata[i + 6], pdata[i + 7]);
+ }
+ }
+#endif
+ vpp_i2c_lock = 0;
+ vpp_i2c_set_lock(0);
+ return ret;
+}
+EXPORT_SYMBOL(vpp_i2c_write);
+
+void DelayMS(int ms)
+{
+ mdelay(ms);
+}
+EXPORT_SYMBOL(DelayMS);
+
+/*----------------------- VPP debug --------------------------------------*/
+#define VPP_DEBUG_FUNC
+#ifdef VPP_DEBUG_FUNC
+#define VPP_DBG_TMR_NUM 3
+/* #define VPP_DBG_DIAG_NUM 100 */
+#ifdef VPP_DBG_DIAG_NUM
+char vpp_dbg_diag_str[VPP_DBG_DIAG_NUM][100];
+int vpp_dbg_diag_index;
+int vpp_dbg_diag_delay;
+#endif
+
+int vpp_check_dbg_level(vpp_dbg_level_t level)
+{
+ if (level == VPP_DBGLVL_ALL)
+ return 1;
+
+ switch (g_vpp.dbg_msg_level) {
+ case VPP_DBGLVL_DISABLE:
+ break;
+ case VPP_DBGLVL_ALL:
+ return 1;
+ default:
+ if (g_vpp.dbg_msg_level == level)
+ return 1;
+ break;
+ }
+ return 0;
+}
+
+void vpp_dbg_show(int level, int tmr, char *str)
+{
+#ifdef __KERNEL__
+ static struct timeval pre_tv[VPP_DBG_TMR_NUM];
+ struct timeval tv;
+ unsigned int tm_usec = 0;
+
+ if (vpp_check_dbg_level(level) == 0)
+ return;
+
+ if (tmr && (tmr <= VPP_DBG_TMR_NUM)) {
+ do_gettimeofday(&tv);
+ if (pre_tv[tmr - 1].tv_sec)
+ tm_usec = (tv.tv_sec == pre_tv[tmr - 1].tv_sec) ?
+ (tv.tv_usec - pre_tv[tmr - 1].tv_usec) :
+ (1000000 + tv.tv_usec - pre_tv[tmr - 1].tv_usec);
+ pre_tv[tmr - 1] = tv;
+ }
+
+#ifdef VPP_DBG_DIAG_NUM
+ if (level == VPP_DBGLVL_DIAG) {
+ if (str) {
+ char *ptr = &vpp_dbg_diag_str[vpp_dbg_diag_index][0];
+ sprintf(ptr, "%s (%d,%d)(T%d %d usec)", str,
+ (int)tv.tv_sec, (int)tv.tv_usec, tmr,
+ (int) tm_usec);
+ vpp_dbg_diag_index = (vpp_dbg_diag_index + 1)
+ % VPP_DBG_DIAG_NUM;
+ }
+
+ if (vpp_dbg_diag_delay) {
+ vpp_dbg_diag_delay--;
+ if (vpp_dbg_diag_delay == 0) {
+ int i;
+
+ DPRINT("----- VPP DIAG -----\n");
+ for (i = 0; i < VPP_DBG_DIAG_NUM; i++) {
+ DPRINT("%02d : %s\n", i,
+ &vpp_dbg_diag_str[vpp_dbg_diag_index][0]);
+ vpp_dbg_diag_index = (vpp_dbg_diag_index + 1)
+ % VPP_DBG_DIAG_NUM;
+ }
+ }
+ }
+ return;
+ }
+#endif
+
+ if (str) {
+ if (tmr)
+ DPRINT("[VPP] %s (T%d period %d usec)\n", str,
+ tmr - 1, (int) tm_usec);
+ else
+ DPRINT("[VPP] %s\n", str);
+ }
+#else
+ if (vpp_check_dbg_level(level) == 0)
+ return;
+
+ if (str)
+ DPRINT("[VPP] %s\n", str);
+#endif
+} /* End of vpp_dbg_show */
+
+void vpp_dbg_show_val1(int level, int tmr, char *str, int val)
+{
+ if (vpp_check_dbg_level(level)) {
+ char buf[50];
+
+ sprintf(buf, "%s 0x%x", str, val);
+ vpp_dbg_show(level, tmr, buf);
+ }
+}
+
+#ifdef __KERNEL__
+static DECLARE_WAIT_QUEUE_HEAD(vpp_dbg_wq);
+void vpp_dbg_wait(char *str)
+{
+ DPRINT("[VPP] vpp_dbg_wait(%s)\n", str);
+ wait_event_interruptible(vpp_dbg_wq, (g_vpp.dbg_wait));
+ g_vpp.dbg_wait = 0;
+ DPRINT("[VPP] Exit vpp_dbg_wait\n");
+}
+
+void vpp_dbg_wake_up(void)
+{
+ wake_up(&vpp_dbg_wq);
+}
+
+int vpp_dbg_get_period_usec(vpp_dbg_period_t *p, int cmd)
+{
+ struct timeval tv;
+ int tm_usec = 0;
+
+ if (p == 0)
+ return 0;
+
+ do_gettimeofday(&tv);
+ if (p->pre_tv.tv_sec)
+ tm_usec = (tv.tv_sec == p->pre_tv.tv_sec) ?
+ (tv.tv_usec - p->pre_tv.tv_usec) :
+ (1000000 + tv.tv_usec - p->pre_tv.tv_usec);
+ p->pre_tv = tv;
+ if (cmd == 0) { /* reset */
+ p->index = 0;
+ memset(&p->period_us, 0, VPP_DBG_PERIOD_NUM);
+ } else if (p->index < VPP_DBG_PERIOD_NUM) {
+ p->period_us[p->index] = tm_usec;
+ p->index++;
+ }
+
+ if (cmd == 2) { /* show */
+ int i, sum = 0;
+
+ DPRINT("[VPP] period");
+ for (i = 0; i < VPP_DBG_PERIOD_NUM; i++) {
+ DPRINT(" %d", p->period_us[i]);
+ sum += p->period_us[i];
+ }
+ DPRINT(",sum %d\n", sum);
+ }
+ return tm_usec;
+}
+
+void vpp_dbg_timer(vpp_dbg_timer_t *p, char *str, int cmd)
+{
+ struct timeval tv;
+ int tm_usec = 0;
+ int initial = 0;
+
+ if (p == 0)
+ return;
+
+ if (p->reset == 0) { /* default */
+ p->reset = 150;
+ initial = 1;
+ }
+
+ do_gettimeofday(&tv);
+ if (p->pre_tv.tv_sec)
+ tm_usec = (tv.tv_sec == p->pre_tv.tv_sec) ?
+ (tv.tv_usec - p->pre_tv.tv_usec) :
+ (1000000 + tv.tv_usec - p->pre_tv.tv_usec);
+ p->pre_tv = tv;
+ switch (cmd) {
+ case 0: /* initial */
+ initial = 1;
+ break;
+ case 1: /* start */
+ break;
+ case 2: /* end */
+ p->cnt++;
+ p->sum += tm_usec;
+ if (p->min > tm_usec)
+ p->min = tm_usec;
+ if (p->max < tm_usec)
+ p->max = tm_usec;
+ if (p->threshold && (tm_usec >= p->threshold))
+ MSG("%s tmr %d over %d\n", str, tm_usec, p->threshold);
+ if (p->cnt >= p->reset) {
+ int us_1t;
+
+ us_1t = p->sum / p->cnt;
+ MSG("%s(Cnt %d)Sum %d us,Avg %d,Min %d,Max %d,fps %d.%02d\n",
+ str, p->cnt, p->sum, us_1t,
+ p->min, p->max, 1000000 / us_1t,
+ (100000000 / us_1t) % 100);
+ initial = 1;
+ }
+ break;
+ default:
+ break;
+ }
+
+ if (initial) {
+ p->cnt = 0;
+ p->sum = 0;
+ p->min = ~0;
+ p->max = 0;
+ }
+}
+#endif
+#else
+void vpp_dbg_show(int level, int tmr, char *str) {}
+static void vpp_dbg_show_val1(int level, int tmr, char *str, int val) {}
+void vpp_dbg_wait(char *str) {}
+#endif
+
+#if 0
+static void load_regs(struct pt_regs *ptr)
+{
+ asm volatile(
+ "stmia %0, {r0 - r15}\n\t"
+ :
+ : "r" (ptr)
+ : "memory"
+ );
+}
+
+void vpp_dbg_back_trace(void)
+{
+ struct pt_regs *ptr;
+ unsigned int fp;
+ unsigned long flags;
+
+ ptr = kmalloc(sizeof(struct pt_regs), GFP_KERNEL);
+
+ local_irq_save(flags);
+
+ MSG("\n\nstart back trace...\n");
+ load_regs(ptr);
+ fp = ptr->ARM_fp;
+ c_backtrace(fp, 0x1f);
+ MSG("back trace end...\n\n");
+
+ local_irq_restore(flags);
+
+ kfree(ptr);
+}
+EXPORT_SYMBOL(vpp_dbg_back_trace);
+#endif
diff --git a/drivers/video/wmt/vpp-osif.h b/drivers/video/wmt/vpp-osif.h
new file mode 100644
index 00000000..988766fd
--- /dev/null
+++ b/drivers/video/wmt/vpp-osif.h
@@ -0,0 +1,247 @@
+/*++
+ * linux/drivers/video/wmt/vpp-osif.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#ifndef VPP_OSIF_H
+#define VPP_OSIF_H
+
+/*-------------------- DEPENDENCY -------------------------------------*/
+#ifdef CFG_LOADER
+ #define CONFIG_UBOOT
+
+ #ifdef __KERNEL__
+ #undef __KERNEL__
+ #endif
+#elif defined(__KERNEL__)
+ #define CONFIG_KERNEL
+#else
+ #define CONFIG_VPOST
+#endif
+
+/* -------------------------------------------------- */
+#ifdef DEBUG
+ #define DBG_MSG(fmt, args...) DPRINT("{%s} " fmt, __func__, ## args)
+#else
+ #define DBG_MSG(fmt, args...)
+#endif
+
+#ifdef DEBUG_DETAIL
+#define DBG_DETAIL(fmt, args...) DPRINT("{%s} " fmt, __func__, ## args)
+#else
+#define DBG_DETAIL(fmt, args...)
+#endif
+#define MSG(fmt, args...) DPRINT("" fmt, ## args)
+#define DBG_ERR(fmt, args...) \
+ DPRINT(KERN_ERR "*E* {%s} " fmt, __func__, ## args)
+#define DMSG(fmt, args...) \
+ DPRINT("{%s,%d} " fmt, __func__, __LINE__, ## args)
+
+#ifdef DEBUG
+#define DBGMSG(fmt, args...) DPRINT("{%s} " fmt, __func__, ## args)
+#else
+#define DBGMSG(fmt, args...)
+#endif
+
+#if 0 /* disable all msg */
+#undef MSG
+#undef DBGMSG
+
+#define MSG(fmt, args...)
+#define DBGMSG(fmt, args...)
+#endif
+
+/* -------------------------------------------------- */
+#ifdef CONFIG_KERNEL
+#include <linux/version.h>
+#include <linux/kernel.h>
+#include <linux/string.h>
+#include <linux/delay.h>
+#include <linux/timer.h>
+#include <linux/interrupt.h>
+#include <linux/spinlock.h>
+#include <linux/slab.h>
+#include <linux/module.h>
+#include <linux/i2c.h>
+#include <linux/wmt-mb.h>
+#include <linux/netlink.h>
+#include <linux/switch.h>
+#include <net/sock.h>
+#include <mach/hardware.h>
+#include <mach/wmt_mmap.h>
+
+#define SA_INTERRUPT IRQF_DISABLED
+#endif
+
+/* -------------------------------------------------- */
+#ifdef CONFIG_UBOOT
+#define CONFIG_WMT_EDID
+#define CONFIG_WMT_EXT_DEV_PLUG_DISABLE
+
+#include <linux/types.h>
+#include "../../board/wmt/include/common_def.h"
+#include <common.h>
+#include <malloc.h>
+#include "hw_devices.h"
+#include "hw/wmt_mmap.h"
+#include "hw/wmt-pwm.h"
+#include "hw/wmt-ost.h"
+#include "hw/wmt_gpio.h"
+#include "wmt_display.h"
+#include "../../board/wmt/include/wmt_clk.h"
+#include "../../board/wmt/include/i2c.h"
+
+#define abs(a) ((a >= 0) ? a : (-1 * a))
+#endif
+
+/* -------------------------------------------------- */
+#ifdef CONFIG_VPOST
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "global.h"
+#define __ASM_ARCH_HARDWARE_H
+#include "../include/wmt_mmap.h"
+#include "../pmc/wmt_clk.h"
+#include "../i2c/i2c.h"
+#include "linux/wmt-mb.h"
+#endif
+
+/* following is the C++ header */
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*-------------------- EXPORTED PRIVATE CONSTANTS ----------------------------*/
+#ifdef CONFIG_KERNEL
+#define THE_MB_USER "VPP-MB"
+#define DPRINT printk
+/* -------------------------------------------------- */
+#endif
+
+#ifdef CONFIG_UBOOT
+#define IRQ_GPIO 0
+
+#define mdelay(x) wmt_delayus(1000 * x)
+#define udelay(x) wmt_delayus(x)
+#define REG32_VAL(addr) (*(volatile unsigned int *)(addr))
+#define REG16_VAL(addr) (*(volatile unsigned short *)(addr))
+#define REG8_VAL(addr) (*(volatile unsigned char *)(addr))
+
+#define mb_alloc(a) malloc(a)
+#define kmalloc(a, b) malloc(a)
+#define kfree(a) free(a)
+#define GFP_KERNEL 0
+#define module_init(a)
+
+#define DPRINT printf
+#define mb_phys_to_virt(a) (a)
+#define mb_virt_to_phys(a) (a)
+#define EXPORT_SYMBOL(a)
+
+#define IRQF_SHARED 0
+#define IRQF_DISABLED 0
+#define SA_INTERRUPT 0
+
+#define KERN_ALERT
+#define KERN_ERR
+#define KERN_DEBUG
+#define KERN_WARNING
+#define KERN_INFO
+
+#define printk printf
+#define BIT(x) (1 << x)
+#endif
+
+/*-------------------- EXPORTED PRIVATE TYPES---------------------------------*/
+/* typedef void hdmi_xxx_t; *//*Example*/
+
+/*-------------------- EXPORTED PRIVATE VARIABLES ----------------------------*/
+#ifdef VPP_OSIF_C
+#define EXTERN
+#else
+#define EXTERN extern
+#endif /* ifdef VPP_OSIF_C */
+
+/* EXTERN int hdmi_xxx; *//*Example*/
+
+#undef EXTERN
+
+/*--------------------- EXPORTED PRIVATE MACROS ------------------------------*/
+/* #define HDMI_XXX_YYY xxxx *//*Example*/
+
+/*--------------------- EXPORTED PRIVATE FUNCTIONS --------------------------*/
+/* extern void hdmi_xxx(void); *//*Example*/
+#ifdef CONFIG_KERNEL
+extern void wmt_i2c_xfer_continue_if(struct i2c_msg *msg,
+ unsigned int num);
+extern void wmt_i2c_xfer_if(struct i2c_msg *msg);
+extern int wmt_i2c_xfer_continue_if_4(struct i2c_msg *msg,
+ unsigned int num, int bus_id);
+#endif
+
+#ifdef CONFIG_UBOOT
+extern int wmt_getsyspara(char *varname, char *varval, int *varlen);
+extern int auto_pll_divisor(enum dev_id dev, enum clk_cmd cmd,
+ int unit, int freq);
+extern struct fb_var_screeninfo vfb_var;
+#endif
+
+#ifdef CONFIG_VPOST
+void vpp_initialization(int FunctionNumber);
+void udelay(int us);
+void mdelay(int ms);
+extern int auto_pll_divisor(enum dev_id dev, enum clk_cmd cmd,
+ int unit, int freq);
+extern void vpp_post_delay(U32 tmr);
+extern int get_num(unsigned int min, unsigned int max,
+ char *message, unsigned int retry);
+#endif
+
+int wmt_getsyspara(char *varname, char *varval, int *varlen);
+int vpp_request_irq(unsigned int irq_no, void *routine,
+ unsigned int flags, char *name, void *arg);
+void vpp_free_irq(unsigned int irq_no, void *arg);
+int vpp_parse_param(char *buf, unsigned int *param,
+ int cnt, unsigned int hex_mask);
+void vpp_lock_l(void);
+void vpp_unlock(void);
+
+#define vpp_lock() vpp_lock_l(); \
+ /* DPRINT("vpp_lock %s %d\n",__FUNCTION__,__LINE__); */
+
+int vpp_i2c_write(int id, unsigned int addr, unsigned int index,
+ char *pdata, int len);
+int vpp_i2c_read(int id, unsigned int addr, unsigned int index,
+ char *pdata, int len);
+int vpp_i2c_enhanced_ddc_read(int id, unsigned int addr,
+ unsigned int index, char *pdata, int len);
+int vpp_i2c_init(int i2c_id, unsigned short addr);
+int vpp_i2c_release(void);
+void vpp_set_clock_enable(enum dev_id dev, int enable, int force);
+void vpp_udelay(unsigned int us);
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* VPP_OSIF_H */
+
diff --git a/drivers/video/wmt/vpp.c b/drivers/video/wmt/vpp.c
new file mode 100644
index 00000000..8b0865ac
--- /dev/null
+++ b/drivers/video/wmt/vpp.c
@@ -0,0 +1,1674 @@
+/*++
+ * linux/drivers/video/wmt/vpp.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define VPP_C
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+#include "vpp.h"
+
+vpp_mod_base_t *vpp_mod_base_list[VPP_MOD_MAX];
+
+unsigned int vpp_get_chipid(void)
+{
+ /* byte 3,2: chip id, byte 1:ver id, byte 0:sub id */
+ /* ex: 0x34290101 (0x3429 A0), 0x34290102 (0x3429 A1) */
+ return REG32_VAL(SYSTEM_CFG_CTRL_BASE_ADDR);
+}
+
+__inline__ void vpp_cache_sync(void)
+{
+ /* TODO */
+}
+
+void vpp_set_clock_enable(enum dev_id dev, int enable, int force)
+{
+#ifdef CONFIG_VPP_DISABLE_PM
+ return;
+#else
+ int cnt;
+
+ do {
+ cnt = auto_pll_divisor(dev,
+ (enable) ? CLK_ENABLE : CLK_DISABLE, 0, 0);
+ if (enable) {
+ if (cnt)
+ break;
+ } else {
+ if (cnt == 0)
+ break;
+ }
+ } while (force);
+/* MSG("%s(%d,%d,%d)\n", __FUNCTION__, dev, enable, cnt); */
+#endif
+}
+
+/*----------------------- vpp module --------------------------------------*/
+void vpp_mod_unregister(vpp_mod_t mod)
+{
+ vpp_mod_base_t *mod_p;
+
+ if (mod >= VPP_MOD_MAX)
+ return;
+
+ mod_p = vpp_mod_base_list[mod];
+ if (!mod_p)
+ return;
+
+ kfree(mod_p->fb_p);
+ kfree(mod_p);
+ vpp_mod_base_list[mod] = 0;
+}
+
+vpp_mod_base_t *vpp_mod_register(vpp_mod_t mod,
+ int size, unsigned int flags)
+{
+ vpp_mod_base_t *mod_p;
+
+ if (mod >= VPP_MOD_MAX)
+ return 0;
+
+ if (vpp_mod_base_list[mod])
+ vpp_mod_unregister(mod);
+
+ mod_p = kmalloc(size, GFP_KERNEL);
+ if (!mod_p)
+ return 0;
+
+ vpp_mod_base_list[mod] = mod_p;
+ memset(mod_p, 0, size);
+ mod_p->mod = mod;
+
+ if (flags & VPP_MOD_FLAG_FRAMEBUF) {
+ mod_p->fb_p = kmalloc(sizeof(vpp_fb_base_t), GFP_KERNEL);
+ if (!mod_p->fb_p)
+ goto error;
+ memset(mod_p->fb_p, 0, sizeof(vpp_fb_base_t));
+ }
+ DBG_DETAIL(" %d,0x%x,0x%x\n", mod, (int)mod_p, (int)mod_p->fb_p);
+ return mod_p;
+error:
+ vpp_mod_unregister(mod);
+ DPRINT("vpp mod register NG %d\n", mod);
+ return 0;
+}
+
+vpp_mod_base_t *vpp_mod_get_base(vpp_mod_t mod)
+{
+ if (mod >= VPP_MOD_MAX)
+ return 0;
+ return vpp_mod_base_list[mod];
+}
+
+vpp_fb_base_t *vpp_mod_get_fb_base(vpp_mod_t mod)
+{
+ vpp_mod_base_t *mod_p;
+ mod_p = vpp_mod_get_base(mod);
+ if (mod_p)
+ return mod_p->fb_p;
+ return 0;
+}
+
+vdo_framebuf_t *vpp_mod_get_framebuf(vpp_mod_t mod)
+{
+ vpp_mod_base_t *mod_p;
+
+ mod_p = vpp_mod_get_base(mod);
+ if (mod_p && mod_p->fb_p)
+ return &mod_p->fb_p->fb;
+ return 0;
+}
+
+void vpp_mod_set_clock(vpp_mod_t mod, vpp_flag_t enable, int force)
+{
+ vpp_mod_base_t *base;
+ enum dev_id pll_dev;
+ int cur_sts;
+ int ret;
+
+#ifdef CONFIG_VPP_DISABLE_PM
+ return;
+#endif
+
+ base = vpp_mod_get_base(mod);
+ if (base == 0)
+ return;
+
+ pll_dev = (base->pm & 0xFF);
+ if (pll_dev == 0)
+ return;
+
+ enable = (enable) ? VPP_FLAG_ENABLE : VPP_FLAG_DISABLE;
+ if (force) {
+ ret = auto_pll_divisor(pll_dev,
+ (enable) ? CLK_ENABLE : CLK_DISABLE, 0, 0);
+ DBG_DETAIL("[VPP] clk force(%s,%d),ret %d\n",
+ vpp_mod_str[mod], enable, ret);
+ return;
+ }
+
+ cur_sts = (base->pm & VPP_MOD_CLK_ON) ? 1 : 0;
+ if (cur_sts != enable) {
+ ret = auto_pll_divisor(pll_dev,
+ (enable) ? CLK_ENABLE : CLK_DISABLE, 0, 0);
+ base->pm = (enable) ? (base->pm | VPP_MOD_CLK_ON)
+ : (base->pm & ~VPP_MOD_CLK_ON);
+ DBG_MSG("[VPP] clk enable(%s,%d,cur %d),ret %d\n",
+ vpp_mod_str[mod], enable, cur_sts, ret);
+ }
+}
+
+vpp_display_format_t vpp_get_fb_field(vdo_framebuf_t *fb)
+{
+ if (fb->flag & VDO_FLAG_INTERLACE)
+ return VPP_DISP_FMT_FIELD;
+ return VPP_DISP_FMT_FRAME;
+}
+
+unsigned int vpp_get_base_clock(vpp_mod_t mod)
+{
+ unsigned int clock = 0;
+
+ switch (mod) {
+ default:
+ clock = auto_pll_divisor(DEV_VPP, GET_FREQ, 0, 0);
+ break;
+ case VPP_MOD_GOVRH:
+ clock = (p_govrh->vo_clock == 0) ?
+ auto_pll_divisor(DEV_HDMILVDS, GET_FREQ, 0, 0) :
+ p_govrh->vo_clock;
+ break;
+ case VPP_MOD_GOVRH2:
+ clock = (p_govrh2->vo_clock == 0) ?
+ auto_pll_divisor(DEV_DVO, GET_FREQ, 0, 0) :
+ p_govrh2->vo_clock;
+ break;
+ }
+ DBG_DETAIL("%d %d\n", mod, clock);
+ return clock;
+}
+
+#if 1
+void vpp_show_timing(char *str, struct fb_videomode *vmode,
+ vpp_clock_t *clk)
+{
+ DPRINT("----- %s timing -----\n", str);
+ if (vmode) {
+ DPRINT("res(%d,%d),fps %d\n",
+ vmode->xres, vmode->yres, vmode->refresh);
+ DPRINT("pixclk %d(%d),hsync %d,vsync %d\n", vmode->pixclock,
+ (int)(PICOS2KHZ(vmode->pixclock) * 1000),
+ vmode->hsync_len, vmode->vsync_len);
+ DPRINT("left %d,right %d,upper %d,lower %d\n",
+ vmode->left_margin, vmode->right_margin,
+ vmode->upper_margin, vmode->lower_margin);
+ DPRINT("vmode 0x%x,sync 0x%x\n", vmode->vmode, vmode->sync);
+ }
+
+ if (clk) {
+ DPRINT("H beg %d,end %d,total %d\n", clk->begin_pixel_of_active,
+ clk->end_pixel_of_active, clk->total_pixel_of_line);
+ DPRINT("V beg %d,end %d,total %d\n", clk->begin_line_of_active,
+ clk->end_line_of_active, clk->total_line_of_frame);
+ DPRINT("Hsync %d, Vsync %d\n", clk->hsync, clk->vsync);
+ DPRINT("VBIE %d,PVBI %d\n", clk->line_number_between_VBIS_VBIE,
+ clk->line_number_between_PVBI_VBIS);
+ }
+ DPRINT("-----------------------\n");
+}
+
+void vpp_show_framebuf(char *str, vdo_framebuf_t *fb)
+{
+ if (fb == 0)
+ return;
+ DPRINT("----- %s framebuf -----\n", str);
+ DPRINT("Y addr 0x%x, size %d\n", fb->y_addr, fb->y_size);
+ DPRINT("C addr 0x%x, size %d\n", fb->c_addr, fb->c_size);
+ DPRINT("W %d, H %d, FB W %d, H %d\n", fb->img_w, fb->img_h,
+ fb->fb_w, fb->fb_h);
+ DPRINT("bpp %d, color fmt %s\n", fb->bpp, vpp_colfmt_str[fb->col_fmt]);
+ DPRINT("H crop %d, V crop %d, flag 0x%x\n",
+ fb->h_crop, fb->v_crop, fb->flag);
+ DPRINT("-----------------------\n");
+}
+
+void vpp_show_videomode(char *str, struct fb_videomode *v)
+{
+ if (v == 0)
+ return;
+ DPRINT("----- %s videomode -----\n", str);
+ DPRINT("%dx%d@%d,%d\n", v->xres, v->yres, v->refresh, v->pixclock);
+ DPRINT("h sync %d,bp %d,fp %d\n", v->hsync_len,
+ v->left_margin, v->right_margin);
+ DPRINT("v sync %d,bp %d,fp %d\n", v->vsync_len,
+ v->upper_margin, v->lower_margin);
+ DPRINT("sync 0x%x,vmode 0x%x,flag 0x%x\n", v->sync, v->vmode, v->flag);
+ DPRINT("hsync %s,vsync %s\n",
+ (v->sync & FB_SYNC_HOR_HIGH_ACT) ? "hi" : "lo",
+ (v->sync & FB_SYNC_VERT_HIGH_ACT) ? "hi" : "lo");
+ DPRINT("interlace %d,double %d\n",
+ (v->vmode & FB_VMODE_INTERLACED) ? 1 : 0,
+ (v->vmode & FB_VMODE_DOUBLE) ? 1 : 0);
+ DPRINT("-----------------------\n");
+}
+#endif
+
+vpp_csc_t vpp_check_csc_mode(vpp_csc_t mode, vdo_color_fmt src_fmt,
+ vdo_color_fmt dst_fmt, unsigned int flags)
+{
+ if (mode >= VPP_CSC_MAX)
+ return VPP_CSC_BYPASS;
+
+ mode = (mode >= VPP_CSC_RGB2YUV_MIN) ?
+ (mode - VPP_CSC_RGB2YUV_MIN) : mode;
+ if (src_fmt >= VDO_COL_FMT_ARGB) {
+ mode = VPP_CSC_RGB2YUV_MIN + mode;
+ src_fmt = VDO_COL_FMT_ARGB;
+ } else {
+ src_fmt = VDO_COL_FMT_YUV444;
+ }
+ dst_fmt = (dst_fmt >= VDO_COL_FMT_ARGB) ?
+ VDO_COL_FMT_ARGB : VDO_COL_FMT_YUV444;
+ if (flags == 0)
+ mode = (src_fmt != dst_fmt) ? mode : VPP_CSC_BYPASS;
+ return mode;
+}
+
+int vpp_get_gcd(int A, int B)
+{
+ while (A != B) {
+ if (A > B)
+ A = A - B;
+ else
+ B = B - A;
+ }
+ return A;
+}
+
+int vpp_set_recursive_scale(vdo_framebuf_t *src_fb,
+ vdo_framebuf_t *dst_fb)
+{
+#ifdef WMT_FTBLK_SCL
+ int ret;
+
+ ret = p_scl->scale(src_fb, dst_fb);
+ return ret;
+#else
+ DBG_ERR("No scale\n");
+ return 0;
+#endif
+}
+
+void vpp_reg_dump(unsigned int addr, int size)
+{
+ int i;
+
+ for (i = 0; i < size; i += 16) {
+ DPRINT("0x%8x : 0x%08x 0x%08x 0x%08x 0x%08x\n",
+ addr + i, vppif_reg32_in(addr + i),
+ vppif_reg32_in(addr + i + 4),
+ vppif_reg32_in(addr + i + 8),
+ vppif_reg32_in(addr + i + 12));
+ }
+} /* End of vpp_reg_dump */
+
+unsigned int vpp_convert_colfmt(int yuv2rgb, unsigned int data)
+{
+ unsigned int r, g, b;
+ unsigned int y, u, v;
+ unsigned int alpha;
+
+ alpha = data & 0xff000000;
+ if (yuv2rgb) {
+ y = (data & 0xff0000) >> 16;
+ u = (data & 0xff00) >> 8;
+ v = (data & 0xff) >> 0;
+
+ r = ((1000 * y) + 1402 * (v - 128)) / 1000;
+ if (r > 0xFF)
+ r = 0xFF;
+ g = ((100000 * y) - (71414 * (v - 128))
+ - (34414 * (u - 128))) / 100000;
+ if (g > 0xFF)
+ g = 0xFF;
+ b = ((1000 * y) + (1772 * (u - 128))) / 1000;
+ if (b > 0xFF)
+ b = 0xFF;
+
+ data = ((r << 16) + (g << 8) + b);
+ } else {
+ r = (data & 0xff0000) >> 16;
+ g = (data & 0xff00) >> 8;
+ b = (data & 0xff) >> 0;
+
+ y = ((2990 * r) + (5870 * g) + (1440 * b)) / 10000;
+ if (y > 0xFF)
+ y = 0xFF;
+ u = (1280000 - (1687 * r) - (3313 * g) + (5000 * b)) / 10000;
+ if (u > 0xFF)
+ u = 0xFF;
+ v = (1280000 + (5000 * r) - (4187 * g) - (813 * b)) / 10000;
+ if (v > 0xFF)
+ v = 0xFF;
+
+ data = ((y << 16) + (v << 8) + u);
+ }
+ data = data + alpha;
+ return data;
+}
+
+unsigned int *vpp_backup_reg(unsigned int addr, unsigned int size)
+{
+ unsigned int *ptr;
+ int i;
+
+ size += 4;
+ ptr = kmalloc(size, GFP_KERNEL);
+ if (ptr == 0) {
+ DPRINT("[VPP] *E* malloc backup fail\n");
+ return 0;
+ }
+
+ for (i = 0; i < size; i += 4)
+ ptr[i / 4] = REG32_VAL(addr + i);
+ return ptr;
+} /* End of vpp_backup_reg */
+
+int vpp_restore_reg(unsigned int addr, unsigned int size,
+ unsigned int *reg_ptr)
+{
+ int i;
+
+ if (reg_ptr == NULL)
+ return 0;
+
+ size += 4;
+ for (i = 0; i < size; i += 4)
+ REG32_VAL(addr + i) = reg_ptr[i / 4];
+ kfree(reg_ptr);
+ reg_ptr = 0;
+ return 0;
+} /* End of vpp_restore_reg */
+
+void vpp_get_sys_parameter(void)
+{
+#ifndef CONFIG_VPOST
+ char buf[40];
+ int varlen = 40;
+#else
+ struct env_para_def param;
+#endif
+
+ /* vpp attribute by default */
+ g_vpp.dbg_msg_level = 0;
+ g_vpp.hdmi_audio_interface = VPP_HDMI_AUDIO_SPDIF;
+ g_vpp.hdmi_cp_enable = 1;
+
+#if 0
+ if (wmt_getsyspara("wmt.display.direct_path", buf, &varlen) == 0) {
+ sscanf(buf, "%d", &g_vpp.direct_path);
+ DPRINT("[VPP] direct path %d\n", g_vpp.direct_path);
+ }
+#endif
+
+#ifndef CONFIG_VPOST
+ if (wmt_getsyspara("wmt.display.hdmi_audio_inf", buf, &varlen) == 0) {
+ if (memcmp(buf, "i2s", 3) == 0)
+ g_vpp.hdmi_audio_interface = VPP_HDMI_AUDIO_I2S;
+ else if (memcmp(buf, "spdif", 5) == 0)
+ g_vpp.hdmi_audio_interface = VPP_HDMI_AUDIO_SPDIF;
+ }
+
+ if (wmt_getsyspara("wmt.display.hdmi.vmode", buf, &varlen) == 0) {
+ if (memcmp(buf, "720p", 4) == 0)
+ g_vpp.hdmi_video_mode = 720;
+ else if (memcmp(buf, "1080p", 5) == 0)
+ g_vpp.hdmi_video_mode = 1080;
+ else
+ g_vpp.hdmi_video_mode = 0;
+ DPRINT("[VPP] HDMI video mode %d\n", g_vpp.hdmi_video_mode);
+ }
+
+ g_vpp.mb_colfmt = VPP_UBOOT_COLFMT;
+ /* [uboot parameter] fb param : no:xresx:yres:xoffset:yoffset */
+ if (wmt_getsyspara("wmt.gralloc.param", buf, &varlen) == 0) {
+ unsigned int parm[1];
+
+ vpp_parse_param(buf, (unsigned int *)parm, 1, 0x1);
+ if (parm[0] == 32)
+ g_vpp.mb_colfmt = VDO_COL_FMT_ARGB;
+ MSG("mb colfmt : %s,%s\n", buf,
+ vpp_colfmt_str[g_vpp.mb_colfmt]);
+ }
+ p_govrh->fb_p->fb.col_fmt = g_vpp.mb_colfmt;
+ p_govrh2->fb_p->fb.col_fmt = g_vpp.mb_colfmt;
+
+ /* [uboot parameter] dual display : 0-single display, 1-dual display */
+ g_vpp.dual_display = 1;
+ if (wmt_getsyspara("wmt.display.dual", buf, &varlen) == 0) {
+ unsigned int parm[1];
+
+ MSG("display dual : %s\n", buf);
+ vpp_parse_param(buf, (unsigned int *)parm, 1, 0);
+ g_vpp.dual_display = parm[0];
+ }
+
+ if (g_vpp.dual_display == 0)
+ g_vpp.alloc_framebuf = 0;
+
+ if (wmt_getsyspara("wmt.display.hdmi", buf, &varlen) == 0) {
+ unsigned int parm[1];
+
+ MSG("hdmi sp mode : %s\n", buf);
+ vpp_parse_param(buf, (unsigned int *)parm, 1, 0);
+ g_vpp.hdmi_sp_mode = (parm[0]) ? 1 : 0;
+ }
+
+ if (wmt_getsyspara("wmt.hdmi.disable", buf, &varlen) == 0)
+ g_vpp.hdmi_disable = 1;
+#else
+ if (env_read_para("wmt.display.hdmi", &param) == 0) {
+ g_vpp.hdmi_sp_mode = strtoul(param.value, 0, 16);
+ free(param.value);
+ }
+#endif
+} /* End of vpp_get_sys_parameter */
+
+void vpp_init(void)
+{
+ vpp_mod_base_t *mod_p;
+ unsigned int mod_mask;
+ int i;
+ int no;
+
+ vpp_get_sys_parameter();
+
+ auto_pll_divisor(DEV_NA12, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_VPP, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_HDCE, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_HDMII2C, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_HDMI, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_GOVRHD, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_DVO, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_LVDS, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_HDMILVDS, CLK_ENABLE, 0, 0);
+ auto_pll_divisor(DEV_SCL444U, CLK_ENABLE, 0, 0);
+#ifdef CONFIG_KERNEL
+ if (1) {
+ if (govrh_get_MIF_enable(p_govrh))
+ g_vpp.govrh_preinit = 1;
+ if (govrh_get_MIF_enable(p_govrh2))
+ g_vpp.govrh_preinit = 1;
+ MSG("[VPP] govrh preinit %d\n", g_vpp.govrh_preinit);
+ }
+#endif
+
+#ifndef CONFIG_VPP_DYNAMIC_ALLOC
+ if (g_vpp.alloc_framebuf)
+ g_vpp.alloc_framebuf(VPP_HD_MAX_RESX, VPP_HD_MAX_RESY);
+#endif
+
+ /* init video out module first */
+ if (g_vpp.govrh_preinit == 0) {
+ mod_mask = BIT(VPP_MOD_GOVRH2) | BIT(VPP_MOD_GOVRH)
+ | BIT(VPP_MOD_DISP) | BIT(VPP_MOD_LCDC);
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ if (!(mod_mask & (0x01 << i)))
+ continue;
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->init)
+ mod_p->init(mod_p);
+ }
+ }
+
+#ifndef CONFIG_UBOOT
+ /* init other module */
+ mod_mask = BIT(VPP_MOD_GOVW) | BIT(VPP_MOD_GOVM) | BIT(VPP_MOD_SCL)
+ | BIT(VPP_MOD_SCLW) | BIT(VPP_MOD_VPU) | BIT(VPP_MOD_VPUW)
+ | BIT(VPP_MOD_PIP) | BIT(VPP_MOD_VPPM);
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ if (!(mod_mask & (0x01 << i)))
+ continue;
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->init)
+ mod_p->init(mod_p);
+ }
+#endif
+
+#ifdef WMT_FTBLK_LVDS
+ if (!g_vpp.govrh_preinit)
+ lvds_init();
+#endif
+#ifdef WMT_FTBLK_VOUT_HDMI
+ hdmi_init();
+#endif
+
+ vpp_set_clock_enable(DEV_SCL444U, 0, 1);
+
+#ifndef CONFIG_VPOST
+ /* init vout device & get default resolution */
+ vout_init();
+#endif
+
+#ifdef CONFIG_KERNEL
+ no = (g_vpp.virtual_display_mode == 1) ? 1 : 0;
+ if (g_vpp.govrh_preinit) {
+ struct fb_videomode vmode;
+ vout_info_t *info;
+ govrh_mod_t *govr;
+
+ info = vout_get_info_entry(no);
+ govr = vout_info_get_govr(no);
+ govrh_get_videomode(govr, &vmode);
+ govrh_get_framebuffer(govr, &info->fb);
+ g_vpp.govrh_init_yres = vmode.yres;
+ if ((info->resx != vmode.xres) || (info->resy != vmode.yres)) {
+ g_vpp.govrh_preinit = 0;
+ DPRINT("preinit not match (%dx%d) --> (%dx%d)\n",
+ vmode.xres, vmode.yres, info->resx, info->resy);
+ if (g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ if(!hdmi_get_plugin()) {
+ vout_t *vo = vout_get_entry(VPP_VOUT_NUM_DVI);
+ if(vo->dev && !strcmp(vo->dev->name, "CS8556") && vo->dev->init) {
+ vo->dev->init(vo);
+ }
+ }
+ }
+ }
+ }
+
+ if (!g_vpp.govrh_preinit) {
+ struct fb_videomode vmode;
+ vout_info_t *info;
+
+ info = vout_get_info_entry(no);
+ memset(&vmode, 0, sizeof(struct fb_videomode));
+ vmode.xres = info->resx;
+ vmode.yres = info->resy;
+ vmode.refresh = info->fps;
+ if (vout_find_match_mode(no, &vmode, 1) == 0)
+ vout_config(VPP_VOUT_ALL, info, &vmode);
+ }
+#endif
+ vpp_set_clock_enable(DEV_HDMII2C, 0, 0);
+ vpp_set_clock_enable(DEV_HDMI, 0, 0);
+ vpp_set_clock_enable(DEV_HDCE, 0, 0);
+ vpp_set_clock_enable(DEV_LVDS, 0, 0);
+}
+
+void vpp_get_colfmt_bpp(vdo_color_fmt colfmt, int *y_bpp, int *c_bpp)
+{
+ switch (colfmt) {
+ case VDO_COL_FMT_YUV420:
+ *y_bpp = 8;
+ *c_bpp = 4;
+ break;
+ case VDO_COL_FMT_YUV422H:
+ case VDO_COL_FMT_YUV422V:
+ *y_bpp = 8;
+ *c_bpp = 8;
+ break;
+ case VDO_COL_FMT_RGB_565:
+ case VDO_COL_FMT_RGB_1555:
+ case VDO_COL_FMT_RGB_5551:
+ *y_bpp = 16;
+ *c_bpp = 0;
+ break;
+ case VDO_COL_FMT_YUV444:
+ *y_bpp = 8;
+ *c_bpp = 16;
+ break;
+ case VDO_COL_FMT_RGB_888:
+ case VDO_COL_FMT_RGB_666:
+ *y_bpp = 24;
+ *c_bpp = 0;
+ break;
+ case VDO_COL_FMT_ARGB:
+ *y_bpp = 32;
+ *c_bpp = 0;
+ break;
+ default:
+ break;
+ }
+}
+
+int vpp_calc_refresh(int pixclk, int xres, int yres)
+{
+ int refresh = 60;
+ int temp;
+
+ temp = xres * yres;
+ if (temp) {
+ refresh = pixclk / temp;
+ if (pixclk % temp)
+ refresh += 1;
+ }
+ return refresh;
+}
+
+int vpp_calc_align(int value, int align)
+{
+ if (value % align) {
+ value &= ~(align - 1);
+ value += align;
+ }
+ return value;
+}
+
+int vpp_calc_fb_width(vdo_color_fmt colfmt, int width)
+{
+ int y_bpp, c_bpp;
+
+ vpp_get_colfmt_bpp(colfmt, &y_bpp, &c_bpp);
+ return vpp_calc_align(width, VPP_FB_WIDTH_ALIGN / (y_bpp / 8));
+}
+
+void vpp_set_NA12_hiprio(int type)
+{
+#if 0
+ static int reg1, reg2;
+
+ switch (type) {
+ case 0: /* restore NA12 priority */
+ vppif_reg32_out(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0x8, reg1);
+ vppif_reg32_out(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0xC, reg2);
+ break;
+ case 1: /* set NA12 to high priority */
+ reg1 = vppif_reg32_in(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0x8);
+ reg2 = vppif_reg32_in(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0xC);
+ vppif_reg32_out(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0x8, 0x600000);
+ vppif_reg32_out(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0xC, 0x0ff00000);
+ break;
+ case 2:
+ reg1 = vppif_reg32_in(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0x8);
+ reg2 = vppif_reg32_in(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0xC);
+ vppif_reg32_out(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0x8, 0x20003f);
+ vppif_reg32_out(MEMORY_CTRL_V4_CFG_BASE_ADDR + 0xC, 0x00ffff00);
+ break;
+ default:
+ break;
+ }
+#endif
+}
+
+#ifdef __KERNEL__
+int vpp_set_audio(int format, int sample_rate, int channel)
+{
+ vout_audio_t info;
+
+ DBG_MSG("set audio(fmt %d,rate %d,ch %d)\n",
+ format, sample_rate, channel);
+ info.fmt = format;
+ info.sample_rate = sample_rate;
+ info.channel = channel;
+ return vout_set_audio(&info);
+}
+
+static DEFINE_SEMAPHORE(vpp_sem);
+static DEFINE_SEMAPHORE(vpp_sem2);
+void vpp_set_mutex(int idx, int lock)
+{
+ struct semaphore *sem;
+
+ sem = ((g_vpp.dual_display == 0) || (idx == 0)) ? &vpp_sem : &vpp_sem2;
+ if (lock)
+ down(sem);
+ else
+ up(sem);
+}
+
+void vpp_free_framebuffer(void)
+{
+ if (g_vpp.mb[0] == 0)
+ return;
+ MSG("mb free 0x%x\n", g_vpp.mb[0]);
+ mb_free(g_vpp.mb[0]);
+ vpp_lock();
+ g_vpp.mb[0] = 0;
+ vpp_unlock();
+}
+
+int vpp_alloc_framebuffer(unsigned int resx, unsigned int resy)
+{
+ unsigned int y_size;
+ unsigned int fb_size;
+ unsigned int colfmt;
+ int y_bpp, c_bpp;
+ int i;
+
+#ifdef CONFIG_VPP_DYNAMIC_ALLOC
+ if (g_vpp.mb[0]) {
+ vpp_free_framebuffer();
+ }
+#endif
+
+ if ((resx == 0) && (resy == 0)) {
+ return -1;
+ }
+
+ /* alloc govw & govrh frame buffer */
+ if (g_vpp.mb[0] == 0) {
+ unsigned int mb_resx, mb_resy;
+ int fb_num;
+ unsigned int phy_base;
+
+#ifdef CONFIG_VPP_DYNAMIC_ALLOC
+ mb_resx = resx;
+ mb_resy = resy;
+ colfmt = g_vpp.mb_colfmt;
+ vpp_get_colfmt_bpp(colfmt, &y_bpp, &c_bpp);
+ fb_num = VPP_MB_ALLOC_NUM;
+#else
+ char buf[100];
+ int varlen = 100;
+
+ if (wmt_getsyspara("wmt.display.mb", (unsigned char *)buf,
+ &varlen) == 0) {
+ unsigned int parm[10];
+
+ vpp_parse_param(buf, (unsigned int *)parm, 4, 0);
+ MSG("boot parm mb (%d,%d),bpp %d,fb %d\n",
+ parm[0], parm[1], parm[2], parm[3]);
+ mb_resx = parm[0];
+ mb_resy = parm[1];
+ y_bpp = parm[2] * 8;
+ c_bpp = 0;
+ fb_num = parm[3];
+ } else {
+ mb_resx = VPP_HD_MAX_RESX;
+ mb_resy = VPP_HD_MAX_RESY;
+ colfmt = g_vpp.mb_colfmt;
+ vpp_get_colfmt_bpp(colfmt, &y_bpp, &c_bpp);
+ fb_num = VPP_MB_ALLOC_NUM;
+ }
+#endif
+ mb_resx = vpp_calc_align(mb_resx,
+ VPP_FB_ADDR_ALIGN / (y_bpp / 8));
+ y_size = mb_resx * mb_resy * y_bpp / 8;
+ fb_size = mb_resx * mb_resy * (y_bpp + c_bpp) / 8;
+ g_vpp.mb_fb_size = fb_size;
+ g_vpp.mb_y_size = y_size;
+ phy_base = mb_alloc(fb_size * fb_num);
+ if (phy_base) {
+ MSG("mb alloc 0x%x,%d\n", phy_base, fb_size * fb_num);
+ for (i = 0; i < fb_num; i++) {
+ g_vpp.mb[i] = (unsigned int)(phy_base +
+ (fb_size * i));
+ MSG("mb 0x%x,fb %d,y %d\n", g_vpp.mb[i],
+ fb_size, y_size);
+ }
+ } else {
+ DBG_ERR("alloc fail\n");
+ return -1;
+ }
+ if (!g_vpp.govrh_preinit) { /* keep uboot logo */
+ memset(mb_phys_to_virt(phy_base), 0, fb_size);
+ MSG("mb clean 0x%x %d\n", phy_base, fb_size);
+ }
+ }
+
+ vpp_lock();
+#ifdef CONFIG_VPP_STREAM_ROTATE
+ {
+ unsigned int resx_mb;
+
+ g_vpp.stream_mb_cnt = VPP_MB_ALLOC_NUM;
+ colfmt = VDO_COL_FMT_YUV422H;
+ vpp_get_colfmt_bpp(colfmt, &y_bpp, &c_bpp);
+ resx_mb = vpp_calc_fb_width(colfmt, resx);
+ y_size = resx_mb * resy * y_bpp / 8;
+ fb_size = vpp_calc_align(resx_mb, 64) * resy *
+ (y_bpp + c_bpp) / 8;
+ for (i = 0; i < g_vpp.stream_mb_cnt; i++) {
+ g_vpp.stream_mb[i] = g_vpp.mb[0] + fb_size * i;
+ }
+ }
+#else
+ /* assign mb to stream mb */
+ {
+ int index = 0, offset = 0;
+ unsigned int size = g_vpp.mb_fb_size;
+ unsigned int resx_fb;
+
+ colfmt = VPP_UBOOT_COLFMT;
+ vpp_get_colfmt_bpp(colfmt, &y_bpp, &c_bpp);
+ resx_fb = vpp_calc_fb_width(colfmt, resx);
+ y_size = resx_fb * resy * y_bpp / 8;
+ fb_size = vpp_calc_align(resx_fb, 64) * resy * (y_bpp + c_bpp) / 8;
+ g_vpp.stream_mb_y_size = y_size;
+ g_vpp.stream_mb_cnt = VPP_STREAM_MB_ALLOC_NUM;
+ for (i = 0; i < VPP_STREAM_MB_ALLOC_NUM; i++) {
+ if (size < fb_size) {
+ index++;
+ if (index >= VPP_MB_ALLOC_NUM) {
+ index = 0;
+ g_vpp.stream_mb_cnt = i;
+ break;
+ }
+ offset = 0;
+ size = g_vpp.mb_fb_size;
+ }
+ g_vpp.stream_mb[i] = g_vpp.mb[index] + offset;
+ size -= fb_size;
+ offset += fb_size;
+ DBG_DETAIL("stream mb %d 0x%x\n", i, g_vpp.stream_mb[i]);
+ }
+ }
+#endif
+ vpp_unlock();
+ return 0;
+} /* End of vpp_alloc_framebuffer */
+
+/*----------------------- vpp mb for stream ---------------------------------*/
+#ifdef CONFIG_VPP_STREAM_CAPTURE
+#ifdef CONFIG_VPP_STREAM_BLOCK
+DECLARE_WAIT_QUEUE_HEAD(vpp_mb_event);
+#endif
+
+unsigned int vpp_mb_get_mask(unsigned int phy)
+{
+ int i;
+ unsigned int mask;
+
+ for (i = 0; i < g_vpp.stream_mb_cnt; i++) {
+ if (g_vpp.stream_mb[i] == phy)
+ break;
+ }
+ if (i >= g_vpp.stream_mb_cnt)
+ return 0;
+ mask = 0x1 << i;
+ return mask;
+}
+
+int vpp_mb_get(unsigned int phy)
+{
+ unsigned int mask;
+ int i, cnt;
+
+#ifdef CONFIG_VPP_STREAM_BLOCK
+ vpp_unlock();
+ i = wait_event_interruptible(vpp_mb_event,
+ (g_vpp.stream_mb_sync_flag != 1));
+ vpp_lock();
+ if (i)
+ return -1;
+#else /* non-block */
+ if (g_vpp.stream_mb_sync_flag) { /* not new mb updated */
+ vpp_dbg_show(VPP_DBGLVL_STREAM, 0,
+ "*W* mb_get addr not update");
+ return -1;
+ }
+#endif
+ g_vpp.stream_mb_sync_flag = 1;
+ for (i = 0, cnt = 0; i < g_vpp.stream_mb_cnt; i++) {
+ if (g_vpp.stream_mb_lock & (0x1 << i))
+ cnt++;
+ }
+
+#if 0
+ if (cnt >= (g_vpp.stream_mb_cnt - 2)) {
+ vpp_dbg_show(VPP_DBGLVL_STREAM, 0, "*W* mb_get addr not free");
+ return -1;
+ }
+#endif
+
+ mask = vpp_mb_get_mask(phy);
+ if (mask == 0) {
+ vpp_dbg_show(VPP_DBGLVL_STREAM, 0, "*W* mb_get invalid addr");
+ return -1;
+ }
+ if (g_vpp.stream_mb_lock & mask) {
+ vpp_dbg_show(VPP_DBGLVL_STREAM, 0, "*W* mb_get lock addr");
+ return -1;
+ }
+ g_vpp.stream_mb_lock |= mask;
+ if (vpp_check_dbg_level(VPP_DBGLVL_STREAM)) {
+ char buf[50];
+
+ sprintf(buf, "stream mb get 0x%x,mask 0x%x(0x%x)",
+ phy, mask, g_vpp.stream_mb_lock);
+ vpp_dbg_show(VPP_DBGLVL_STREAM, 1, buf);
+ }
+ return 0;
+}
+
+int vpp_mb_put(unsigned int phy)
+{
+ unsigned int mask;
+
+ if (phy == 0) {
+ g_vpp.stream_mb_lock = 0;
+ g_vpp.stream_mb_index = 0;
+ return 0;
+ }
+
+ mask = vpp_mb_get_mask(phy);
+ if (mask == 0) {
+ DPRINT("[VPP] *W* mb_put addr 0x%x\n", phy);
+ return 1;
+ }
+ if (!(g_vpp.stream_mb_lock & mask))
+ DPRINT("[VPP] *W* mb_put nonlock addr 0x%x\n", phy);
+ g_vpp.stream_mb_lock &= ~mask;
+ if (vpp_check_dbg_level(VPP_DBGLVL_STREAM)) {
+ char buf[50];
+
+ sprintf(buf, "stream mb put 0x%x,mask 0x%x(0x%x)",
+ phy, mask, g_vpp.stream_mb_lock);
+ vpp_dbg_show(VPP_DBGLVL_STREAM, 2, buf);
+ }
+ return 0;
+}
+
+int vpp_mb_irqproc_sync(int arg)
+{
+ if (!g_vpp.stream_enable)
+ return 0;
+
+ g_vpp.stream_sync_cnt++;
+ if ((g_vpp.stream_sync_cnt % 2) == 0) {
+ g_vpp.stream_mb_sync_flag = 0;
+#ifdef CONFIG_VPP_STREAM_BLOCK
+ wake_up_interruptible(&vpp_mb_event);
+#endif
+ }
+ return 0;
+}
+
+void vpp_mb_scale_bitblit(vdo_framebuf_t *fb)
+{
+ int index = g_vpp.stream_mb_index;
+ vdo_framebuf_t src, dst;
+
+ if (p_scl->scale_complete == 0)
+ return;
+
+#ifdef CONFIG_VPP_STREAM_ROTATE
+ index = g_vpp.stream_mb_index + 1;
+ index = (index >= g_vpp.stream_mb_cnt) ? 0 : index;
+#else
+ do {
+ index++;
+ if (index >= g_vpp.stream_mb_cnt)
+ index = 0;
+
+ if (g_vpp.stream_mb_lock & (0x1 << index))
+ continue;
+ break;
+ } while (1);
+#endif
+
+ g_vpp.stream_mb_index = index;
+ p_scl->scale_sync = 1;
+ src = *fb;
+#ifdef CONFIG_VPP_STREAM_FIX_RESOLUTION
+ dst = g_vpp.stream_fb;
+#else
+ dst = *fb;
+ dst.col_fmt = VDO_COL_FMT_YUV422H;
+ dst.fb_w = vpp_calc_align(dst.fb_w, 64);
+#endif
+ dst.y_addr = g_vpp.stream_mb[index];
+ dst.c_addr = dst.y_addr + (dst.fb_w * dst.img_h);
+ vpp_set_recursive_scale(&src, &dst);
+}
+#endif
+
+/*----------------------- irq proc --------------------------------------*/
+vpp_irqproc_t *vpp_irqproc_array[32];
+struct list_head vpp_irqproc_free_list;
+vpp_proc_t vpp_proc_array[VPP_PROC_NUM];
+static void vpp_irqproc_do_tasklet(unsigned long data);
+
+void vpp_irqproc_init(void)
+{
+ int i;
+
+ INIT_LIST_HEAD(&vpp_irqproc_free_list);
+
+ for (i = 0; i < VPP_PROC_NUM; i++)
+ list_add_tail(&vpp_proc_array[i].list, &vpp_irqproc_free_list);
+}
+
+vpp_irqproc_t *vpp_irqproc_get_entry(vpp_int_t vpp_int)
+{
+ int no;
+
+ if (vpp_int == 0)
+ return 0;
+
+ for (no = 0; no < 32; no++) {
+ if (vpp_int & (0x1 << no))
+ break;
+ }
+
+ if (vpp_irqproc_array[no] == 0) { /* will create in first use */
+ vpp_irqproc_t *irqproc;
+
+ irqproc = kmalloc(sizeof(vpp_irqproc_t), GFP_KERNEL);
+ vpp_irqproc_array[no] = irqproc;
+ INIT_LIST_HEAD(&irqproc->list);
+ tasklet_init(&irqproc->tasklet,
+ vpp_irqproc_do_tasklet, vpp_int);
+ irqproc->ref = 0;
+ }
+ return vpp_irqproc_array[no];
+} /* End of vpp_irqproc_get_entry */
+
+void vpp_irqproc_set_ref(vpp_irqproc_t *irqproc,
+ vpp_int_t type, int enable)
+{
+ if (enable) {
+ irqproc->ref++;
+ if (vppm_get_int_enable(type) == 0)
+ vppm_set_int_enable(1, type);
+ } else {
+ irqproc->ref--;
+ if (irqproc->ref == 0)
+ vppm_set_int_enable(0, type);
+ }
+}
+
+static void vpp_irqproc_do_tasklet
+(
+ unsigned long data /*!<; // tasklet input data */
+)
+{
+ vpp_irqproc_t *irqproc;
+
+ vpp_lock();
+ irqproc = vpp_irqproc_get_entry(data);
+ if (irqproc) {
+ struct list_head *cur;
+ struct list_head *next;
+ vpp_proc_t *entry;
+
+ next = (&irqproc->list)->next;
+ while (next != &irqproc->list) {
+ cur = next;
+ next = cur->next;
+ entry = list_entry(cur, vpp_proc_t, list);
+ if (entry->func) {
+ if (entry->func(entry->arg))
+ continue;
+ }
+
+ if (entry->work_cnt == 0)
+ continue;
+
+ entry->work_cnt--;
+ if (entry->work_cnt == 0) {
+ if (entry->wait_ms == 0)
+ vpp_irqproc_set_ref(irqproc, data, 0);
+ else
+ up(&entry->sem);
+ list_del_init(cur);
+ list_add_tail(&entry->list,
+ &vpp_irqproc_free_list);
+ }
+ }
+ }
+ vpp_unlock();
+} /* End of vpp_irqproc_do_tasklet */
+
+int vpp_irqproc_work(
+ vpp_int_t type, /* interrupt type */
+ int (*func)(void *argc), /* proc function pointer */
+ void *arg, /* proc argument */
+ int wait_ms, /* wait complete timeout (ms) */
+ int work_cnt /* 0 - forever */
+)
+{
+ int ret;
+ vpp_proc_t *entry;
+ struct list_head *ptr;
+ vpp_irqproc_t *irqproc;
+
+ /* DPRINT("[VPP] vpp_irqproc_work(type 0x%x,wait %d)\n",type,wait); */
+
+ if ((vpp_irqproc_free_list.next == 0) ||
+ list_empty(&vpp_irqproc_free_list)) {
+ if (func)
+ func(arg);
+ return 0;
+ }
+
+ ret = 0;
+ vpp_lock();
+
+ ptr = vpp_irqproc_free_list.next;
+ entry = list_entry(ptr, vpp_proc_t, list);
+ list_del_init(ptr);
+ entry->func = func;
+ entry->arg = arg;
+ entry->type = type;
+ entry->wait_ms = wait_ms;
+ entry->work_cnt = work_cnt;
+ sema_init(&entry->sem, 1);
+ down(&entry->sem);
+
+ irqproc = vpp_irqproc_get_entry(type);
+ if (irqproc) {
+ list_add_tail(&entry->list, &irqproc->list);
+ } else {
+ irqproc = vpp_irqproc_array[31];
+ list_add_tail(&entry->list, &irqproc->list);
+ }
+ vpp_irqproc_set_ref(irqproc, type, 1);
+ vpp_unlock();
+
+ if (wait_ms) {
+ unsigned int tmr_cnt;
+
+ tmr_cnt = (wait_ms * HZ) / 1000;
+ ret = down_timeout(&entry->sem, tmr_cnt);
+ if (ret) {
+ DPRINT("*W* vpp_irqproc_work timeout(type 0x%x,%d)\n",
+ type, wait_ms);
+ vpp_lock();
+ list_del_init(ptr);
+ list_add_tail(ptr, &vpp_irqproc_free_list);
+ vpp_unlock();
+ if (func)
+ func(arg);
+ }
+ }
+
+ if ((work_cnt == 0) || (wait_ms == 0)) {
+ /* don't clear ref, forever will do in delete,
+ no wait will do in proc complete */
+ } else {
+ vpp_lock();
+ vpp_irqproc_set_ref(irqproc, type, 0);
+ vpp_unlock();
+ }
+ return ret;
+} /* End of vpp_irqproc_work */
+
+void vpp_irqproc_del_work(
+ vpp_int_t type, /* interrupt type */
+ int (*func)(void *argc) /* proc function pointer */
+)
+{
+ vpp_irqproc_t *irqproc;
+
+ vpp_lock();
+ irqproc = vpp_irqproc_get_entry(type);
+ if (irqproc) {
+ struct list_head *cur;
+ struct list_head *next;
+ vpp_proc_t *entry;
+
+ next = (&irqproc->list)->next;
+ while (next != &irqproc->list) {
+ cur = next;
+ next = cur->next;
+ entry = list_entry(cur, vpp_proc_t, list);
+ if (entry->func == func) {
+ vpp_irqproc_set_ref(irqproc, type, 0);
+ list_del_init(cur);
+ list_add_tail(&entry->list,
+ &vpp_irqproc_free_list);
+ }
+ }
+ }
+ vpp_unlock();
+}
+
+/*----------------------- Linux Netlink --------------------------------------*/
+#ifdef CONFIG_VPP_NOTIFY
+#define VPP_NETLINK_PROC_MAX 2
+
+struct vpp_netlink_proc_t {
+ __u32 pid;
+ rwlock_t lock;
+};
+
+struct switch_dev vpp_sdev = {
+ .name = "hdmi",
+};
+
+static struct switch_dev vpp_sdev_hdcp = {
+ .name = "hdcp",
+};
+
+#if 0
+static struct switch_dev vpp_sdev_audio = {
+ .name = "hdmi_audio",
+};
+#endif
+
+struct vpp_netlink_proc_t vpp_netlink_proc[VPP_NETLINK_PROC_MAX];
+static struct sock *vpp_nlfd;
+static DEFINE_SEMAPHORE(vpp_netlink_receive_sem);
+
+struct vpp_netlink_proc_t *vpp_netlink_get_proc(int no)
+{
+ if (no == 0)
+ return 0;
+ if (no > VPP_NETLINK_PROC_MAX)
+ return 0;
+ return &vpp_netlink_proc[no - 1];
+}
+
+static void vpp_netlink_receive(struct sk_buff *skb)
+{
+ struct nlmsghdr *nlh = NULL;
+ struct vpp_netlink_proc_t *proc;
+
+ if (down_trylock(&vpp_netlink_receive_sem))
+ return;
+
+ if (skb->len >= sizeof(struct nlmsghdr)) {
+ nlh = nlmsg_hdr(skb);
+ if ((nlh->nlmsg_len >= sizeof(struct nlmsghdr))
+ && (skb->len >= nlh->nlmsg_len)) {
+ proc = vpp_netlink_get_proc(nlh->nlmsg_type);
+ if (proc) {
+ write_lock_bh(&proc->lock);
+ proc->pid = nlh->nlmsg_pid;
+ write_unlock_bh(&proc->lock);
+ DPRINT("[VPP] rx user pid 0x%x\n", proc->pid);
+ }
+ }
+ }
+ up(&vpp_netlink_receive_sem);
+ wmt_enable_mmfreq(WMT_MMFREQ_HDMI_PLUG, hdmi_get_plugin());
+}
+
+void vpp_netlink_init(void)
+{
+ vpp_netlink_proc[0].pid = 0;
+ vpp_netlink_proc[1].pid = 0;
+ rwlock_init(&(vpp_netlink_proc[0].lock));
+ rwlock_init(&(vpp_netlink_proc[1].lock));
+ vpp_nlfd = netlink_kernel_create(&init_net, NETLINK_CEC_TEST, 0,
+ vpp_netlink_receive, NULL, THIS_MODULE);
+ if (!vpp_nlfd)
+ DPRINT(KERN_ERR "can not create a netlink socket\n");
+}
+
+static ssize_t attr_show_parsed_edid(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ ssize_t len = 0;
+ int i, j;
+ unsigned char audio_format, sample_freq, bitrate;
+ int sample_freq_num, bitrate_num;
+
+#ifdef DEBUG
+ DPRINT("------- EDID Parsed ------\n");
+ if(strlen(edid_parsed.tv_name.vendor_name) != 0)
+ DPRINT("Vendor Name: %s\n", edid_parsed.tv_name.vendor_name);
+
+ if(strlen(edid_parsed.tv_name.monitor_name) != 0)
+ DPRINT("Monitor Name: %s\n", edid_parsed.tv_name.monitor_name);
+
+ for(i = 0; i < AUD_SAD_NUM; i++) {
+ if(edid_parsed.sad[i].flag == 0) {
+ if(i == 0)
+ printk("No SAD Data\n");
+ break;
+ }
+ DPRINT("SAD %d: 0x%02X 0x%02X 0x%02X\n", i,
+ edid_parsed.sad[i].sad_byte[0], edid_parsed.sad[i].sad_byte[1],
+ edid_parsed.sad[i].sad_byte[2]);
+ }
+ DPRINT("--------------------------\n");
+#endif
+ /* print Vendor Name */
+ if(strlen(edid_parsed.tv_name.vendor_name) != 0) {
+ len += sprintf(buf + len, "%-16s", "Vendor Name");
+ len += sprintf(buf + len, ": %s\n", edid_parsed.tv_name.vendor_name);
+ }
+
+ /* print Monitor Name */
+ if(strlen(edid_parsed.tv_name.monitor_name) != 0) {
+ len += sprintf(buf + len, "%-16s", "Monitor Name");
+ len += sprintf(buf + len, ": %s\n", edid_parsed.tv_name.monitor_name);
+ }
+
+ for(i = 0; i < AUD_SAD_NUM; i++) {
+ if(edid_parsed.sad[i].flag == 0)
+ break;
+ /*
+ SAD Byte 1 (format and number of channels):
+ bit 7: Reserved (0)
+ bit 6..3: Audio format code
+ 1 = Linear Pulse Code Modulation (LPCM)
+ 2 = AC-3
+ 3 = MPEG1 (Layers 1 and 2)
+ 4 = MP3
+ 5 = MPEG2
+ 6 = AAC
+ 7 = DTS
+ 8 = ATRAC
+ 0, 15: Reserved
+ 9 = One-bit audio aka SACD
+ 10 = DD+
+ 11 = DTS-HD
+ 12 = MLP/Dolby TrueHD
+ 13 = DST Audio
+ 14 = Microsoft WMA Pro
+ bit 2..0: number of channels minus 1 (i.e. 000 = 1 channel; 001 = 2 channels; 111 =
+ 8 channels)
+ */
+ audio_format = (edid_parsed.sad[i].sad_byte[0] & 0x78) >> 3;
+ if(audio_format == 0 || audio_format == 15)
+ continue;
+
+ /* print header */
+ len += sprintf(buf + len, "%-16s", "Audio Format");
+ len += sprintf(buf + len, ": ");
+
+ switch(audio_format) {
+ case 1:
+ len += sprintf(buf + len, "pcm");
+ break;
+ case 2:
+ len += sprintf(buf + len, "ac3");
+ break;
+ case 3:
+ len += sprintf(buf + len, "mpeg1");
+ break;
+ case 4:
+ len += sprintf(buf + len, "mp3");
+ break;
+ case 5:
+ len += sprintf(buf + len, "mpeg2");
+ break;
+ case 6:
+ len += sprintf(buf + len, "aac");
+ break;
+ case 7:
+ len += sprintf(buf + len, "dts");
+ break;
+ case 8:
+ len += sprintf(buf + len, "atrac");
+ break;
+ case 9:
+ len += sprintf(buf + len, "one_bit_audio");
+ break;
+ case 10:
+ len += sprintf(buf + len, "eac3");
+ break;
+ case 11:
+ len += sprintf(buf + len, "dts-hd");
+ break;
+ case 12:
+ len += sprintf(buf + len, "mlp");
+ break;
+ case 13:
+ len += sprintf(buf + len, "dst");
+ break;
+ case 14:
+ len += sprintf(buf + len, "wmapro");
+ break;
+ default:
+ break;
+ }
+
+ /* separator */
+ len += sprintf(buf + len, ",");
+
+ /* number of channels */
+ len += sprintf(buf + len, "%d", (edid_parsed.sad[i].sad_byte[0] & 0x7) + 1);
+
+ /* separator */
+ len += sprintf(buf + len, ",");
+
+ /*
+ SAD Byte 2 (sampling frequencies supported):
+ bit 7: Reserved (0)
+ bit 6: 192kHz
+ bit 5: 176kHz
+ bit 4: 96kHz
+ bit 3: 88kHz
+ bit 2: 48kHz
+ bit 1: 44kHz
+ bit 0: 32kHz
+ */
+ sample_freq = edid_parsed.sad[i].sad_byte[1];
+ sample_freq_num = 0;
+ for(j = 0; j < 7; j++) {
+ if(sample_freq & (1 << j)) {
+ if(sample_freq_num != 0)
+ len += sprintf(buf + len, "|"); /* separator */
+ switch(j) {
+ case 0:
+ len += sprintf(buf + len, "32KHz");
+ break;
+ case 1:
+ len += sprintf(buf + len, "44KHz");
+ break;
+ case 2:
+ len += sprintf(buf + len, "48KHz");
+ break;
+ case 3:
+ len += sprintf(buf + len, "88KHz");
+ break;
+ case 4:
+ len += sprintf(buf + len, "96KHz");
+ break;
+ case 5:
+ len += sprintf(buf + len, "176KHz");
+ break;
+ case 6:
+ len += sprintf(buf + len, "192KHz");
+ break;
+ default:
+ break;
+ }
+ sample_freq_num++;
+ }
+ }
+
+ if(sample_freq_num == 0)
+ len += sprintf(buf +len, "0");
+
+ /* separator */
+ len += sprintf(buf + len, ",");
+
+ /*
+ SAD Byte 3 (bitrate):
+ For LPCM, bits 7:3 are reserved and the remaining bits define bit depth
+ bit 2: 24 bit
+ bit 1: 20 bit
+ bit 0: 16 bit
+ For all other sound formats, bits 7..0 designate the maximum supported bitrate divided by
+ 8 kbit/s.
+ */
+ bitrate = edid_parsed.sad[i].sad_byte[2];
+ bitrate_num = 0;
+ if(audio_format == 1) { /* for LPCM */
+ for(j = 0; j < 3; j++) {
+ if(bitrate & (1 << j)) {
+ if(bitrate_num != 0)
+ len += sprintf(buf + len, "|"); /* separator */
+ switch(j) {
+ case 0:
+ len += sprintf(buf + len, "16bit");
+ break;
+ case 1:
+ len += sprintf(buf + len, "20bit");
+ break;
+ case 2:
+ len += sprintf(buf + len, "24bit");
+ break;
+ default:
+ break;
+ }
+ bitrate_num++;
+ }
+ }
+ } else if(audio_format >= 2 && audio_format <= 8) /* From AC3 to ATRAC */
+ len += sprintf(buf + len, "%dkbps", bitrate * 8);
+ else /* From One-bit-audio to WMA Pro*/
+ len += sprintf(buf + len, "%d", bitrate);
+
+ len += sprintf(buf + len, "\n");
+ }
+
+ if(len == 0)
+ len += sprintf(buf + len, "\n");
+
+ return len;
+}
+
+static DEVICE_ATTR(edid_parsed, 0444, attr_show_parsed_edid, NULL);
+
+void vpp_switch_state_init(void)
+{
+ /* /sys/class/switch/hdmi/state */
+ switch_dev_register(&vpp_sdev);
+ switch_set_state(&vpp_sdev, hdmi_get_plugin() ? 1 : 0);
+ switch_dev_register(&vpp_sdev_hdcp);
+ switch_set_state(&vpp_sdev_hdcp, 0);
+#if 0
+ switch_dev_register(&vpp_sdev_audio);
+#endif
+ device_create_file(vpp_sdev.dev, &dev_attr_edid_parsed);
+}
+
+void vpp_netlink_notify(int no, int cmd, int arg)
+{
+ int ret;
+ int size;
+ unsigned char *old_tail;
+ struct sk_buff *skb;
+ struct nlmsghdr *nlh;
+ struct vpp_netlink_proc_t *proc;
+
+ proc = vpp_netlink_get_proc(no);
+ if (!proc)
+ return;
+
+ MSG("[VPP] netlink notify %d,cmd %d,0x%x\n", no, cmd, arg);
+
+ switch (cmd) {
+ case DEVICE_RX_DATA:
+ size = NLMSG_SPACE(sizeof(struct wmt_cec_msg));
+ break;
+ case DEVICE_PLUG_IN:
+ case DEVICE_PLUG_OUT:
+ case DEVICE_STREAM:
+ size = NLMSG_SPACE(sizeof(struct wmt_cec_msg));
+ break;
+ default:
+ return;
+ }
+
+ skb = alloc_skb(size, GFP_ATOMIC);
+ if (skb == NULL)
+ return;
+ old_tail = skb->tail;
+ nlh = NLMSG_PUT(skb, 0, 0, 0, size-sizeof(*nlh));
+ nlh->nlmsg_len = skb->tail - old_tail;
+
+ switch (cmd) {
+ case DEVICE_RX_DATA:
+ nlh->nlmsg_type = DEVICE_RX_DATA;
+ memcpy(NLMSG_DATA(nlh), (struct wmt_cec_msg *)arg,
+ sizeof(struct wmt_cec_msg));
+ break;
+ case DEVICE_PLUG_IN:
+ case DEVICE_PLUG_OUT:
+ {
+ static int cnt;
+ struct wmt_cec_msg *msg;
+
+ msg = (struct wmt_cec_msg *)NLMSG_DATA(nlh);
+ msg->msgdata[0] = cnt;
+ cnt++;
+ }
+ if (arg) {
+ nlh->nlmsg_type = DEVICE_PLUG_IN;
+ nlh->nlmsg_flags = edid_get_hdmi_phy_addr();
+ } else {
+ nlh->nlmsg_type = DEVICE_PLUG_OUT;
+ }
+ size = NLMSG_SPACE(sizeof(0));
+ break;
+ case DEVICE_STREAM:
+ nlh->nlmsg_type = cmd;
+ nlh->nlmsg_flags = arg;
+ break;
+ default:
+ return;
+ }
+
+ NETLINK_CB(skb).pid = 0;
+ NETLINK_CB(skb).dst_group = 0;
+
+ if (proc->pid != 0) {
+ ret = netlink_unicast(vpp_nlfd, skb, proc->pid, MSG_DONTWAIT);
+ return;
+ }
+nlmsg_failure: /* NLMSG_PUT go to */
+ if (skb != NULL)
+ kfree_skb(skb);
+ return;
+}
+
+void vpp_netlink_notify_plug(int vo_num, int plugin)
+{
+ /* set unplug flag for check_var */
+ if (plugin == 0) {
+ int mask = 0;
+
+ if (g_vpp.virtual_display)
+ mask = ~1;
+ else if (vo_num == VPP_VOUT_ALL)
+ mask = ~0;
+ else {
+ vout_info_t *vo_info;
+ vo_info = vout_get_info_entry(vo_num);
+ if (vo_info)
+ mask = 0x1 << (vo_info->num);
+ }
+ g_vpp.fb_recheck |= mask;
+ }
+
+ if ((vpp_netlink_proc[0].pid == 0) && (vpp_netlink_proc[1].pid == 0))
+ return;
+
+ /* if hdmi unplug, clear edid_parsed */
+ if(hdmi_get_plugin() == 0)
+ memset(&edid_parsed, 0, sizeof(edid_parsed_t));
+
+ vpp_netlink_notify(USER_PID, DEVICE_PLUG_IN, plugin);
+ vpp_netlink_notify(WP_PID, DEVICE_PLUG_IN, plugin);
+
+ /* hdmi plugin/unplug */
+ plugin = hdmi_get_plugin();
+ switch_set_state(&vpp_sdev, plugin ? 1 : 0);
+ wmt_enable_mmfreq(WMT_MMFREQ_HDMI_PLUG, plugin);
+}
+
+void vpp_netlink_notify_cp(int enable)
+{
+ switch_set_state(&vpp_sdev_hdcp, enable);
+}
+#endif /* CONFIG_VPP_NOTIFY */
+#endif /* __KERNEL__ */
diff --git a/drivers/video/wmt/vpp.h b/drivers/video/wmt/vpp.h
new file mode 100644
index 00000000..6da2a8b2
--- /dev/null
+++ b/drivers/video/wmt/vpp.h
@@ -0,0 +1,687 @@
+/*++
+ * linux/drivers/video/wmt/vpp.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp-osif.h"
+#include "./hw/wmt-vpp-hw.h"
+#include "com-vpp.h"
+#include "vout.h"
+
+#ifndef VPP_H
+#define VPP_H
+
+/* #define CONFIG_VPP_SHENZHEN */ /* for ShenZhen code */
+
+/* VPP feature config */
+/* #define CONFIG_VPP_DEMO */ /* HDMI EDID, CP disable */
+#define CONFIG_VPP_STREAM_CAPTURE /* stream capture current video display */
+#define CONFIG_VPP_STREAM_BLOCK
+#define CONFIG_VPP_STREAM_FIX_RESOLUTION
+#define CONFIG_VPP_STREAM_ROTATE
+/* #define CONFIG_VPP_DISABLE_PM */ /* disable power management */
+#define CONFIG_VPP_VIRTUAL_DISPLAY /* virtual fb dev */
+#define CONFIG_VPP_NOTIFY
+#define CONFIG_VPP_DYNAMIC_ALLOC /* frame buffer dynamic allocate */
+
+/* VPP constant define */
+#define VPP_MB_ALLOC_NUM 3
+#define VPP_STREAM_MB_ALLOC_NUM (VPP_MB_ALLOC_NUM*2)
+
+typedef enum {
+ VPP_INT_NULL = 0,
+ VPP_INT_ALL = 0xffffffff,
+
+ VPP_INT_GOVRH_PVBI = BIT0,
+ VPP_INT_GOVRH_VBIS = BIT1, /* write done */
+ VPP_INT_GOVRH_VBIE = BIT2,
+
+ VPP_INT_GOVW_PVBI = BIT3,
+ VPP_INT_GOVW_VBIS = BIT4,
+ VPP_INT_GOVW_VBIE = BIT5,
+
+ VPP_INT_DISP_PVBI = BIT6,
+ VPP_INT_DISP_VBIS = BIT7,
+ VPP_INT_DISP_VBIE = BIT8,
+
+ VPP_INT_LCD_EOF = BIT9,
+
+ VPP_INT_SCL_PVBI = BIT12,
+ VPP_INT_SCL_VBIS = BIT13,
+ VPP_INT_SCL_VBIE = BIT14,
+
+ VPP_INT_VPU_PVBI = BIT15,
+ VPP_INT_VPU_VBIS = BIT16,
+ VPP_INT_VPU_VBIE = BIT17,
+
+ VPP_INT_GOVRH2_PVBI = BIT18,
+ VPP_INT_GOVRH2_VBIS = BIT19, /* write done */
+ VPP_INT_GOVRH2_VBIE = BIT20,
+
+ VPP_INT_MAX = BIT31,
+
+} vpp_int_t;
+
+typedef enum {
+ /* SCL */
+ VPP_INT_ERR_SCL_TG = BIT0,
+ VPP_INT_ERR_SCLR1_MIF = BIT1,
+ VPP_INT_ERR_SCLR2_MIF = BIT2,
+ VPP_INT_ERR_SCLW_MIFRGB = BIT3,
+ VPP_INT_ERR_SCLW_MIFY = BIT4,
+ VPP_INT_ERR_SCLW_MIFC = BIT5,
+
+ /* GOVRH2 */
+ VPP_INT_ERR_GOVRH2_MIF = BIT19,
+
+ /* GOVRH */
+ VPP_INT_ERR_GOVRH_MIF = BIT20,
+
+} vpp_int_err_t;
+
+/* VPP FB capability flag */
+#define VPP_FB_FLAG_COLFMT 0xFFFF
+#define VPP_FB_FLAG_SCALE BIT(16)
+#define VPP_FB_FLAG_CSC BIT(17)
+#define VPP_FB_FLAG_MEDIA BIT(18)
+#define VPP_FB_FLAG_FIELD BIT(19)
+
+typedef struct {
+ vdo_framebuf_t fb;
+ vpp_csc_t csc_mode;
+ int framerate;
+ vpp_media_format_t media_fmt;
+ int wait_ready;
+ unsigned int capability;
+
+ void (*set_framebuf)(vdo_framebuf_t *fb);
+ void (*get_framebuf)(vdo_framebuf_t *fb);
+ void (*set_addr)(unsigned int yaddr, unsigned int caddr);
+ void (*get_addr)(unsigned int *yaddr, unsigned int *caddr);
+ void (*set_csc)(vpp_csc_t mode);
+ vdo_color_fmt (*get_color_fmt)(void);
+ void (*set_color_fmt)(vdo_color_fmt colfmt);
+ void (*fn_view)(int read, vdo_view_t *view);
+} vpp_fb_base_t;
+
+#define VPP_MOD_BASE \
+ vpp_mod_t mod; /* module id*/\
+ void *mmio; /* regs base address */\
+ unsigned int int_catch; /* interrupt catch */\
+ vpp_fb_base_t *fb_p; /* framebuf base pointer */\
+ unsigned int pm; /* power dev id,bit31-0:power off */\
+ unsigned int *reg_bk; /* register backup pointer */\
+ void (*init)(void *base); /* module initial */\
+ void (*dump_reg)(void); /* dump hardware register */\
+ void (*set_enable)(vpp_flag_t enable); /* module enable/disable */\
+ void (*set_colorbar)(vpp_flag_t enable, int mode, int inv); \
+ /* hw colorbar enable/disable & mode */\
+ void (*set_tg)(vpp_clock_t *tmr, unsigned int pixel_clock); /*set tg*/\
+ void (*get_tg)(vpp_clock_t *tmr); /* get timing */\
+ unsigned int (*get_sts)(void); /* get interrupt or error status */\
+ void (*clr_sts)(unsigned int sts); /* clear interrupt or err status */\
+ void (*suspend)(int sts); /* module suspend */\
+ void (*resume)(int sts) /* module resume */
+/* End of vpp_mod_base_t */
+
+typedef struct {
+ VPP_MOD_BASE;
+} vpp_mod_base_t;
+
+#define VPP_MOD_FLAG_FRAMEBUF BIT(0)
+#define VPP_MOD_CLK_ON BIT(31)
+
+typedef enum {
+ VPP_SCALE_MODE_REC_TABLE, /* old design but 1/32 limit */
+ VPP_SCALE_MODE_RECURSIVE, /*no rec table,not smooth than bilinear mode*/
+ VPP_SCALE_MODE_BILINEAR,/*more smooth but less than 1/2 will drop line*/
+ VPP_SCALE_MODE_ADAPTIVE,/* scl dn 1-1/2 bilinear mode, other rec mode */
+ VPP_SCALE_MODE_MAX
+} vpp_scale_mode_t;
+
+typedef enum {
+ VPP_HDMI_AUDIO_I2S,
+ VPP_HDMI_AUDIO_SPDIF,
+ VPP_HDMI_AUDIO_MAX
+} vpp_hdmi_audio_inf_t;
+
+typedef enum {
+ VPP_FILTER_SCALE,
+ VPP_FILTER_DEBLOCK,
+ VPP_FILTER_FIELD_DEFLICKER,
+ VPP_FILTER_FRAME_DEFLICKER,
+ VPP_FILTER_MODE_MAX
+} vpp_filter_mode_t;
+
+#define VPP_DBG_PERIOD_NUM 10
+typedef struct {
+ int index;
+ int period_us[VPP_DBG_PERIOD_NUM];
+ struct timeval pre_tv;
+} vpp_dbg_period_t;
+
+typedef struct {
+ struct timeval pre_tv;
+ unsigned int threshold;
+ unsigned int reset;
+ unsigned int cnt;
+ unsigned int sum;
+ unsigned int min;
+ unsigned int max;
+} vpp_dbg_timer_t;
+
+#ifdef __KERNEL__
+#define VPP_PROC_NUM 10
+typedef struct {
+ int (*func)(void *arg); /* function pointer */
+ void *arg; /* function argument */
+ struct list_head list;
+ vpp_int_t type; /* interrupt type */
+ struct semaphore sem; /* wait sem */
+ int wait_ms; /* wait complete timout (ms) */
+ int work_cnt; /* work counter if 0 then forever */
+} vpp_proc_t;
+
+typedef struct {
+ struct list_head list;
+ struct tasklet_struct tasklet;
+ int ref;
+} vpp_irqproc_t;
+#endif
+
+#ifndef CFG_LOADER
+#include "vppm.h"
+#endif
+#include "lcd.h"
+
+#ifndef CFG_LOADER
+/* #ifdef WMT_FTBLK_SCL */
+#include "scl.h"
+/* #endif */
+#endif
+/*
+#ifdef WMT_FTBLK_GE
+#include "ge.h"
+#endif
+*/
+#ifdef WMT_FTBLK_GOVRH
+#include "govrh.h"
+#endif
+#ifdef WMT_FTBLK_LVDS
+#include "lvds.h"
+#endif
+/* #ifdef WMT_FTBLK_HDMI */
+#include "hdmi.h"
+#ifndef CFG_LOADER
+#include "cec.h"
+#endif
+/* #endif */
+#ifdef CONFIG_WMT_EDID
+#include "edid.h"
+#endif
+
+typedef enum {
+ VPP_DBGLVL_DISABLE = 0x0,
+ VPP_DBGLVL_SCALE = 1,
+ VPP_DBGLVL_DISPFB = 2,
+ VPP_DBGLVL_INT = 3,
+ VPP_DBGLVL_FPS = 4,
+ VPP_DBGLVL_IOCTL = 5,
+ VPP_DBGLVL_DIAG = 6,
+ VPP_DBGLVL_STREAM = 7,
+ VPP_DBGLVL_ALL = 0xFF,
+} vpp_dbg_level_t;
+
+typedef struct {
+ /* internal parameter */
+ int govrh_preinit;
+ int (*alloc_framebuf)(unsigned int resx, unsigned int resy);
+ int dual_display; /* use 2 govr */
+ int virtual_display;
+ int fb0_bitblit;
+ int fb_manual; /* not check var & internel timing */
+ int fb_recheck; /* recheck for plug but no change res */
+ int govrh_init_yres;
+ int virtual_display_mode;
+
+ /* hdmi */
+ int hdmi_video_mode; /* 0-auto,720,1080 */
+ vpp_hdmi_audio_inf_t hdmi_audio_interface; /* 0-I2S, 1-SPDIF */
+ int hdmi_cp_enable; /* 0-off, 1-on */
+ unsigned int hdmi_ctrl;
+ unsigned int hdmi_audio_pb1;
+ unsigned int hdmi_audio_pb4;
+ unsigned int hdmi_i2c_freq;
+ unsigned int hdmi_i2c_udelay;
+ int hdmi_init;
+ unsigned int hdmi_bksv[2];
+ char *hdmi_cp_p;
+ int hdmi_3d_type;
+ unsigned int hdmi_pixel_clock;
+ int hdmi_certify_flag;
+ int hdmi_sp_mode;
+ int hdmi_disable;
+
+ /* alloc frame buffer */
+ unsigned int mb[VPP_MB_ALLOC_NUM];
+ unsigned int mb_y_size;
+ unsigned int mb_fb_size;
+ int mb_colfmt;
+
+ /* debug */
+ int dbg_msg_level; /* debug message level */
+ int dbg_wait;
+ int dbg_flag;
+
+#if 0
+ /* HDMI DDC debug */
+ int dbg_hdmi_ddc_ctrl_err;
+ int dbg_hdmi_ddc_read_err;
+ int dbg_hdmi_ddc_crc_err;
+#endif
+
+#ifdef CONFIG_VPP_STREAM_CAPTURE
+ /* stream capture current video display */
+ int stream_enable;
+ unsigned int stream_mb_lock;
+ int stream_mb_sync_flag;
+ int stream_mb_index;
+ unsigned int stream_mb[VPP_STREAM_MB_ALLOC_NUM];
+ unsigned int stream_mb_y_size;
+ int stream_mb_cnt;
+ unsigned int stream_sync_cnt;
+#ifdef CONFIG_VPP_STREAM_FIX_RESOLUTION
+ vdo_framebuf_t stream_fb;
+#endif
+#endif
+} vpp_info_t;
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifdef VPP_C
+#define EXTERN
+
+const unsigned int vpp_csc_parm[VPP_CSC_MAX][7] = {
+ /* C1,C2 C3,C4 C5,C6 C7,C8 C9,I
+ J,K b0:YC2RGB,b8:clamp */
+ {0x000004a8, 0x04a80662, 0x1cbf1e70, 0x081204a8, 0x00010000,
+ 0x00010001, 0x00000101}, /* YUV2RGB_SDTV_0_255 */
+ {0x00000400, 0x0400057c, 0x1d351ea8, 0x06ee0400, 0x00010000,
+ 0x00010001, 0x00000001}, /* YUV2RGB_SDTV_16_235 */
+ {0x000004a8, 0x04a8072c, 0x1ddd1f26, 0x087604a8, 0x00010000,
+ 0x00010001, 0x00000101}, /* YUV2RGB_HDTV_0_255 */
+ {0x00000400, 0x04000629, 0x1e2a1f45, 0x07440400, 0x00010000,
+ 0x00010001, 0x00000001}, /* YUV2RGB_HDTV_16_235 */
+ {0x00000400, 0x0400059c, 0x1d251ea0, 0x07170400, 0x00010000,
+ 0x00010001, 0x00000001}, /* YUV2RGB_JFIF_0_255 */
+ {0x00000400, 0x0400057c, 0x1d351ea8, 0x06ee0400, 0x00010000,
+ 0x00010001, 0x00000001}, /* YUV2RGB_SMPTE170M */
+ {0x00000400, 0x0400064d, 0x1e001f19, 0x074f0400, 0x00010000,
+ 0x00010001, 0x00000001}, /* YUV2RGB_SMPTE240M */
+ {0x02040107, 0x1f680064, 0x01c21ed6, 0x1e8701c2, 0x00211fb7,
+ 0x01010101, 0x00000000}, /* RGB2YUV_SDTV_0_255 */
+ {0x02590132, 0x1f500075, 0x020b1ea5, 0x1e4a020b, 0x00011fab,
+ 0x01010101, 0x00000000}, /* RGB2YUV_SDTV_16_235 */
+ {0x027500bb, 0x1f99003f, 0x01c21ea6, 0x1e6701c2, 0x00211fd7,
+ 0x01010101, 0x00000000}, /* RGB2YUV_HDTV_0_255 */
+ {0x02dc00da, 0x1f88004a, 0x020b1e6d, 0x1e25020b, 0x00011fd0,
+ 0x01010101, 0x00000000}, /* RGB2YUV_HDTV_16_235 */
+ {0x02590132, 0x1f530075, 0x02001ead, 0x1e530200, 0x00011fad,
+ 0x00ff00ff, 0x00000000}, /* RGB2YUV_JFIF_0_255 */
+ {0x02590132, 0x1f500075, 0x020b1ea5, 0x1e4a020b, 0x00011fab,
+ 0x01010101, 0x00000000}, /* RGB2YUV_SMPTE170M */
+ {0x02ce00d9, 0x1f890059, 0x02001e77, 0x1e380200, 0x00011fc8,
+ 0x01010101, 0x00000000}, /* RGB2YUV_SMPTE240M */
+};
+
+const struct fb_videomode vpp_videomode[] = {
+ /* 640x480@60 DMT/CEA861 */
+ { NULL, 60, 640, 480, KHZ2PICOS(25175), 48, 16, 33, 10, 96, 2,
+ 0, FB_VMODE_NONINTERLACED, 0},
+#if 0
+ /* 640x480@60 CVT */
+ { NULL, 60, 640, 480, KHZ2PICOS(23750), 80, 16, 13, 3, 64, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 640x480@75 DMT */
+ { NULL, 75, 640, 480, KHZ2PICOS(31500), 120, 16, 16, 1, 64, 3,
+ 0, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 640x480@75 CVT */
+ { NULL, 75, 640, 480, KHZ2PICOS(30750), 88, 24, 17, 3, 64, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 720x480@60 CEA861 */
+ { NULL, 60, 720, 480, KHZ2PICOS(27027), 60, 16, 30, 9, 62, 6,
+ 0, FB_VMODE_NONINTERLACED, 0},
+ /* 720x480i@60 CEA861 */
+ { NULL, 60, 720, 480, KHZ2PICOS(27000), 114, 38, 30, 8, 124, 6,
+ 0, FB_VMODE_INTERLACED + FB_VMODE_DOUBLE, 0},
+ /* 720x576@50 CEA861 */
+ { NULL, 50, 720, 576, KHZ2PICOS(27000), 68, 12, 39, 5, 64, 5,
+ 0, FB_VMODE_NONINTERLACED, 0},
+ /* 720x576i@50 CEA861 */
+ { NULL, 50, 720, 576, KHZ2PICOS(27000), 138, 24, 38, 4, 126, 6,
+ 0, FB_VMODE_INTERLACED + FB_VMODE_DOUBLE, 0},
+ /* 800x480@60 CVT */
+ { NULL, 60, 800, 480, KHZ2PICOS(29500), 96, 24, 10, 3, 72, 7,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 800x480@75 CVT */
+ { NULL, 75, 800, 480, KHZ2PICOS(38500), 112, 32, 14, 3, 80, 7,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 800x600@60 DMT */
+ { NULL, 60, 800, 600, KHZ2PICOS(40000), 88, 40, 23, 1, 128, 4,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 800x600@60 CVT */
+ { NULL, 60, 800, 600, KHZ2PICOS(38250), 112, 32, 17, 3, 80, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 800x600@75 DMT */
+ { NULL, 75, 800, 600, KHZ2PICOS(49500), 160, 16, 21, 1, 80, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 800x600@75 CVT */
+ { NULL, 75, 800, 600, KHZ2PICOS(49000), 120, 40, 22, 3, 80, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 848x480@60 DMT */
+ { NULL, 60, 848, 480, KHZ2PICOS(33750), 112, 16, 23, 6, 112, 8,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1024x600@60 DMT */
+ { NULL, 60, 1024, 600, KHZ2PICOS(49000), 144, 40, 11, 3, 104, 10,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1024x768@60 DMT */
+ { NULL, 60, 1024, 768, KHZ2PICOS(65000), 160, 24, 29, 3, 136, 6,
+ 0, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1024x768@60 CVT */
+ { NULL, 60, 1024, 768, KHZ2PICOS(63500), 152, 48, 23, 3, 104, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 1024x768@75 DMT */
+ { NULL, 75, 1024, 768, KHZ2PICOS(78750), 176, 16, 28, 1, 96, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 1024x768@75 CVT */
+ { NULL, 75, 1024, 768, KHZ2PICOS(82000), 168, 64, 30, 3, 104, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 1152x864@60 CVT */
+ { NULL, 60, 1152, 864, KHZ2PICOS(81750), 184, 64, 26, 3, 120, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 1152x864@75 DMT */
+ { NULL, 75, 1152, 864, KHZ2PICOS(108000), 256, 64, 32, 1, 128, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1152x864@75 CVT */
+ { NULL, 75, 1152, 864, KHZ2PICOS(104000), 192, 72, 34, 3, 120, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 1280x720@50 CEA861,HDMI_1280x720p50_16x9 */
+ { NULL, 50, 1280, 720, KHZ2PICOS(74250), 220, 440, 20, 5, 40, 5,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1280x720@60 CEA861,HDMI_1280x720p60_16x9 */
+ { NULL, 60, 1280, 720, KHZ2PICOS(74250), 220, 110, 20, 5, 40, 5,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+#if 0
+ /* 1280x720@60 CVT */
+ { NULL, 60, 1280, 720, KHZ2PICOS(74500), 192, 64, 20, 3, 128, 5,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 1280x720@75 CVT */
+ { NULL, 75, 1280, 720, KHZ2PICOS(95750), 208, 80, 27, 3, 128, 5,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 1280x768@60 DMT/CVT */
+ { NULL, 60, 1280, 768, KHZ2PICOS(79500), 192, 64, 20, 3, 128, 7,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1280x768@75 DMT/CVT */
+ { NULL, 75, 1280, 768, KHZ2PICOS(102250), 208, 80, 27, 3, 128, 7,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1280x800@60 DMT/CVT */
+ { NULL, 60, 1280, 800, KHZ2PICOS(83500), 200, 72, 22, 3, 128, 6,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1280x800@75 DMT/CVT */
+ { NULL, 75, 1280, 800, KHZ2PICOS(106500), 208, 80, 29, 3, 128, 6,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1280x960@60 DMT */
+ { NULL, 60, 1280, 960, KHZ2PICOS(108000), 312, 96, 36, 1, 112, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 1280x960@60 CVT */
+ { NULL, 60, 1280, 960, KHZ2PICOS(101250), 208, 80, 29, 3, 128, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 1280x960@75 CVT */
+ { NULL, 75, 1280, 960, KHZ2PICOS(130000), 224, 88, 38, 3, 136, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+ /* 1280x1024@60 DMT */
+ { NULL, 60, 1280, 1024, KHZ2PICOS(108000), 248, 48, 38, 1, 112, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 1280x1024@60 CVT */
+ { NULL, 60, 1280, 1024, KHZ2PICOS(109000), 216, 80, 29, 3, 136, 7,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 1280x1024@75 DMT */
+ { NULL, 75, 1280, 1024, KHZ2PICOS(135000), 248, 16, 38, 1, 144, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 1280x1024@75 CVT */
+ { NULL, 75, 1280, 1024, KHZ2PICOS(138750), 224, 88, 38, 3, 136, 7,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, 0},
+#endif
+ /* 1360x768@60 */
+ { NULL, 60, 1360, 768, KHZ2PICOS(85500), 256, 64, 18, 3, 112, 6,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1366x768@60 */
+ { NULL, 60, 1366, 768, KHZ2PICOS(85500), 213, 70, 24, 3, 143, 3,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1400x1050@60 DMT/CVT */
+ { NULL, 60, 1400, 1050, KHZ2PICOS(121750), 232, 88, 32, 3, 144, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#if 0
+ /* 1400x1050@60+R DMT/CVT */
+ { NULL, 60, 1400, 1050, KHZ2PICOS(101000), 80, 48, 23, 3, 32, 4,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+#endif
+ /* 1440x480p@60 CEA861 */
+ { NULL, 60, 1400, 480, KHZ2PICOS(54054), 120, 32, 30, 9, 124, 6,
+ 0, FB_VMODE_NONINTERLACED, 0},
+ /* 1440x900@60 DMT/CVT */
+ { NULL, 60, 1440, 900, KHZ2PICOS(106500), 232, 80, 25, 3, 152, 6,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1440x900@75 DMT/CVT */
+ { NULL, 75, 1440, 900, KHZ2PICOS(136750), 248, 96, 33, 3, 152, 6,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1600x1200@60 DMT/CVT */
+ { NULL, 60, 1600, 1200, KHZ2PICOS(162000), 304, 64, 46, 1, 192, 3,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1680x1050@60 DMT/CVT */
+ { NULL, 60, 1680, 1050, KHZ2PICOS(146250), 280, 104, 30, 3, 176, 6,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1920x1080@25,HDMI_1920x1080p25_16x9 */
+ { NULL, 25, 1920, 1080, KHZ2PICOS(74250), 148, 528, 36, 4, 44, 5,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1920x1080@30,HDMI_1920x1080p30_16x9 */
+ { NULL, 30, 1920, 1080, KHZ2PICOS(74250), 148, 88, 36, 4, 44, 5,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1920x1080@50,HDMI_1920x1080p50_16x9 */
+ { NULL, 50, 1920, 1080, KHZ2PICOS(148500), 148, 528, 36, 4, 44, 5,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1920x1080i@50 */
+ { NULL, 50, 1920, 1080, KHZ2PICOS(74250), 148, 528, 30, 4, 44, 10,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_INTERLACED, 0},
+ /* 1920x1080@60 */
+ { NULL, 60, 1920, 1080, KHZ2PICOS(148500), 148, 88, 36, 4, 44, 5,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_NONINTERLACED, 0},
+ /* 1920x1080i@60 */
+ { NULL, 60, 1920, 1080, KHZ2PICOS(74250), 148, 88, 30, 4, 44, 10,
+ FB_SYNC_VERT_HIGH_ACT + FB_SYNC_HOR_HIGH_ACT,
+ FB_VMODE_INTERLACED, 0},
+ /* 1920x1200@60+R DMT/CVT */
+ { NULL, 60, 1920, 1200, KHZ2PICOS(154000), 80, 48, 26, 3, 32, 6,
+ FB_SYNC_HOR_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ /* 1920x1200@60 DMT/CVT */
+ { NULL, 60, 1920, 1200, KHZ2PICOS(193250), 336, 136, 36, 3, 200, 6,
+ FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED, FB_MODE_IS_VESA},
+ { NULL, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0 },
+};
+
+char *vpp_colfmt_str[] = {"YUV420", "YUV422H", "YUV422V", "YUV444", "YUV411",
+ "GRAY", "ARGB", "AUTO", "RGB888", "RGB666", "RGB565",
+ "RGB1555", "RGB5551"};
+const char *vpp_mod_str[] = {"GOVRH2", "GOVRH", "DISP", "GOVW", "GOVM", "SCL",
+ "SCLW", "VPU", "VPUW", "PIP", "VPPM", "LCDC", "CUR", "MAX"};
+
+#else
+#define EXTERN extern
+
+extern const unsigned int vpp_csc_parm[VPP_CSC_MAX][7];
+extern char *vpp_colfmt_str[];
+extern const struct fb_videomode vpp_videomode[];
+extern char *vpp_mod_str[];
+#endif
+
+EXTERN vpp_info_t g_vpp;
+
+static __inline__ int vpp_get_hdmi_spdif(void)
+{
+ return (g_vpp.hdmi_audio_interface == VPP_HDMI_AUDIO_SPDIF) ? 1 : 0;
+}
+
+/* Internal functions */
+EXTERN int get_key(void);
+EXTERN U8 vppif_reg8_in(U32 offset);
+EXTERN U8 vppif_reg8_out(U32 offset, U8 val);
+EXTERN U16 vppif_reg16_in(U32 offset);
+EXTERN U16 vppif_reg16_out(U32 offset, U16 val);
+EXTERN U32 vppif_reg32_in(U32 offset);
+EXTERN U32 vppif_reg32_out(U32 offset, U32 val);
+EXTERN U32 vppif_reg32_write(U32 offset, U32 mask, U32 shift, U32 val);
+EXTERN U32 vppif_reg32_read(U32 offset, U32 mask, U32 shift);
+EXTERN U32 vppif_reg32_mask(U32 offset, U32 mask, U32 shift);
+EXTERN unsigned int vpp_get_chipid(void);
+
+/* Export functions */
+EXTERN void vpp_mod_unregister(vpp_mod_t mod);
+EXTERN vpp_mod_base_t *vpp_mod_register(vpp_mod_t mod, int size,
+ unsigned int flags);
+EXTERN vpp_mod_base_t *vpp_mod_get_base(vpp_mod_t mod);
+EXTERN vpp_fb_base_t *vpp_mod_get_fb_base(vpp_mod_t mod);
+EXTERN vdo_framebuf_t *vpp_mod_get_framebuf(vpp_mod_t mod);
+EXTERN void vpp_mod_init(void);
+EXTERN void vpp_mod_set_clock(vpp_mod_t mod,
+ vpp_flag_t enable, int force);
+
+EXTERN unsigned int vpp_get_base_clock(vpp_mod_t mod);
+EXTERN void vpp_set_video_scale(vdo_view_t *vw);
+EXTERN int vpp_set_recursive_scale(vdo_framebuf_t *src_fb,
+ vdo_framebuf_t *dst_fb);
+EXTERN vpp_display_format_t vpp_get_fb_field(vdo_framebuf_t *fb);
+EXTERN void vpp_wait_vsync(int no, int cnt);
+EXTERN int vpp_get_gcd(int A, int B);
+EXTERN vpp_csc_t vpp_check_csc_mode(vpp_csc_t mode,
+ vdo_color_fmt src_fmt, vdo_color_fmt dst_fmt, unsigned int flags);
+EXTERN __inline__ void vpp_cache_sync(void);
+EXTERN int vpp_calc_refresh(int pixclk, int xres, int yres);
+EXTERN int vpp_calc_align(int value, int align);
+EXTERN int vpp_calc_fb_width(vdo_color_fmt colfmt, int width);
+EXTERN void vpp_get_colfmt_bpp(vdo_color_fmt colfmt,
+ int *y_bpp, int *c_bpp);
+EXTERN int vpp_irqproc_work(vpp_int_t type, int (*func)(void *argc),
+ void *arg, int wait_ms, int work_cnt);
+EXTERN int vpp_check_dbg_level(vpp_dbg_level_t level);
+
+#ifdef __KERNEL__
+EXTERN void vpp_dbg_show(int level, int tmr, char *str);
+EXTERN void vpp_dbg_wake_up(void);
+EXTERN int vpp_dbg_get_period_usec(vpp_dbg_period_t *p, int cmd);
+EXTERN void vpp_dbg_timer(vpp_dbg_timer_t *p, char *str, int cmd);
+EXTERN void vpp_dbg_back_trace(void);
+EXTERN void vpp_dbg_show_val1(int level, int tmr, char *str, int val);
+EXTERN void vpp_dbg_wait(char *str);
+EXTERN void vpp_irqproc_init(void);
+EXTERN void vpp_irqproc_del_work(vpp_int_t type,
+ int (*func)(void *argc));
+EXTERN vpp_irqproc_t *vpp_irqproc_get_entry(vpp_int_t vpp_int);
+
+/* dev-vpp.c */
+EXTERN void vpp_get_info(int fbn, struct fb_var_screeninfo *var);
+EXTERN int vpp_set_par(struct fb_info *info);
+EXTERN unsigned int *vpp_backup_reg(unsigned int addr, unsigned int size);
+EXTERN int vpp_restore_reg(unsigned int addr,
+ unsigned int size, unsigned int *reg_ptr);
+EXTERN void vpp_backup_reg2(unsigned int addr,
+ unsigned int size, unsigned int *ptr);
+EXTERN void vpp_restore_reg2(unsigned int addr,
+ unsigned int size, unsigned int *reg_ptr);
+EXTERN int vpp_pan_display(struct fb_var_screeninfo *var,
+ struct fb_info *info);
+EXTERN void vpp_netlink_init(void);
+EXTERN void vpp_netlink_notify(int no, int cmd, int arg);
+EXTERN void vpp_netlink_notify_plug(int vout_num, int plugin);
+EXTERN void vpp_netlink_notify_cp(int enable);
+EXTERN void vpp_switch_state_init(void);
+EXTERN int vpp_set_blank(struct fb_info *info, int blank);
+#endif
+
+EXTERN void vpp_reg_dump(unsigned int addr, int size);
+EXTERN unsigned int vpp_convert_colfmt(int yuv2rgb, unsigned int data);
+EXTERN void vpp_init(void);
+
+/* EXTERN void vpp_set_power_mgr(int arg); */
+EXTERN void vpp_show_timing(char *str,
+ struct fb_videomode *vmode, vpp_clock_t *clk);
+EXTERN void vpp_show_framebuf(char *str, vdo_framebuf_t *fb);
+EXTERN void vpp_show_videomode(char *str, struct fb_videomode *v);
+EXTERN void vpp_set_mutex(int idx, int lock);
+EXTERN void vpp_set_NA12_hiprio(int type);
+EXTERN void vpp_free_framebuffer(void);
+EXTERN int vpp_alloc_framebuffer(unsigned int resx, unsigned int resy);
+
+EXTERN int vpp_mb_get(unsigned int phy);
+EXTERN int vpp_mb_put(unsigned int phy);
+EXTERN int vpp_mb_irqproc_sync(int arg);
+EXTERN void vpp_mb_scale_bitblit(vdo_framebuf_t *fb);
+
+#undef EXTERN
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* VPP_H */
diff --git a/drivers/video/wmt/vppm.c b/drivers/video/wmt/vppm.c
new file mode 100644
index 00000000..f3bc7d9f
--- /dev/null
+++ b/drivers/video/wmt/vppm.c
@@ -0,0 +1,283 @@
+/*++
+ * linux/drivers/video/wmt/vppm.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define VPPM_C
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+#include "vppm.h"
+
+void vppm_set_int_enable(vpp_flag_t enable, vpp_int_t int_bit)
+{
+#ifdef WMT_FTBLK_SCL
+ if (int_bit & VPP_INT_SCL_VBIE)
+ vppif_reg32_write(VPP_SCL_INTEN_VBIE, enable);
+ if (int_bit & VPP_INT_SCL_VBIS)
+ vppif_reg32_write(VPP_SCL_INTEN_VBIS, enable);
+ if (int_bit & VPP_INT_SCL_PVBI)
+ vppif_reg32_write(VPP_SCL_INTEN_PVBI, enable);
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (int_bit & VPP_INT_GOVRH_VBIE)
+ vppif_reg32_write(VPP_GOVRH_INTEN_VBIE, enable);
+ if (int_bit & VPP_INT_GOVRH_VBIS)
+ vppif_reg32_write(VPP_GOVRH_INTEN_VBIS, enable);
+ if (int_bit & VPP_INT_GOVRH_PVBI)
+ vppif_reg32_write(VPP_GOVRH_INTEN_PVBI, enable);
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (int_bit & VPP_INT_GOVRH2_VBIE)
+ vppif_reg32_write(VPP_GOVRH2_INTEN_VBIE, enable);
+ if (int_bit & VPP_INT_GOVRH2_VBIS)
+ vppif_reg32_write(VPP_GOVRH2_INTEN_VBIS, enable);
+ if (int_bit & VPP_INT_GOVRH2_PVBI)
+ vppif_reg32_write(VPP_GOVRH2_INTEN_PVBI, enable);
+#endif
+}
+
+int vppm_get_int_enable(vpp_int_t int_bit)
+{
+ int ret = 0;
+
+#ifdef WMT_FTBLK_SCL
+ if (int_bit & VPP_INT_SCL_VBIE)
+ ret = vppif_reg32_read(VPP_SCL_INTEN_VBIE);
+ if (int_bit & VPP_INT_SCL_VBIS)
+ ret = vppif_reg32_read(VPP_SCL_INTEN_VBIS);
+ if (int_bit & VPP_INT_SCL_PVBI)
+ ret = vppif_reg32_read(VPP_SCL_INTEN_PVBI);
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (int_bit & VPP_INT_GOVRH_VBIE)
+ ret = vppif_reg32_read(VPP_GOVRH_INTEN_VBIE);
+ if (int_bit & VPP_INT_GOVRH_VBIS)
+ ret = vppif_reg32_read(VPP_GOVRH_INTEN_VBIS);
+ if (int_bit & VPP_INT_GOVRH_PVBI)
+ ret = vppif_reg32_read(VPP_GOVRH_INTEN_PVBI);
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (int_bit & VPP_INT_GOVRH2_VBIE)
+ ret = vppif_reg32_read(VPP_GOVRH2_INTEN_VBIE);
+ if (int_bit & VPP_INT_GOVRH2_VBIS)
+ ret = vppif_reg32_read(VPP_GOVRH2_INTEN_VBIS);
+ if (int_bit & VPP_INT_GOVRH2_PVBI)
+ ret = vppif_reg32_read(VPP_GOVRH2_INTEN_PVBI);
+#endif
+ return ret;
+}
+
+vpp_int_t vppm_get_int_status(void)
+{
+ unsigned int int_enable_reg;
+ unsigned int int_sts_reg;
+ vpp_int_t int_sts = 0;
+
+ int_enable_reg = vppif_reg32_in(REG_VPP_INTEN);
+ int_sts_reg = vppif_reg32_in(REG_VPP_INTSTS);
+
+#ifdef WMT_FTBLK_SCL
+ if ((int_enable_reg & BIT18) && (int_sts_reg & BIT18))
+ int_sts |= VPP_INT_SCL_VBIE;
+ if ((int_enable_reg & BIT17) && (int_sts_reg & BIT17))
+ int_sts |= VPP_INT_SCL_VBIS;
+ if ((int_enable_reg & BIT16) && (int_sts_reg & BIT16))
+ int_sts |= VPP_INT_SCL_PVBI;
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if ((int_enable_reg & BIT10) && (int_sts_reg & BIT10))
+ int_sts |= VPP_INT_GOVRH_VBIE;
+ if ((int_enable_reg & BIT9) && (int_sts_reg & BIT9))
+ int_sts |= VPP_INT_GOVRH_VBIS;
+ if ((int_enable_reg & BIT8) && (int_sts_reg & BIT8))
+ int_sts |= VPP_INT_GOVRH_PVBI;
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if ((int_enable_reg & BIT14) && (int_sts_reg & BIT14))
+ int_sts |= VPP_INT_GOVRH2_VBIE;
+ if ((int_enable_reg & BIT13) && (int_sts_reg & BIT13))
+ int_sts |= VPP_INT_GOVRH2_VBIS;
+ if ((int_enable_reg & BIT12) && (int_sts_reg & BIT12))
+ int_sts |= VPP_INT_GOVRH2_PVBI;
+#endif
+ return int_sts;
+}
+
+void vppm_clean_int_status(vpp_int_t int_sts)
+{
+#ifdef WMT_FTBLK_SCL
+ if (int_sts & VPP_INT_SCL_VBIE)
+ vppif_reg8_out(REG_VPP_INTSTS + 0x2, 0x4);
+ if (int_sts & VPP_INT_SCL_VBIS)
+ vppif_reg8_out(REG_VPP_INTSTS + 0x2, 0x2);
+ if (int_sts & VPP_INT_SCL_PVBI)
+ vppif_reg8_out(REG_VPP_INTSTS + 0x2, 0x1);
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (int_sts & VPP_INT_GOVRH_VBIE)
+ vppif_reg8_out(REG_VPP_INTSTS + 0x1, 0x4);
+ if (int_sts & VPP_INT_GOVRH_VBIS)
+ vppif_reg8_out(REG_VPP_INTSTS + 0x1, 0x2);
+ if (int_sts & VPP_INT_GOVRH_PVBI)
+ vppif_reg8_out(REG_VPP_INTSTS + 0x1, 0x1);
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (int_sts & VPP_INT_GOVRH2_VBIE)
+ vppif_reg32_out(REG_VPP_INTSTS, 0x00004000);
+ if (int_sts & VPP_INT_GOVRH2_VBIS)
+ vppif_reg32_out(REG_VPP_INTSTS, 0x00002000);
+ if (int_sts & VPP_INT_GOVRH2_PVBI)
+ vppif_reg32_out(REG_VPP_INTSTS, 0x00001000);
+#endif
+}
+
+void vppm_set_module_reset(vpp_mod_t mod)
+{
+ unsigned int value1 = 0x00, value2 = 0x00;
+ unsigned int value3 = 0x00;
+
+#ifdef WMT_FTBLK_SCL
+ if (mod == VPP_MOD_SCL)
+ value1 |= BIT0;
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ if (mod == VPP_MOD_GOVRH)
+ value2 |= (BIT0 | BIT8 | BIT9);
+#endif
+ vppif_reg32_out(REG_VPP_SWRST1_SEL, ~value1);
+ vppif_reg32_out(REG_VPP_SWRST1_SEL, 0x1010101);
+ vppif_reg32_out(REG_VPP_SWRST2_SEL, ~value2);
+ vppif_reg32_out(REG_VPP_SWRST2_SEL, 0x1011311);
+ vppif_reg32_out(REG_VPP_SWRST3_SEL, ~value3);
+ vppif_reg32_out(REG_VPP_SWRST3_SEL, 0x10101);
+}
+
+void vppm_reg_dump(void)
+{
+ unsigned int reg1, reg2;
+
+ DPRINT("========== VPPM register dump ==========\n");
+ vpp_reg_dump(REG_VPP_BEGIN, REG_VPP_END - REG_VPP_BEGIN);
+
+ DPRINT("---------- VPP Interrupt ----------\n");
+ reg1 = vppif_reg32_in(REG_VPP_INTSTS);
+ reg2 = vppif_reg32_in(REG_VPP_INTEN);
+ DPRINT("GOVRH PVBI(En %d,%d),VBIS(En %d,%d),VBIE(En %d,%d)\n",
+ vppif_reg32_read(VPP_GOVRH_INTEN_PVBI),
+ vppif_reg32_read(VPP_GOVRH_INTSTS_PVBI),
+ vppif_reg32_read(VPP_GOVRH_INTEN_VBIS),
+ vppif_reg32_read(VPP_GOVRH_INTSTS_VBIS),
+ vppif_reg32_read(VPP_GOVRH_INTEN_VBIE),
+ vppif_reg32_read(VPP_GOVRH_INTSTS_VBIE));
+
+ DPRINT("GOVRH2 PVBI(En %d,%d),VBIS(En %d,%d),VBIE(En %d,%d)\n",
+ vppif_reg32_read(VPP_GOVRH2_INTEN_PVBI),
+ vppif_reg32_read(VPP_GOVRH2_INTSTS_PVBI),
+ vppif_reg32_read(VPP_GOVRH2_INTEN_VBIS),
+ vppif_reg32_read(VPP_GOVRH2_INTSTS_VBIS),
+ vppif_reg32_read(VPP_GOVRH2_INTEN_VBIE),
+ vppif_reg32_read(VPP_GOVRH2_INTSTS_VBIE));
+
+ DPRINT("SCL PVBI(En %d,%d),VBIS(En %d,%d),VBIE(En %d,%d)\n",
+ vppif_reg32_read(VPP_SCL_INTEN_PVBI),
+ vppif_reg32_read(VPP_SCL_INTSTS_PVBI),
+ vppif_reg32_read(VPP_SCL_INTEN_VBIS),
+ vppif_reg32_read(VPP_SCL_INTSTS_VBIS),
+ vppif_reg32_read(VPP_SCL_INTEN_VBIE),
+ vppif_reg32_read(VPP_SCL_INTSTS_VBIE));
+}
+
+#ifdef CONFIG_PM
+void vppm_suspend(int sts)
+{
+ switch (sts) {
+ case 0: /* disable module */
+ break;
+ case 1: /* disable tg */
+ break;
+ case 2: /* backup register */
+ p_vppm->reg_bk = vpp_backup_reg(REG_VPP_BEGIN,
+ (REG_VPP_END - REG_VPP_BEGIN));
+ break;
+ default:
+ break;
+ }
+}
+
+void vppm_resume(int sts)
+{
+ switch (sts) {
+ case 0: /* restore register */
+ vpp_restore_reg(REG_VPP_BEGIN, (REG_VPP_END - REG_VPP_BEGIN),
+ p_vppm->reg_bk);
+ p_vppm->reg_bk = 0;
+ break;
+ case 1: /* enable module */
+ break;
+ case 2: /* enable tg */
+ break;
+ default:
+ break;
+ }
+}
+#else
+#define vppm_suspend NULL
+#define vppm_resume NULL
+#endif
+
+void vppm_init(void *base)
+{
+ vppm_mod_t *mod_p;
+
+ mod_p = (vppm_mod_t *) base;
+
+ vppm_set_module_reset(0);
+ vppm_set_int_enable(VPP_FLAG_ENABLE, mod_p->int_catch);
+ vppm_clean_int_status(~0);
+}
+
+int vppm_mod_init(void)
+{
+ vppm_mod_t *mod_p;
+
+ mod_p = (vppm_mod_t *) vpp_mod_register(VPP_MOD_VPPM,
+ sizeof(vppm_mod_t), 0);
+ if (!mod_p) {
+ DPRINT("*E* VPP module register fail\n");
+ return -1;
+ }
+
+ /* module member variable */
+ mod_p->int_catch = VPP_INT_NULL;
+
+ /* module member function */
+ mod_p->init = vppm_init;
+ mod_p->dump_reg = vppm_reg_dump;
+ mod_p->get_sts = vppm_get_int_status;
+ mod_p->clr_sts = vppm_clean_int_status;
+ mod_p->suspend = vppm_suspend;
+ mod_p->resume = vppm_resume;
+
+ p_vppm = mod_p;
+ return 0;
+}
+module_init(vppm_mod_init);
diff --git a/drivers/video/wmt/vppm.h b/drivers/video/wmt/vppm.h
new file mode 100644
index 00000000..c0d3a418
--- /dev/null
+++ b/drivers/video/wmt/vppm.h
@@ -0,0 +1,62 @@
+/*++
+ * linux/drivers/video/wmt/vppm.h
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#include "vpp.h"
+
+/* #ifdef WMT_FTBLK_VPP */
+
+#ifndef VPPM_H
+#define VPPM_H
+
+typedef struct {
+ VPP_MOD_BASE;
+} vppm_mod_t;
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifdef VPPM_C
+#define EXTERN
+#else
+#define EXTERN extern
+#endif
+
+EXTERN vppm_mod_t *p_vppm;
+
+EXTERN void vppm_set_int_enable(vpp_flag_t enable, vpp_int_t int_bit);
+EXTERN int vppm_get_int_enable(vpp_int_t int_bit);
+EXTERN vpp_int_t vppm_get_int_status(void);
+EXTERN void vppm_clean_int_status(vpp_int_t int_sts);
+EXTERN void vppm_set_module_reset(vpp_mod_t mod_bit);
+EXTERN int vppm_mod_init(void);
+EXTERN void vppm_set_DVO_select(int govr1);
+EXTERN void vppm_set_NA_mode(int hi_prio);
+
+#undef EXTERN
+
+#ifdef __cplusplus
+}
+#endif
+#endif /* VPP_H */
+/* #endif */ /* WMT_FTBLK_VPP */
diff --git a/drivers/video/wmt/wmt-mb.c b/drivers/video/wmt/wmt-mb.c
new file mode 100644
index 00000000..375d53c8
--- /dev/null
+++ b/drivers/video/wmt/wmt-mb.c
@@ -0,0 +1,3357 @@
+/*++
+ * WonderMedia Memory Block driver
+ *
+ * Copyright c 2010 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 10F, 529, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+/*----------------------------------------------------------------------------
+ * MODULE : driver/video/wmt/memblock.c
+ * DATE : 2008/12/08
+ * DESCRIPTION : Physical memory management driver
+ * HISTORY :
+ * Version 1.0.0.2 , 2013/11/12
+*----------------------------------------------------------------------------*/
+#ifndef MEMBLOCK_C
+#define MEMBLOCK_C
+#endif
+
+#include <linux/version.h>
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/fs.h>
+#include <linux/kernel.h>
+#include <linux/platform_device.h>
+#include <linux/errno.h>
+#include <linux/kdev_t.h>
+#include <linux/cdev.h>
+#include <linux/mman.h>
+#include <linux/list.h>
+#include <linux/proc_fs.h>
+#include <asm/page.h>
+#include <asm/uaccess.h>
+#include <asm/cacheflush.h>
+#include <linux/interrupt.h>
+#include <linux/spinlock.h>
+#include <linux/delay.h>
+#include <linux/dma-mapping.h>
+#include <linux/major.h>
+#include <linux/sched.h>
+#include <linux/swap.h>
+#include <mach/wmt_env.h>
+
+#include "com-mb.h"
+#include <linux/wmt-mb.h>
+
+#define THE_MB_USER "WMT-MB"
+#define MB_VERSION_MAJOR 1
+#define MB_VERSION_MINOR 0
+#define MB_VERSION_MICRO 0
+#define MB_VERSION_BUILD 3
+
+#define MB_DEBUG
+#define MB_INFO(fmt, args...) \
+ printk(KERN_INFO "["THE_MB_USER"] " fmt , ## args)
+#define MB_WARN(fmt, args...) \
+ printk(KERN_WARNING "["THE_MB_USER" *W*] " fmt, ## args)
+#define MB_ERROR(fmt, args...) \
+ printk(KERN_ERR "["THE_MB_USER" *E*] " fmt , ## args)
+
+#ifdef MB_DEBUG
+#define MB_OPDBG(fmt, args...) { /* FILE OPERATION DBG */ \
+ if (MBMSG_LEVEL) { \
+ printk("\t" KERN_DEBUG "["THE_MB_USER"] %s: " fmt, \
+ __func__ , ## args); \
+ } \
+}
+#define MB_DBG(fmt, args...) { \
+ if (MBMSG_LEVEL > 1) { \
+ printk("\t" KERN_DEBUG "["THE_MB_USER"] %s: " fmt, \
+ __func__ , ## args); \
+ } \
+}
+#define MB_WDBG(fmt, args...) { \
+ if (MBMSG_LEVEL > 2) { \
+ printk("\t" KERN_DEBUG "["THE_MB_USER"] %s: " fmt, \
+ __func__ , ## args); \
+ } \
+}
+#else
+#define MB_OPDBG(fmt, args...) do {} while (0)
+#define MB_DBG(fmt, args...) do {} while (0)
+#define MB_WDBG(fmt, args...) do {} while (0)
+#endif
+
+#ifdef CONFIG_WMT_MB_RESERVE_FROM_IO
+#include <linux/fb.h>
+#define MBIO_FSCREENINFO FBIOGET_FSCREENINFO
+#define MBA_MAX_ORDER 20 /* 2^20 pages = 4 GB */
+#else
+#define MBA_MAX_ORDER (MAX_ORDER - 1) /* 2^10 pages = 4 MB */
+#endif
+#define MBA_MIN_ORDER (MAX_ORDER - 5) /* 2^6 pages = 256 KB */
+
+#define MBAFLAG_STATIC 0x80000000 /* static, without release if empty */
+
+#define MBFLAG_CACHED 0x00000001 /* MB mapping as cachable */
+
+#define MBFIND_VIRT 0x00000000 /* search virt address */
+#define MBFIND_PHYS 0x00000001 /* search phys address */
+
+#define MBUFIND_USER 0x00000001 /* search user address */
+#define MBUFIND_VIRT 0x00000002 /* search kernel virt address */
+#define MBUFIND_PHYS 0x00000004 /* search kernel phys address */
+#define MBUFIND_ALL 0x00000010 /* search all mbu */
+#define MBUFIND_CREATOR 0x00000020 /* search creator */
+#define MBUFIND_MMAP 0x00000040 /* search mbu which mapping from user */
+#define MBUFIND_GETPUT 0x00000080 /* search mbu which came from get/put */
+#define MBUFIND_ADDRMASK (MBUFIND_USER|MBUFIND_VIRT|MBUFIND_PHYS)
+
+#define MBPROC_BUFSIZE (16*1024)
+#define MBA_SHOW_BUFSIZE (8*1024)
+#define MB_SHOW_BUFSIZE (4*1024)
+
+#define MB_COUNT(mb) atomic_read(&(mb)->count)
+#define MB_IN_USE(mb) (MB_COUNT(mb)) /* count != 0 */
+#define PAGE_KB(a) ((a)*(PAGE_SIZE/1024))
+#define MB_COMBIND_MBA(h, t) { \
+ (h)->pgi.pfn_end = (t)->pgi.pfn_end; \
+ (h)->pages += (t)->pages; \
+ (h)->tot_free_pages += (t)->tot_free_pages; \
+ (h)->max_available_pages += (t)->max_available_pages; \
+ list_del(&((t)->mba_list)); \
+ wmt_mbah->nr_mba--; \
+ kmem_cache_free(wmt_mbah->mba_cachep, t); \
+}
+
+#define list_loop list_for_each_entry
+struct mb_user;
+
+struct page_info {
+ unsigned long pfn_start; /* start pfn of this block */
+ unsigned long pfn_end; /* end pfn of this block */
+};
+
+struct mba_host_struct {
+ /* MB area link list link all MBAs */
+ struct list_head mba_list;
+
+ /* page information of this MBA */
+ unsigned int nr_mba;
+
+ /* total pages of all manager MBAs */
+ unsigned long tot_pages;
+
+ /* total static pages of all manager MBAs */
+ unsigned long tot_static_pages;
+
+ /* total free pages of all manager MBAs */
+ unsigned long tot_free_pages;
+
+ /* max free pages of all manager MBAs */
+ unsigned long max_available_pages;
+
+ /* allocator of MBA,
+ use slab to prevent memory from fragment. */
+ struct kmem_cache *mba_cachep;
+
+ /* allocator of MB,
+ use slab to prevent memory from fragment. */
+ struct kmem_cache *mb_cachep;
+
+ /* allocator of mb_user,
+ use slab to prevent memory from fragment. */
+ struct kmem_cache *mbu_cachep;
+
+ /* allocator of mb_task_info,
+ use slab to prevent memory from fragment. */
+ struct kmem_cache *mbti_cachep;
+};
+
+struct mb_area_struct {
+ /* MB area link list link all MBAs */
+ struct list_head mba_list;
+
+ /* link list link to all MBs
+ belong to this MBA */
+ struct list_head mb_list;
+
+ /* pointer point to dedicated MBAH */
+ struct mba_host_struct *mbah;
+
+ /* flags of MBA */
+ unsigned int flags;
+
+ /* prefech record task id */
+ pid_t tgid;
+
+ /* start physical address of this MBA */
+ unsigned long phys;
+
+ /* start virtual address of this MBA */
+ void *virt;
+
+ /* size of this MB area. Normally,
+ MAX kernel permitted size */
+ unsigned long pages;
+
+ /* page information of this MBA */
+ struct page_info pgi;
+
+ /* page information of this MBA */
+ unsigned int nr_mb;
+
+ /* cur total free pages of this MBA */
+ unsigned long tot_free_pages;
+
+ /* cur max free pages of this MBA */
+ unsigned long max_available_pages;
+};
+
+struct mb_struct {
+ /* MB link list link all MBs
+ in dedicated MBA */
+ struct list_head mb_list;
+
+ /* pointer point to dedicated MBA */
+ struct mb_area_struct *mba;
+
+ /* MB kernel page information */
+ struct page_info pgi;
+
+ /* start physical address of this MB */
+ unsigned long phys;
+
+ /* start virtual address of this MB */
+ void *virt;
+
+ /* allocate size */
+ unsigned long size;
+
+ /* current MB use count,
+ release until zero */
+ atomic_t count;
+
+ /* flags of MB */
+ unsigned int flags;
+
+ /* point to owner created the mb.
+ this enlisted in mbu_list */
+ struct mb_user *creator;
+
+ /* use for trace the user of mb */
+ struct list_head mbu_list;
+};
+
+struct mb_user {
+ /* user link list link all users
+ (include creator) of belonged MB */
+ struct list_head mbu_list;
+
+ /* the mb to which this user belong */
+ struct mb_struct *mb;
+
+ /* task id to which user belonged,
+ user space user only */
+ pid_t tgid;
+
+ /* mb_mmap and MBIO_GET : user address
+ mb_get and mb->creator : physical address */
+ unsigned long addr;
+
+ /* kernel space user: mb size
+ * user space user: mmap size
+ * zero: owner - user space allocate but not mapped yet
+ * not owner -come from get/put */
+ unsigned long size;
+
+ /* user name for recoder */
+ char the_user[TASK_COMM_LEN+1];
+};
+
+struct mb_task_info {
+ /* task link list link all tasks which use MBDev */
+ struct list_head mbti_list;
+ pid_t tgid; /* task pid */
+ atomic_t count; /* multi open record */
+ char task_name[TASK_COMM_LEN+1];
+ struct task_struct *task;
+};
+
+#ifdef CONFIG_WMT_MB_SIZE
+static int MB_TOTAL_SIZE = CONFIG_WMT_MB_SIZE * 1024;
+#else
+static int MB_TOTAL_SIZE = 32 * 1024;
+#endif
+static struct mba_host_struct *wmt_mbah;
+static struct mb_task_info wmt_mbti;
+const struct file_operations mb_fops;
+static unsigned char MBMSG_LEVEL;
+static unsigned char MBMAX_ORDER;
+static unsigned char MBMIN_ORDER;
+static unsigned char USR2PRDT_METHOD;
+
+/* read/write spinlock for multientry protection. */
+static spinlock_t mb_do_lock;
+static spinlock_t mb_search_lock;
+static spinlock_t mb_ioctl_lock;
+static spinlock_t mb_task_mm_lock;
+static spinlock_t mb_task_lock;
+static struct page *pg_user[12800]; /* 12800 pages = 50 MB */
+static char show_mb_buffer[MB_SHOW_BUFSIZE];
+static char show_mba_buffer[MBA_SHOW_BUFSIZE];
+
+#define MMU_SEARCH_DONE 1
+typedef int (*mb_page_proc)(void *, dma_addr_t, int);
+
+static inline int mmu_search_pte(
+ pte_t *pte,
+ unsigned long boundary,
+ unsigned long *addr,
+ unsigned long *offset,
+ mb_page_proc proc,
+ void *priv)
+{
+ int ret = 0, index = 0;
+
+ /* for each pte */
+ while (pte && (*addr < boundary)) {
+ unsigned long pfn = pte_pfn(*pte);
+ struct page *pg = pfn_to_page(pfn);
+ void *virt = page_address(pg);
+ dma_addr_t phys = virt_to_phys(virt);
+
+ MB_WDBG("\t\t[PTE%3d] PTE[%p]=%x n %lx V %p P %x U %lx[%lx]\n",
+ index++, pte, pte_val(*pte), pfn,
+ virt, phys, *addr, *offset);
+ if (!pfn_valid(pfn)) {
+ MB_WARN("invalid pfn %ld of addr %lx\n", pfn, *addr);
+ return -1;
+ }
+ if (*offset < PAGE_SIZE) {
+ ret = proc(priv, phys, *offset);
+ if (ret)
+ break;
+ *offset = 0;
+ } else
+ *offset -= PAGE_SIZE;
+ *addr += PAGE_SIZE;
+ pte++;
+#ifdef MB_WORDY
+ msleep(30);
+#endif
+ };
+ return ret;
+}
+
+static inline int mmu_search_pmd(
+ pmd_t *pmd,
+ unsigned long boundary,
+ unsigned long *addr,
+ unsigned long *offset,
+ mb_page_proc proc,
+ void *priv)
+{
+ int ret = 0, index = 0;
+ unsigned long end;
+ pte_t *pte;
+
+ /* for each pmd */
+ while (pmd && (*addr < boundary)) {
+ /* from start to PMD alignment */
+ end = (*addr + PMD_SIZE) & PMD_MASK;
+ end = min(end, boundary);
+ pte = pte_offset_map(pmd, *addr);
+ if (pte == NULL) {
+ MB_WARN("[%08lx] *pmd=%08llx unknown pte\n",
+ *addr, (long long)pmd_val(*pmd));
+ return -1;
+ }
+ MB_WDBG("\t[PMD%3d, %lx, %lx] *(%p)=%x addr %lx\n", index++,
+ end - PMD_SIZE, end, pmd, pmd_val(*pmd), *addr);
+ ret = mmu_search_pte(pte, end, addr, offset, proc, priv);
+ if (ret)
+ break;
+ pmd++;
+ };
+ return ret;
+}
+
+static inline int mmu_search_pgd(
+ pgd_t *pgd,
+ unsigned long boundary,
+ unsigned long *addr,
+ unsigned long *offset,
+ mb_page_proc proc,
+ void *priv)
+{
+ int ret = 0, index = 0;
+ unsigned long end;
+ pmd_t *pmd;
+ pud_t *pud;
+
+ /* for each pgd */
+ while (pgd && (*addr < boundary)) {
+ /* from start to PGDIR alignment */
+ end = (*addr + PGDIR_SIZE) & PGDIR_MASK;
+ end = min(end, boundary);
+ MB_WDBG("[PGD%3d, %lx, %lx] *(%p)=%x addr %lx\n", index++,
+ end - PGDIR_SIZE, end, pgd, pgd_val(*pgd), *addr);
+ pud = pud_offset(pgd, *addr);
+ if (pud_none(*pud) || pud_bad(*pud)) {
+ MB_WARN("[%08lx] *pgd=%08llx %s pud\n",
+ *addr, (long long)pgd_val(*pgd),
+ pud_none(*pud) ? "(None)" : "(Bad)");
+ return -1;
+ }
+ pmd = pmd_offset(pud, *addr);
+ if (pmd == NULL) {
+ MB_WARN("[%08lx] *pgd=%08llx unknown pmd\n",
+ *addr, (long long)pgd_val(*pgd));
+ return -1;
+ }
+ ret = mmu_search_pmd(pmd, end, addr, offset, proc, priv);
+ if (ret)
+ break;
+ pgd++;
+ };
+
+ return ret;
+}
+
+struct prdt_search_info {
+ int index;
+ int items;
+ unsigned int size;
+ struct prdt_struct *prev;
+ struct prdt_struct *next;
+};
+
+static int __user_to_prdt_proc(void *priv, dma_addr_t phys, int offset)
+{
+ struct prdt_search_info *info = (struct prdt_search_info *)priv;
+ struct prdt_struct *prev = info->prev, *next = info->next;
+ int len = PAGE_SIZE - offset;
+
+ if (len > info->size)
+ len = info->size;
+
+ /* Check it could combind with previous one */
+ if (prev &&
+ /* prd size boundary check, MAX 60K */
+ (prev->size <= ((1 << 16) - (2 * PAGE_SIZE))) &&
+ /* page continuity check */
+ ((prev->addr + prev->size) == phys))
+ prev->size += len;
+ else { /* create new one */
+ info->prev = prev = next;
+ info->next = next + 1;
+ prev->addr = phys + offset;
+ prev->size = len;
+ info->index++;
+ info->items--;
+ if (!info->items) {
+ MB_WARN("PRD table full (ptr %p # %d left %d).\n",
+ next, info->index, info->size);
+ prev->EDT = 1;
+ return -1;
+ }
+ }
+ info->size -= len;
+ prev->reserve = 0;
+ prev->EDT = (info->size) ? 0 : 1;
+ MB_WDBG("\t\t\t[PRD %3d] %p start %x(%x + %d) size %x edt %x left %x\n",
+ info->index, prev, prev->addr, phys, offset,
+ prev->size, prev->EDT, info->size);
+
+ if (prev->EDT)
+ return MMU_SEARCH_DONE;
+
+ return 0;
+}
+
+static int __user_to_prdt(
+ unsigned long user,
+ unsigned int size,
+ struct prdt_struct *prdt,
+ unsigned int items)
+{
+ struct mm_struct *mm = current->mm;
+ struct vm_area_struct *vma;
+ unsigned long addr, end, offset;
+ struct prdt_search_info info = {0};
+ unsigned int idx;
+ pgd_t *pgd;
+ int ret;
+
+ if (!prdt || ((size / PAGE_SIZE) + 2) > items) {
+ MB_WARN("PRD table space not enough (ptr %p at least %lu).\n",
+ prdt, (size/PAGE_SIZE)+2);
+ return -EINVAL;
+ }
+
+ MB_DBG("Memory(%#lx,%d) PRDT(%p,%d)\n", user, size, prdt, items);
+ info.size = size;
+ info.next = prdt;
+ info.items = items;
+ addr = user;
+
+ down_read(&mm->mmap_sem);
+ while (info.size > 0) {
+ vma = find_vma(mm, addr);
+ if (vma == NULL) {
+ MB_WARN("user addr %lx not found in task %s\n",
+ addr, current->comm);
+ info.next->EDT = 1;
+ goto fault;
+ }
+
+ MB_WDBG("VMA found: mm %p start %lx end %lx flags %lx\n",
+ mm, vma->vm_start, vma->vm_end, vma->vm_flags);
+ end = PAGE_ALIGN(vma->vm_end);
+ offset = addr - vma->vm_start;
+ addr = vma->vm_start;
+ pgd = pgd_offset(mm, vma->vm_start);
+ ret = mmu_search_pgd(pgd, end, &addr, &offset,
+ __user_to_prdt_proc, &info);
+ if (ret == MMU_SEARCH_DONE)
+ break;
+ if (ret)
+ goto fault;
+ }
+
+ MB_WDBG("PRDT %p, from %lx size %d\n", prdt, user, size);
+ for (idx = 0;; idx++) {
+ MB_WDBG("PRDT[%d] adddr %x size %d EDT %d\n",
+ idx, prdt[idx].addr, prdt[idx].size, prdt[idx].EDT);
+ dmac_flush_range(__va(prdt[idx].addr),
+ __va(prdt[idx].addr + prdt[idx].size));
+ outer_flush_range(prdt[idx].addr,
+ prdt[idx].addr + prdt[idx].size);
+ if (prdt[idx].EDT)
+ break;
+ }
+ up_read(&mm->mmap_sem);
+ return 0;
+fault:
+ MB_WARN("USER TO PRDT unfinished, remain size %d\n", info.size);
+ up_read(&mm->mmap_sem);
+ return -EFAULT;
+}
+
+static int __user_to_prdt1(
+ unsigned long user,
+ unsigned int size,
+ struct prdt_struct *next,
+ unsigned int items)
+{
+ void *ptr_start, *ptr_end, *ptr, *virt = NULL;
+ int res, pg_size, pg_idx = 0, nr_pages = 0;
+ struct vm_area_struct *vma;
+
+ ptr_start = ptr = (void *)user;
+ ptr_end = ptr_start + size;
+ MB_DBG("Memory(%#lx,%d) PRDT(%p,%d)\n", user, size, next, items);
+
+ down_read(&current->mm->mmap_sem);
+ vma = find_vma(current->mm, user);
+ up_read(&current->mm->mmap_sem);
+ /* For kernel direct-mapped memory, take the easy way */
+ if (vma && (vma->vm_flags & VM_IO) && (vma->vm_pgoff)) {
+ unsigned long phys = 0;
+ /* kernel-allocated, mmaped-to-usermode addresses */
+ phys = (vma->vm_pgoff << PAGE_SHIFT) + (user - vma->vm_start);
+ virt = __va(phys);
+ MB_INFO("kernel-alloc, mmaped-to-user addr U %lx V %p P %lx",
+ user, virt, phys);
+ BUG_ON(1);
+ return -EFAULT;
+ }
+ MB_WDBG("VMA found: mm %p start %lx end %lx flags %lx\n",
+ current->mm, vma->vm_start, vma->vm_end, vma->vm_flags);
+
+ nr_pages = (size + (PAGE_SIZE - 1)) / PAGE_SIZE;
+ if (!next || ((size/PAGE_SIZE)+2) > items || nr_pages > 2560) {
+ MB_WARN("PRD table space full (ptr %p pages %d prdt %d/%lu).\n",
+ next, nr_pages, items, (size/PAGE_SIZE)+2);
+ return -EINVAL;
+ }
+
+ memset(pg_user, 0x0, sizeof(struct page *)*2560);
+ down_read(&current->mm->mmap_sem);
+ res = get_user_pages(current, current->mm,
+ (unsigned long)ptr, nr_pages, 1, 0, pg_user, NULL);
+ while (res > 0 && size > 0) {
+ pg_size = PAGE_SIZE - ((unsigned long)ptr & ~PAGE_MASK);
+ virt = page_address(pg_user[pg_idx]) +
+ ((unsigned long)ptr & ~PAGE_MASK);
+ if (pg_size > size)
+ pg_size = size;
+ MB_DBG("Get %d-th user page s %d/%d u %p v %p p %lx\n",
+ pg_idx, pg_size, size, ptr, virt, __pa(virt));
+ if ((next->addr + next->size) != __pa(virt) ||
+ (next->size + pg_size) >= 65536 || !pg_idx) {
+ if (pg_idx) {
+ next->EDT = 0;
+ next++;
+ }
+ memset(next, 0x0, sizeof(struct prdt_struct));
+ next->addr = __pa(virt);
+ }
+ next->size += pg_size;
+ next->EDT = 1;
+ size -= pg_size;
+ ptr += pg_size;
+ pg_idx++;
+ }
+ next->EDT = 1;
+ up_read(&current->mm->mmap_sem);
+ return size;
+}
+
+static int __user_to_prdt2(
+ unsigned long user,
+ unsigned int size,
+ struct prdt_struct *next,
+ unsigned int items)
+{
+ void *ptr_start, *ptr_end, *ptr, *virt = NULL;
+ int res, pg_size, pg_idx = 0, ret = 0;
+ struct vm_area_struct *vma;
+ struct page *pages = NULL;
+
+ ptr_start = ptr = (void *)user;
+ ptr_end = ptr_start + size;
+ MB_DBG("Memory(%#lx,%d) PRDT(%p,%d)\n", user, size, next, items);
+
+ down_read(&current->mm->mmap_sem);
+ vma = find_vma(current->mm, user);
+ up_read(&current->mm->mmap_sem);
+ /* For kernel direct-mapped memory, take the easy way */
+ if (vma && (vma->vm_flags & VM_IO) && (vma->vm_pgoff)) {
+ unsigned long phys = 0;
+ /* kernel-allocated, mmaped-to-usermode addresses */
+ phys = (vma->vm_pgoff << PAGE_SHIFT) + (user - vma->vm_start);
+ virt = __va(phys);
+ MB_INFO("kernel-alloc, user-mmaped addr U %lx V %p P %lx",
+ user, virt, phys);
+ BUG_ON(1);
+ return -EFAULT;
+ }
+
+ if (!next || ((size/PAGE_SIZE)+2) > items) {
+ MB_WARN("PRD table space full (ptr %p at least %lu)\n",
+ next, (size/PAGE_SIZE)+2);
+ return -EINVAL;
+ }
+
+ MB_WDBG("VMA found: mm %p start %lx end %lx flags %lx\n",
+ current->mm, vma->vm_start, vma->vm_end, vma->vm_flags);
+
+ while (size) {
+ down_read(&current->mm->mmap_sem);
+ res = get_user_pages(current, current->mm,
+ (unsigned long)ptr, 1, 1, 0, &pages, NULL);
+ up_read(&current->mm->mmap_sem);
+ pg_size = PAGE_SIZE - ((unsigned long)ptr & ~PAGE_MASK);
+ if (res != 1) {
+ MB_ERROR("Get %d-th user pages (a %p s %d/%d) fail\n",
+ pg_idx, ptr, pg_size, size);
+ next->EDT = 1;
+ ret = -EFAULT;
+ break;
+ }
+ virt = page_address(&pages[0]) +
+ ((unsigned long)ptr & ~PAGE_MASK);
+ pg_size = (pg_size > size) ? size : pg_size;
+ MB_DBG("Get %d-th user page s %d/%d u %p v %p p %lx\n",
+ pg_idx, pg_size, size, ptr, virt, __pa(virt));
+ if ((next->addr + next->size) != __pa(virt) ||
+ (next->size + pg_size) >= 65536 || !pg_idx) {
+ if (pg_idx) {
+ next->EDT = 0;
+ next++;
+ }
+ memset(next, 0x0, sizeof(struct prdt_struct));
+ next->addr = __pa(virt);
+ }
+ next->size += pg_size;
+ next->EDT = 1;
+ size -= pg_size;
+ ptr += pg_size;
+ pg_idx++;
+ }
+ return ret;
+}
+
+static void show_prdt(struct prdt_struct *next)
+{
+ int idx = 1;
+ while (!next->EDT) {
+ MB_INFO("PRDT %d-th item: addr %x size %d EDT %d\n",
+ idx, next->addr, next->size, next->EDT);
+ idx++;
+ next++;
+ }
+ MB_INFO("PRDT last(%d-th) item: addr %x size %d EDT %d\n",
+ idx, next->addr, next->size, next->EDT);
+}
+
+/*!*************************************************************************
+* wmt_mmu_table_size
+*
+* Public Function
+*/
+/*!
+* \brief
+* estimate request mmu table size for input size
+*
+* \parameter
+* size [IN] convert size
+*
+* \retval size of needed mmu table
+*/
+unsigned int wmt_mmu_table_size(unsigned int size)
+{
+ unsigned int nPte, nPde, need;
+
+ if (!size)
+ return 0;
+
+ nPte = ((PAGE_ALIGN(size)) / PAGE_SIZE) + 1;
+ nPde = ALIGN(nPte, 1024) / 1024;
+
+ need = (ALIGN(nPde, 1024) / 1024) * PAGE_SIZE;
+ need += nPte * 4;
+
+ printk(KERN_DEBUG "PDE %d PTE %d RequestSize %d\n", nPde, nPte, need);
+ return need;
+}
+EXPORT_SYMBOL(wmt_mmu_table_size);
+
+/*!*************************************************************************
+* wmt_mmu_table_check
+*
+* Public Function
+*/
+/*!
+* \brief
+* check pre-allocated mmu table is valid for input size
+*
+* \parameter
+* mmu_addr [IN] mmu table address
+* mmu_size [IN] mmu table size
+* size [IN] convert size
+*
+* \retval 1 if success
+*/
+int wmt_mmu_table_check(
+ unsigned int *mmu_addr,
+ unsigned int mmu_size,
+ unsigned int size)
+{
+ unsigned int nPte, nPde, request;
+
+ if (!size) {
+ printk(KERN_WARNING "mmu create failure. NULL size\n");
+ return 0;
+ }
+
+ if (!mmu_size) {
+ printk(KERN_WARNING "mmu create failure. NULL mmu size\n");
+ return 0;
+ }
+
+ if ((unsigned int)mmu_addr % PAGE_SIZE) {
+ printk(KERN_WARNING "mmu create fail. PDE Table not align\n");
+ return 0;
+ }
+
+ nPte = ((PAGE_ALIGN(size)) / PAGE_SIZE) + 1;
+ nPde = ALIGN(nPte, 1024) / 1024;
+
+ request = (ALIGN(nPde, 1024) / 1024) * PAGE_SIZE;
+ request += nPte * 4;
+
+ if (mmu_size < request) {
+ printk(KERN_WARNING "mmu create fail. out of mmu size\n");
+ printk(KERN_WARNING "(pde %d pte %d request %d but %d)\n",
+ nPde, nPte, request, mmu_size);
+ return 0;
+ }
+
+ return 1;
+}
+EXPORT_SYMBOL(wmt_mmu_table_check);
+
+/*!*************************************************************************
+* wmt_mmu_table_dump
+*
+* Public Function
+*/
+/*!
+* \brief
+* dump mmu table
+*
+* \parameter
+* mmu_addr [IN] mmu table address
+* size [IN] convert size
+* virBufAddr [IN] offset combination of PDE, PTE, and ADDRESS
+*
+* \retval none
+*/
+void wmt_mmu_table_dump(struct mmu_table_info *info)
+{
+ unsigned int *mmu_addr;
+ unsigned int size;
+ unsigned int virBufAddr;
+ unsigned int *pte = NULL, *pde = NULL;
+ unsigned int pdeOffset, pteOffset, addrOffset, i = 0;
+
+ if (info == 0) {
+ printk(KERN_ERR "[WMT_MMU_TABLE] Null input pointer\n");
+ return;
+ }
+
+ mmu_addr = (unsigned int *)mb_phys_to_virt(info->addr);
+
+ size = info->size;
+ virBufAddr = info->offset;
+
+ addrOffset = virBufAddr % PAGE_SIZE;
+ pteOffset = (virBufAddr >> 12) % 1024;
+ pdeOffset = (virBufAddr >> 22) % 1024;
+
+ printk(KERN_INFO "MMU (%x): offset pde %x pte %x addr %x\n",
+ virBufAddr, pdeOffset, pteOffset, addrOffset);
+
+ pde = mmu_addr;
+ pde += pdeOffset;
+ pte = mb_phys_to_virt(*pde);
+ pte += pteOffset;
+
+ while (size > 0) {
+ printk(KERN_INFO "[%5d] PDE(%p/%lx) -> PTE(%p/%lx) -> addr %x\n",
+ i, pde, mb_virt_to_phys(pde), pte, mb_virt_to_phys(pte),
+ *pte + addrOffset);
+ if ((size + addrOffset) < PAGE_SIZE)
+ break;
+ size -= PAGE_SIZE - addrOffset;
+ addrOffset = 0;
+ i++;
+ pte++;
+ if (!(i % 1024))
+ pde++;
+ }
+}
+EXPORT_SYMBOL(wmt_mmu_table_dump);
+
+/*!*************************************************************************
+* wmt_mmu_table_from_phys
+*
+* Public Function
+*/
+/*!
+* \brief
+* make up mmu table for physical memory
+*
+* \parameter
+* mmu_addr [IN] mmu table address
+* mmu_size [IN] mmu table size
+* addr [IN] physical address of convert data
+* size [IN] convert size
+*
+* \retval 0xFFFFFFFF if fail
+* offset combination of PDE, PTE, and ADDRESS if success
+*/
+unsigned int wmt_mmu_table_from_phys(
+ unsigned int *mmu_addr,
+ unsigned int mmu_size,
+ unsigned int addr,
+ unsigned int size)
+{
+ unsigned int iPte, nPte, nPde, gPde, virBufAddr;
+ unsigned int *pte = NULL, *pde = NULL;
+
+ iPte = gPde = nPte = nPde = 0;
+ if (!wmt_mmu_table_check(mmu_addr, mmu_size, size)) {
+ printk(KERN_WARNING
+ "phys %x (size %d) to mmu fail\n", addr, size);
+ return 0xFFFFFFFF;
+ }
+
+ virBufAddr = addr % PAGE_SIZE;
+ nPte = ((PAGE_ALIGN(size)) / PAGE_SIZE);
+ if (virBufAddr)
+ nPte++;
+ nPde = ALIGN(nPte, 1024) / 1024;
+ gPde = ALIGN(nPde, 1024) / 1024;
+
+ pde = mmu_addr;
+ pte = pde + (gPde * PAGE_SIZE / sizeof(pte));
+ *pde = mb_virt_to_phys(pte);
+ pde++;
+
+ size += virBufAddr;
+ while (size > 0) {
+ *pte = addr & PAGE_MASK;
+ pte++; iPte++;
+ if (!(iPte % 1024)) {
+ *pde = mb_virt_to_phys(pte);
+ pde++;
+ }
+ if (size < PAGE_SIZE)
+ break;
+ size -= PAGE_SIZE;
+ addr += PAGE_SIZE;
+ }
+
+ return virBufAddr;
+}
+EXPORT_SYMBOL(wmt_mmu_table_from_phys);
+
+
+struct mmu_search_info {
+ int iPte;
+ unsigned *pte;
+ unsigned *pde;
+ unsigned int size;
+};
+
+static int wmt_mmu_table_from_user_proc(
+ void *priv, dma_addr_t phys, int offset)
+{
+ struct mmu_search_info *info = (struct mmu_search_info *)priv;
+
+ *info->pte = phys & PAGE_MASK;
+
+ printk(KERN_DEBUG
+ "\t\t\t[PTE] PDE[%d](%p)=%x PTE[%d](%p)=%x left %d\n",
+ info->iPte/1024, info->pde, *info->pde,
+ info->iPte, info->pte, *info->pte, info->size);
+ info->size -= PAGE_SIZE;
+ if (info->size < PAGE_SIZE)
+ return MMU_SEARCH_DONE;
+ info->iPte++;
+ info->pte++;
+ if (!(info->iPte % 1024)) {
+ *info->pde = mb_virt_to_phys(info->pte);
+ info->pde++;
+ }
+ return 0;
+}
+
+/*!*************************************************************************
+* wmt_mmu_table_from_user
+*
+* Public Function
+*/
+/*!
+* \brief
+* make up mmu table for user memory
+*
+* \parameter
+* mmu_addr [IN] mmu table address
+* mmu_size [IN] mmu table size
+* addr [IN] physical address of convert data
+* size [IN] convert size
+*
+* \retval 0xFFFFFFFF if fail
+* offset combination of PDE, PTE, and ADDRESS if success
+*/
+unsigned int wmt_mmu_table_from_user(
+ unsigned int *mmu_addr,
+ unsigned int mmu_size,
+ unsigned int user,
+ unsigned int size)
+{
+#ifndef __KERNEL__
+ return wmt_mmu_table_from_phys(mmu_addr, mmu_size, user, size);
+#else
+ struct mm_struct *mm = current->mm;
+ struct vm_area_struct *vma;
+ unsigned long addr, end, offset;
+ unsigned int nPte, nPde, virBufAddr;
+ struct mmu_search_info info = {0};
+ pgd_t *pgd;
+ int ret;
+
+ nPte = nPde = 0;
+ if (!wmt_mmu_table_check(mmu_addr, mmu_size, size)) {
+ printk(KERN_WARNING "phys %x (size %d) to mmu fail\n",
+ user, size);
+ return 0xFFFFFFFF;
+ }
+
+ virBufAddr = user % PAGE_SIZE;
+ nPte = ((PAGE_ALIGN(size)) / PAGE_SIZE);
+ if (virBufAddr)
+ nPte++;
+ nPde = ALIGN(ALIGN(nPte, 1024) / 1024, 1024) / 1024;
+
+ info.pde = mmu_addr;
+ info.pte = info.pde + (nPde * PAGE_SIZE / sizeof(info.pte));
+ *info.pde = mb_virt_to_phys(info.pte);
+ info.pde++;
+
+ printk(KERN_DEBUG "Memory(%#x,%d)\n", user, size);
+ down_read(&mm->mmap_sem);
+ info.size = size + virBufAddr;
+ addr = (unsigned long)user;
+ while (info.size > 0) {
+ vma = find_vma(mm, addr);
+ if (vma == NULL) {
+ printk(KERN_WARNING
+ "user addr %lx not found in task %s\n",
+ addr, current->comm);
+ goto fault;
+ }
+ printk(KERN_DEBUG
+ "VMA found: start %lx end %lx\n",
+ vma->vm_start, vma->vm_end);
+ offset = addr - vma->vm_start;
+ addr = vma->vm_start;
+ end = PAGE_ALIGN(vma->vm_end);
+ pgd = pgd_offset(mm, vma->vm_start);
+ ret = mmu_search_pgd(pgd, end, &addr, &offset,
+ wmt_mmu_table_from_user_proc, &info);
+ if (ret == MMU_SEARCH_DONE)
+ break;
+ if (ret)
+ goto fault;
+ }
+ up_read(&mm->mmap_sem);
+ return virBufAddr;
+fault:
+ printk(KERN_WARNING "USER TO PRDT unfinished, remain size %d\n", size);
+ up_read(&mm->mmap_sem);
+ return 0xFFFFFFFF;
+#endif
+}
+EXPORT_SYMBOL(wmt_mmu_table_from_user);
+
+/*!*************************************************************************
+* wmt_mmu_table_create
+*
+* Public Function
+*/
+/*!
+* \brief
+* automatically make up mmu table from giving address
+*
+* \parameter
+* addr [IN] start address (user of phys depends on addr_type)
+* size [IN] size of memory
+* addr_type [IN] address type (0: phys 1: user)
+* info [out] pointer of mmu table info
+*
+* \retval 0 if success, otherwise error code
+*/
+unsigned int wmt_mmu_table_create(
+ unsigned int addr,
+ unsigned int size,
+ unsigned int addr_type,
+ struct mmu_table_info *info)
+{
+ unsigned int mmuSize = 0;
+ unsigned int *mmuAddr = NULL;
+
+ if (!addr || !size || !info || addr_type > 1) {
+ printk(KERN_ERR "[WMT_MMU_TABLE] invalid args:\n");
+ printk(KERN_ERR "\t addr %x # %x type %x info %p\n",
+ addr, size, addr_type, info);
+ return -EINVAL;
+ }
+
+ mmuSize = wmt_mmu_table_size(size);
+
+ info->addr = mb_alloc(mmuSize);
+ if (!info->addr) {
+ printk(KERN_ERR "[WMT_MMU_TABLE] create fail. Out of MB (%d)\n",
+ mmuSize);
+ return -ENOMEM;
+ }
+ info->size = size;
+ mmuAddr = (unsigned int *)mb_phys_to_virt(info->addr);
+
+ info->offset = (addr_type) ?
+ wmt_mmu_table_from_user(mmuAddr, mmuSize, addr, size) :
+ wmt_mmu_table_from_phys(mmuAddr, mmuSize, addr, size);
+
+ if (info->offset == 0xFFFFFFFF) {
+ printk(KERN_ERR "[WMT_MMU_TABLE] create fail:");
+ printk(KERN_ERR "\ttype %x addr %x # %x mmuAddr %p/%x # %d\n",
+ addr_type, addr, size, mmuAddr, info->addr, mmuSize);
+ return -EFAULT;
+ }
+
+ return 0;
+}
+EXPORT_SYMBOL(wmt_mmu_table_create);
+
+/*!*************************************************************************
+* wmt_mmu_table_destroy
+*
+* Public Function
+*/
+/*!
+* \brief
+* automatically make up mmu table from giving address
+*
+* \parameter
+* info [IN] start address (user of phys depends on addr_type)
+*
+* \retval 1 if success
+*/
+unsigned int wmt_mmu_table_destroy(struct mmu_table_info *info)
+{
+ if (!info) {
+ printk(KERN_ERR "[WMT_MMU_TABLE] destroy fail. NULL mmu table info");
+ return -EINVAL;
+ }
+ mb_free(info->addr);
+ return 0;
+}
+EXPORT_SYMBOL(wmt_mmu_table_destroy);
+
+/* return address is guaranteeed only under page alignment */
+void *user_to_virt(unsigned long user)
+{
+ struct vm_area_struct *vma;
+ unsigned long flags;
+ void *virt = NULL;
+
+ spin_lock_irqsave(&mb_task_mm_lock, flags);
+ vma = find_vma(current->mm, user);
+ /* For kernel direct-mapped memory, take the easy way */
+ if (vma && (vma->vm_flags & VM_IO) && (vma->vm_pgoff)) {
+ unsigned long phys = 0;
+ /* kernel-allocated, mmaped-to-usermode addresses */
+ phys = (vma->vm_pgoff << PAGE_SHIFT) + (user - vma->vm_start);
+ virt = __va(phys);
+ MB_INFO("%s kernel-alloc, user-mmaped addr U %lx V %p P %lx",
+ __func__, user, virt, phys);
+ } else {
+ /* otherwise, use get_user_pages() for general userland pages */
+ int res, nr_pages = 1;
+ struct page *pages;
+ down_read(&current->mm->mmap_sem);
+ res = get_user_pages(current, current->mm,
+ user, nr_pages, 1, 0, &pages, NULL);
+ up_read(&current->mm->mmap_sem);
+ if (res == nr_pages)
+ virt = page_address(&pages[0]) + (user & ~PAGE_MASK);
+ MB_INFO("%s userland addr U %lx V %p", __func__, user, virt);
+ }
+ spin_unlock_irqrestore(&mb_task_mm_lock, flags);
+
+ return virt;
+}
+EXPORT_SYMBOL(user_to_virt);
+
+int user_to_prdt(
+ unsigned long user,
+ unsigned int size,
+ struct prdt_struct *next,
+ unsigned int items)
+{
+ int ret;
+
+ switch (USR2PRDT_METHOD) {
+ case 1:
+ ret = __user_to_prdt1(user, size, next, items);
+ break;
+ case 2:
+ ret = __user_to_prdt2(user, size, next, items);
+ break;
+ default:
+ ret = __user_to_prdt(user, size, next, items);
+ break;
+ }
+
+ if (MBMSG_LEVEL > 1)
+ show_prdt(next);
+
+ return ret;
+}
+EXPORT_SYMBOL(user_to_prdt);
+
+static int mb_show_mb(
+ struct mb_struct *mb,
+ char *msg,
+ char *str,
+ int size
+)
+{
+ struct mb_user *mbu;
+ char *p;
+
+ if (!str && MBMSG_LEVEL <= 1)
+ return 0;
+
+ if (!mb)
+ return 0;
+
+ memset(show_mb_buffer, 0x0, MB_SHOW_BUFSIZE);
+ p = show_mb_buffer;
+ p += sprintf(p, "%s%s[MB,%p] %p %08lx %4ldKB [%4lx,%4lx] %3x\n",
+ msg ? msg : "", msg ? " " : "", mb, mb->virt, mb->phys,
+ mb->size/1024, mb->pgi.pfn_start, mb->pgi.pfn_end,
+ MB_COUNT(mb));
+
+ if (!list_empty(&mb->mbu_list)) {
+ list_loop(mbu, &mb->mbu_list, mbu_list) {
+ char tmp[256], *p1 = tmp;
+ int expectLen = (int)(p - show_mb_buffer);
+ p1 += sprintf(tmp, " [%s%s,%p] %08lx ",
+ (mbu == mb->creator) ? "O" : " ",
+ (mbu->tgid == MB_DEF_TGID) ? "K" : "U",
+ mbu, mbu->addr);
+ p1 += sprintf(p1, "# %4ldKB By %s(%d)\n",
+ mbu->size/1024, mbu->the_user, mbu->tgid);
+ expectLen += strlen(tmp) + 1;
+ if (expectLen > MB_SHOW_BUFSIZE) {
+ p += sprintf(p, "\n\n .......\n");
+ break;
+ }
+ p += sprintf(p, "%s", tmp);
+ }
+ }
+ if (str) {
+ if (strlen(show_mb_buffer) < size)
+ size = strlen(show_mb_buffer);
+ strncpy(str, show_mb_buffer, size);
+ } else {
+ size = strlen(show_mb_buffer);
+ MB_DBG("%s", show_mb_buffer);
+ }
+
+ return size;
+}
+
+static int mb_show_mba(
+ struct mb_area_struct *mba,
+ char *msg,
+ char *str,
+ int size,
+ int follow
+)
+{
+ struct mb_struct *mb;
+ char *p;
+ int idx = 1;
+
+ if (!str && MBMSG_LEVEL <= 1)
+ return 0;
+
+ if (!mba)
+ return 0;
+
+ memset(show_mba_buffer, 0x0, MBA_SHOW_BUFSIZE);
+
+ p = show_mba_buffer;
+ if (msg)
+ p += sprintf(p, "%s ", msg);
+ p += sprintf(p, "[%p] %p %08lx %3ldMB [%5lx,%5lx] ",
+ mba, mba->virt, mba->phys, PAGE_KB(mba->pages)/1024,
+ mba->pgi.pfn_start, mba->pgi.pfn_end);
+ p += sprintf(p, "%3x %6ldKB/%6ldKB %s\n", mba->nr_mb,
+ PAGE_KB(mba->max_available_pages),
+ PAGE_KB(mba->tot_free_pages),
+ (mba->flags | MBAFLAG_STATIC) ? "S" : "D");
+ if (follow) {
+ p += sprintf(p, " index - [MemBlock] VirtAddr PhysAddr");
+ p += sprintf(p, " size [ zs, ze] cnt\n");
+ list_loop(mb, &mba->mb_list, mb_list) {
+ if ((MBA_SHOW_BUFSIZE - (p - show_mba_buffer)) < 22) {
+ p += sprintf(p, "\n\n more ...\n");
+ break;
+ }
+ p += sprintf(p, " %5d - ", idx++);
+ /* -2 is for \n and zero */
+ p += mb_show_mb(mb, NULL, p,
+ MBA_SHOW_BUFSIZE - (p - show_mba_buffer) - 2);
+ }
+ }
+
+ p += sprintf(p, "\n");
+
+ if (str) {
+ if (strlen(show_mba_buffer) < size)
+ size = strlen(show_mba_buffer);
+ strncpy(str, show_mba_buffer, size);
+ } else {
+ size = strlen(show_mba_buffer);
+ MB_DBG("%s", show_mba_buffer);
+ }
+
+ return size;
+}
+
+/* return physical address */
+#ifdef CONFIG_WMT_MB_RESERVE_FROM_IO
+/* mb_do_lock locked */
+static void *mb_alloc_pages(unsigned long *phys, unsigned long *nr_pages)
+{
+ unsigned int order = get_order(num_physpages << PAGE_SHIFT);
+ unsigned int size = *nr_pages << PAGE_SHIFT;
+ unsigned long addr = ((1 << order) - *nr_pages) << PAGE_SHIFT;
+ void *virt = NULL;
+
+ if (!nr_pages || !phys) {
+ MB_WARN("mb_alloc_pages fail. unknown argument\n");
+ return NULL;
+ }
+
+ *phys = 0;
+ if (*nr_pages == 0) {
+ MB_WARN("mb_alloc_pages zero size\n");
+ return NULL;
+ }
+
+ if (!request_mem_region(addr, size, "memblock")) {
+ MB_WARN("request memory region fail. addr %#lx size %d(KB).\n",
+ addr, size/1024);
+ return NULL;
+ }
+
+ virt = ioremap(addr, size);
+ if (!virt) {
+ MB_WARN("cannot ioremap memory. addr %#lx size %d(KB).\n",
+ addr, size/1024);
+ release_mem_region(addr, size);
+ return NULL;
+ }
+
+ *phys = addr;
+ MB_DBG("allocate mem region. addr V %p P %#lx size %d(KB)\n",
+ virt, addr, size/1024);
+
+ return virt;
+}
+
+/* mb_do_lock locked */
+static void mb_free_pages(void *virt, unsigned long phys, unsigned int nr_pages)
+{
+ iounmap(virt);
+ MB_DBG("release mem region. addr V %p P %#lx size %d(KB)\n",
+ virt, phys, 4*nr_pages);
+ return release_mem_region(phys, nr_pages << PAGE_SHIFT);
+}
+
+#else
+/* mb_do_lock locked */
+static void *mb_alloc_pages(unsigned long *phys, unsigned long *nr_pages)
+{
+ unsigned int i, order;
+ unsigned long virt = 0;
+ struct zone *zone;
+
+ if (!nr_pages || !phys) {
+ MB_WARN("mb_alloc_pages fail. unknown argument\n");
+ return NULL;
+ }
+
+ *phys = 0;
+ if (*nr_pages == 0) {
+ MB_WARN("mb_alloc_pages zero size\n");
+ return NULL;
+ }
+
+ order = get_order(*nr_pages * PAGE_SIZE);
+ order = max(order, (unsigned int)(MBA_MIN_ORDER));
+ MB_DBG(" %ld/%d pages, order %d\n", *nr_pages, 1 << order, order);
+ if (order > MBMAX_ORDER) {
+ MB_WARN("mb_alloc_mba fail. page out of size, %ld/%d/%d\n",
+ *nr_pages, (1<<order), (1<<MBMAX_ORDER));
+ return NULL;
+ }
+
+ zone = (first_online_pgdat())->node_zones;
+ for (i = order; i < MAX_ORDER; i++) {
+ if (zone->free_area[i].nr_free) {
+ virt = __get_free_pages(GFP_ATOMIC | GFP_DMA, order);
+ break;
+ }
+ }
+
+ if (virt) {
+ *nr_pages = 1 << order;
+ *phys = __pa(virt);
+ for (i = 0; i < *nr_pages; i++)
+ SetPageReserved(virt_to_page(virt + i * PAGE_SIZE));
+ MB_DBG("allocate mem region. addr V %#lx P %#lx size %ld(KB)\n",
+ virt, *phys, PAGE_KB(*nr_pages));
+ } else {
+ struct free_area *fa;
+ char msg[256], *p = msg;
+ MB_WARN("__get_free_pages fail! (pages: %d free %lu)\n",
+ 1 << order, (unsigned long)nr_free_pages());
+ zone = (first_online_pgdat())->node_zones;
+ fa = zone->free_area;
+ p += sprintf(msg, "DMA ZONE:");
+ for (i = 0; i < MAX_ORDER; i++)
+ p += sprintf(p, " %ld*%dKB", fa[i].nr_free, 4 << i);
+ MB_WARN("%s= %ldkB\n", msg, PAGE_KB(nr_free_pages()));
+ }
+
+ return (void *)virt;
+}
+
+/* mb_do_lock locked */
+static void mb_free_pages(void *virt, unsigned long phys, unsigned int nr_pages)
+{
+ unsigned int index;
+
+ if (!phys || !nr_pages) {
+ MB_WARN("mb_free_pages unknow addr V %p P %#lx size %d pages\n",
+ virt, phys, nr_pages);
+ return;
+ }
+
+ for (index = 0; index < nr_pages; index++) {
+ unsigned long addr = (unsigned long)virt + index * PAGE_SIZE;
+ ClearPageReserved(virt_to_page(addr));
+ }
+
+ free_pages((unsigned long)virt, get_order(nr_pages * PAGE_SIZE));
+}
+#endif
+
+/* mb_do_lock locked */
+static struct mb_area_struct *mb_alloc_mba(unsigned int pages)
+{
+ struct mba_host_struct *mbah = wmt_mbah;
+ struct mb_area_struct *mba;
+
+ if (!mbah || !pages) {
+ MB_WARN("mb_alloc_mba fail. mbah %p, pages %d\n", mbah, pages);
+ return NULL;
+ }
+
+ mba = kmem_cache_alloc(mbah->mba_cachep, GFP_ATOMIC);
+ if (!mba) {
+ MB_WARN("mb_alloc_mba fail. out of memory\n");
+ return NULL;
+ }
+
+ memset(mba, 0x0, sizeof(struct mb_area_struct));
+ mba->pages = pages;
+ mba->virt = mb_alloc_pages(&mba->phys, &mba->pages);
+ if (!mba->virt) {
+ MB_WARN("mb_alloc_mba fail. no available space\n");
+ kmem_cache_free(mbah->mba_cachep, mba);
+ return NULL;
+ }
+
+ /* initialization */
+ INIT_LIST_HEAD(&mba->mb_list);
+ INIT_LIST_HEAD(&mba->mba_list);
+ mba->mbah = mbah;
+ mba->tot_free_pages = mba->max_available_pages = mba->pages;
+ mba->pgi.pfn_start = mba->phys >> PAGE_SHIFT;
+ mba->pgi.pfn_end = mba->pgi.pfn_start + mba->pages;
+ list_add_tail(&mba->mba_list, &mbah->mba_list);
+
+ /* update MBA host */
+ mbah->nr_mba++;
+ mbah->tot_pages += mba->pages;
+ mbah->tot_free_pages += mba->tot_free_pages;
+ if (mbah->max_available_pages < mba->max_available_pages)
+ mbah->max_available_pages = mba->max_available_pages;
+ mb_show_mba(mba, "alloc", NULL, 0, 0);
+
+ return mba;
+}
+
+/* mb_do_lock locked */
+static int mb_free_mba(struct mb_area_struct *mba)
+{
+ struct mba_host_struct *mbah;
+ struct mb_area_struct *entry;
+
+ if (!mba || !mba->mbah || mba->nr_mb) {
+ MB_WARN("mb_free_mba fail. unknow arg.(%p,%p,%x)\n",
+ mba, mba ? mba->mbah : NULL, mba ? mba->nr_mb : 0);
+ return -EFAULT;
+ }
+
+ mbah = mba->mbah;
+ list_loop(entry, &mbah->mba_list, mba_list) {
+ if (entry == mba)
+ break;
+ }
+
+ if (entry != mba) {
+ MB_WARN("mb_free_mba fail. unknow MBA %p\n", mba);
+ return -EFAULT;
+ }
+ if (mba->flags & MBAFLAG_STATIC)
+ return 0;
+
+ mb_show_mba(mba, "ReleaseMBA", NULL, 0, 0);
+
+ /* free mba */
+ list_del(&mba->mba_list);
+ mb_free_pages(mba->virt, mba->phys, mba->pages);
+ mbah->nr_mba--;
+ mbah->tot_free_pages -= mba->tot_free_pages;
+ mbah->tot_pages -= mba->pages;
+ kmem_cache_free(mbah->mba_cachep, mba);
+ mba = NULL;
+
+ /* update max mb size */
+ mbah->max_available_pages = 0;
+ list_loop(entry, &mbah->mba_list, mba_list) {
+ if (mbah->max_available_pages < entry->max_available_pages)
+ mbah->max_available_pages = entry->max_available_pages;
+ }
+
+ return 0;
+}
+
+/* mb_do_lock locked */
+static struct mb_struct *mb_alloc_mb(
+ struct mb_area_struct *mba,
+ unsigned long size)
+{
+ struct mba_host_struct *mbah;
+ struct mb_struct *mb, *entry;
+ struct list_head *next;
+ unsigned long pages, zs, ze;
+
+ if (!mba || !mba->mbah || !size) {
+ MB_WARN("mb_alloc_mb fail. unknow arg.(%p,%p,%lx)\n",
+ mba, mba ? mba->mbah : NULL, size);
+ return NULL;
+ }
+
+ mbah = mba->mbah;
+ size = PAGE_ALIGN(size);
+ pages = size >> PAGE_SHIFT;
+
+ /* mb free size is not enough */
+ if (mba->max_available_pages < pages) {
+ MB_WARN("mb_alloc_mb fail. no space in MBA (%lx<%lx)\n",
+ pages, mba->max_available_pages);
+ return NULL;
+ }
+
+ /* search available zone
+ zone start from mba start */
+ ze = zs = mba->pgi.pfn_start;
+ next = &mba->mb_list;
+ list_loop(entry, &mba->mb_list, mb_list) {
+ next = &entry->mb_list;
+ ze = entry->pgi.pfn_start;
+ if ((ze - zs) >= pages)
+ break;
+ zs = entry->pgi.pfn_end;
+ }
+
+ if (zs >= ze) {
+ next = &mba->mb_list;
+ ze = mba->pgi.pfn_end;
+ }
+
+ /* impossible, something wrong */
+ if ((ze - zs) < pages) {
+ MB_WARN("something wrong in MBA %p when allocate MB.\n", mba);
+ return NULL;
+ }
+
+ MB_DBG("Zone finding start %lx end %lx size %lx for size %lx\n",
+ zs, ze, ze - zs, pages);
+
+ mb = kmem_cache_alloc(mbah->mb_cachep, GFP_ATOMIC);
+ if (!mb) {
+ MB_WARN("mb_alloc_mb mb_cachep out of memory\n");
+ return NULL;
+ }
+
+ memset(mb, 0x0, sizeof(struct mb_struct));
+ INIT_LIST_HEAD(&mb->mb_list);
+ INIT_LIST_HEAD(&mb->mbu_list);
+ mb->mba = mba;
+ mb->pgi.pfn_start = zs;
+ mb->pgi.pfn_end = mb->pgi.pfn_start + pages;
+ mb->size = size;
+ mb->phys = mba->phys + ((zs - mba->pgi.pfn_start) << PAGE_SHIFT);
+ mb->virt = mba->virt + (mb->phys - mba->phys);
+ list_add_tail(&mb->mb_list, next);
+
+ mba->nr_mb++;
+ mba->tot_free_pages -= pages;
+ mbah->tot_free_pages -= pages;
+
+ /* update mba */
+ zs = mba->pgi.pfn_start;
+ mba->max_available_pages = 0;
+ list_loop(entry, &mba->mb_list, mb_list) {
+ mba->max_available_pages =
+ max(entry->pgi.pfn_start - zs, mba->max_available_pages);
+ zs = entry->pgi.pfn_end;
+ }
+ mba->max_available_pages =
+ max(mba->pgi.pfn_end - zs, mba->max_available_pages);
+
+ /* update mbah */
+ mbah->max_available_pages = 0;
+ list_loop(mba, &mbah->mba_list, mba_list) {
+ if (mbah->max_available_pages < mba->max_available_pages)
+ mbah->max_available_pages = mba->max_available_pages;
+ }
+
+ mb_show_mb(mb, "alloc", NULL, 0);
+
+ return mb;
+}
+
+/* mb_do_lock locked */
+static int mb_free_mb(struct mb_struct *mb)
+{
+ unsigned long zs, nr_pages;
+ struct mba_host_struct *mbah;
+ struct mb_area_struct *mba;
+ struct mb_struct *entry;
+
+ if (!mb) {
+ MB_WARN("mb_free_mb fail. unknow MB %p.\n", mb);
+ return -EFAULT;
+ }
+
+ if (MB_IN_USE(mb)) {
+ MB_WARN("mb_free_mb fail. invalid arg.(%p,%x,%lx,%ldKB)\n",
+ mb, MB_COUNT(mb), mb->phys, mb->size/1024);
+ return -EINVAL;
+ }
+
+ mba = mb->mba;
+ if (!mba || !mba->mbah) {
+ MB_WARN("mb_free_mb fail. unknow para.(%p,%p)\n",
+ mba, mba ? mba->mbah : NULL);
+ return -EFAULT;
+ }
+
+ list_loop(entry, &mba->mb_list, mb_list) {
+ if (entry == mb)
+ break;
+ }
+
+ if (entry != mb) {
+ MB_WARN("mb_free_mb fail. unknow MB %p\n", mb);
+ return -EFAULT;
+ }
+
+ mbah = mba->mbah;
+
+ mb_show_mb(mb, "Retrieve unused MB", NULL, 0);
+
+ /* free mb */
+ list_del(&mb->mb_list);
+ kmem_cache_free(mbah->mb_cachep, mb);
+ mba->nr_mb--;
+ mb = NULL;
+
+ /* unused mba, release it */
+ if (!mba->nr_mb && !(mba->flags & MBAFLAG_STATIC))
+ return mb_free_mba(mba);
+
+ /* update max mb size and free mb size */
+ /* reduce old one and then increase new one */
+ mbah->tot_free_pages -= mba->tot_free_pages;
+ zs = mba->pgi.pfn_start;
+ mba->tot_free_pages = mba->max_available_pages = nr_pages = 0;
+ list_loop(entry, &mba->mb_list, mb_list) {
+ nr_pages = max(entry->pgi.pfn_start - zs, nr_pages);
+ mba->tot_free_pages += entry->pgi.pfn_start - zs;
+ zs = entry->pgi.pfn_end;
+ }
+ mba->tot_free_pages += mba->pgi.pfn_end - zs;
+ mba->max_available_pages = max(mba->pgi.pfn_end - zs, nr_pages);
+ mbah->tot_free_pages += mba->tot_free_pages; /* add new one */
+ mbah->max_available_pages =
+ max(mbah->max_available_pages, mba->max_available_pages);
+
+ return 0;
+}
+
+/* type 0 - virtual address
+ * 1 - physical address */
+static struct mb_struct *mb_find_mb(void *addr, int type)
+{
+ struct mba_host_struct *mbah = wmt_mbah;
+ struct mb_area_struct *mba;
+ struct mb_struct *mb;
+ unsigned long flags;
+
+ if (!addr || (type != MBFIND_PHYS && type != MBFIND_VIRT)) {
+ MB_WARN("mb_find_mb fail. unknow type %d addr %p\n",
+ type, addr);
+ return NULL;
+ }
+
+ MB_DBG("IN, type %x addr %p\n", type, addr);
+
+ spin_lock_irqsave(&mb_search_lock, flags);
+
+ list_loop(mba, &mbah->mba_list, mba_list) {
+ void *eaddr, *saddr = (void *)(mba->phys);
+ if (type != MBFIND_PHYS)
+ saddr = mba->virt;
+ eaddr = saddr + mba->pages*PAGE_SIZE;
+ if (addr < saddr || addr > eaddr)
+ continue; /* address out of mba range */
+ list_loop(mb, &mba->mb_list, mb_list) {
+ void *mbaddr = (void *)(mb->phys);
+ if (type != MBFIND_PHYS)
+ mbaddr = mb->virt;
+ if (addr >= mbaddr && addr < (mbaddr + mb->size)) {
+ spin_unlock_irqrestore(&mb_search_lock, flags);
+ MB_DBG("OUT, va %p pa %lx s %ld(KB) cnt %d\n",
+ mb->virt, mb->phys, mb->size/1024,
+ MB_COUNT(mb));
+ return mb;
+ }
+ }
+ }
+
+ spin_unlock_irqrestore(&mb_search_lock, flags);
+
+ MB_DBG("OUT, NULL mb\n");
+
+ return NULL;
+}
+
+/*
+ * addr - search address
+ * tgid - task pid
+ * st - search type (MBUFIND_XXX)
+ * MBUFIND_USER: user address
+ * MBUFIND_VIRT: kernel virt address
+ * MBUFIND_PHYS: kernel phys address
+ * MBUFIND_ALL:
+ * 1. mbu->tgid == tgid
+ * 2. tgid == MB_DEF_TGID, address and
+ * one of MBUFIND_VIRT and MBUFIND_PHYS must be set
+ * tgid != MB_DEF_TGID, address and MBUFIND_USER must be set
+ * MBUFIND_CREATOR:
+ * 1. mbu->tgid == tgid
+ * 2. tgid == MB_DEF_TGID, address and
+ * one of MBUFIND_VIRT and MBUFIND_PHYS must be set
+ * tgid != MB_DEF_TGID, address and MBUFIND_USER must be set
+ * 3. mbu == mb->creator
+ * 4. mbu->size == mb->size
+ * MBUFIND_GETPUT:
+ * 1. mbu->tgid == tgid
+ * 2. tgid == MB_DEF_TGID, address and
+ * MBUFIND_PHYS must be set (mbu->addr == mb->phys)
+ * tgid != MB_DEF_TGID, address and
+ * MBUFIND_USER must be set (mbu->addr == user address)
+ * 3. mbu->size == 0
+ * MBUFIND_MMAP:
+ * 1. mbu->tgid == tgid
+ * 2. address and MBUFIND_USER must be set
+ * 3. mbu->size != 0
+ */
+static struct mb_user *mb_find_mbu(void *addr, pid_t tgid, int st)
+{
+ struct mba_host_struct *mbah = wmt_mbah;
+ struct mb_area_struct *mba = NULL;
+ struct mb_struct *mb = NULL;
+ struct mb_user *mbu = NULL;
+ unsigned long flags;
+
+ if (!addr || !(st & MBUFIND_ADDRMASK)) {
+ MB_WARN("mb_find_mbu fail. unknow addr %p\n", addr);
+ return NULL;
+ }
+
+ MB_DBG("IN, addr %p search type %x TGID %x CurTask %s\n",
+ addr, st, tgid, current->comm);
+
+ spin_lock_irqsave(&mb_search_lock, flags);
+ list_loop(mba, &mbah->mba_list, mba_list) {
+ if (st & (MBUFIND_VIRT | MBUFIND_PHYS)) {
+ void *eaddr, *saddr = (void *)mba->phys;
+ if (st & MBUFIND_VIRT)
+ saddr = mba->virt;
+ eaddr = saddr + mba->pages*PAGE_SIZE;
+ if (addr < saddr || addr > eaddr)
+ continue; /* address out of mba range */
+ }
+ list_loop(mb, &mba->mb_list, mb_list) {
+ if ((st & MBUFIND_PHYS) && (addr != (void *)mb->phys))
+ continue; /* physical address not match */
+ if ((st & MBUFIND_VIRT) && (addr != mb->virt))
+ continue; /* virtual address not match */
+ list_loop(mbu, &mb->mbu_list, mbu_list) {
+ if (mbu->tgid != tgid && tgid != MB_DEF_TGID)
+ continue; /* tgid not match */
+ if ((st & MBUFIND_USER) &&
+ (addr != (void *)mbu->addr))
+ continue; /* user address not match */
+ if (st & MBUFIND_ALL)
+ goto leave; /* found */
+ if (st & MBUFIND_CREATOR) {
+ mbu = mb->creator;
+ goto leave; /* found */
+ }
+ /* found (if mmap, mbu->size has map size) */
+ if (st & MBUFIND_MMAP && mbu->size)
+ goto leave;
+ /* found (if get/put, mbu->size should be 0) */
+ if (st & MBUFIND_GETPUT &&
+ mbu->addr && !mbu->size)
+ goto leave;
+ }
+ }
+ }
+ mbu = NULL;
+
+leave:
+ if (mbu == NULL)
+ MB_DBG("OUT, NULL mbu\n");
+
+ if (mbu)
+ MB_DBG("OUT, mbu %p (%p TGID %x a %#lx s %ld(KB) by %s)\n",
+ mbu, mbu->mb, mbu->tgid, mbu->addr,
+ mbu->size/1024, mbu->the_user);
+
+ spin_unlock_irqrestore(&mb_search_lock, flags);
+
+ return mbu;
+}
+
+static void mb_update_mbah(void)
+{
+ unsigned long mbah_free = 0, mbah_max = 0;
+ struct mba_host_struct *mbah = wmt_mbah;
+ struct mb_area_struct *mba = NULL;
+ struct mb_struct *mb = NULL;
+
+ if (!mbah) {
+ MB_WARN("mb_update_mbah fail. unknow mbah\n");
+ return;
+ }
+
+ list_loop(mba, &mbah->mba_list, mba_list) {
+ unsigned long mba_free = 0, mba_max = 0, nr_pages = 0;
+ unsigned long zs = mba->pgi.pfn_start; /* zone start */
+ list_loop(mb, &mba->mb_list, mb_list) {
+ nr_pages = max(mb->pgi.pfn_start - zs, nr_pages);
+ mba_free += mb->pgi.pfn_start - zs;
+ zs = mb->pgi.pfn_end;
+ }
+ mba_free += mba->pgi.pfn_end - zs;
+ mbah_free += mba_free;
+ mba_max = max(mba->pgi.pfn_end - zs, nr_pages);
+ mbah_max = max(mbah_max, mba_max);
+ }
+
+ mbah->tot_free_pages = mbah_free;
+ mbah->max_available_pages = mbah_max;
+
+ return;
+}
+
+/* called from kernel only, mb_do_lock isn't needed */
+void *mb_do_phys_to_virt(unsigned long phys, pid_t tgid, char *name)
+{
+ struct mb_struct *mb = NULL;
+ void *virt = NULL;
+
+ MB_DBG("IN, Phys %lx TGID %x NAME %s\n", phys, tgid, name);
+ mb = mb_find_mb((void *)phys, MBFIND_PHYS);
+ if (mb)
+ virt = mb->virt + (phys - mb->phys);
+ else {
+ virt = __va(phys);
+ MB_WARN("%s do phys to virt fail. addr %lx not found\n",
+ name, phys);
+ }
+
+ MB_DBG("OUT, Virt %p\n", virt);
+
+ return virt;
+}
+EXPORT_SYMBOL(mb_do_phys_to_virt);
+
+/* called from kernel only, mb_do_lock isn't needed */
+unsigned long mb_do_virt_to_phys(void *virt, pid_t tgid, char *name)
+{
+ struct mb_struct *mb = NULL;
+ unsigned long phys = 0x0;
+
+ MB_DBG("IN, Virt %p TGID %x NAME %s\n", virt, tgid, name);
+ mb = mb_find_mb(virt, MBFIND_VIRT);
+ if (mb)
+ phys = mb->phys + (virt - mb->virt);
+ else {
+ phys = __pa(virt);
+ MB_WARN("%s do virt to phys fail. addr %p not found\n",
+ name, virt);
+ }
+
+ MB_DBG("OUT, Phys %lx\n", phys);
+
+ return phys;
+}
+EXPORT_SYMBOL(mb_do_virt_to_phys);
+
+unsigned long mb_do_user_to_phys(unsigned long user, pid_t tgid, char *name)
+{
+ unsigned long flags, phys = 0;
+ struct mb_user *mbu = NULL;
+
+ MB_DBG("IN, usr %lx TGID %x name %s\n", user, tgid, name);
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mbu = mb_find_mbu((void *)user, tgid, MBUFIND_USER|MBUFIND_ALL);
+ if (!mbu) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("%s do user to phys. unknow addr %lx\n", name, user);
+ return 0;
+ }
+
+ if (!mbu->mb)
+ MB_WARN("%s do user to phys fail. unknow block (addr %lx)\n",
+ name, user);
+ else
+ phys = mbu->mb->phys;
+
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ MB_DBG("OUT, Phys %#lx\n", phys);
+
+ return phys;
+}
+EXPORT_SYMBOL(mb_do_user_to_phys);
+
+unsigned long mb_do_user_to_virt(unsigned long user, pid_t tgid, char *name)
+{
+ struct mb_user *mbu = NULL;
+ unsigned long flags;
+ void *virt = NULL;
+
+ MB_DBG("IN, usr %lx TGID %x name %s\n", user, tgid, name);
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mbu = mb_find_mbu((void *)user, tgid, MBUFIND_USER|MBUFIND_ALL);
+ if (!mbu) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("%s user to virt fail. unknow addr %lx\n", name, user);
+ return 0;
+ }
+
+ if (!mbu->mb)
+ MB_WARN("%s user to virt fail. unknow block (addr %lx)\n",
+ name, user);
+ else
+ virt = mbu->mb->virt;
+
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ MB_DBG("OUT, Virt %p\n", virt);
+
+ return (unsigned long)virt;
+}
+EXPORT_SYMBOL(mb_do_user_to_virt);
+
+/* physical address */
+int mb_do_counter(unsigned long addr, char *name)
+{
+ struct mb_struct *mb;
+ unsigned long flags;
+ int counter = -1;
+
+ MB_DBG("IN, addr %#lx name %s\n", addr, name);
+
+ if (!addr || !wmt_mbah) {
+ MB_WARN("%s do counter fail. invalid args %p/%lx\n",
+ name, wmt_mbah, addr);
+ return -EINVAL;
+ }
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mb = mb_find_mb((void *)addr, MBFIND_PHYS);
+ counter = (mb) ? MB_COUNT(mb) : -1;
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ MB_DBG("OUT, addr %#lx count %d\n", addr, counter);
+
+ return counter;
+}
+EXPORT_SYMBOL(mb_do_counter);
+
+unsigned long mb_do_alloc(unsigned long size, pid_t tgid, char *name)
+{
+ struct mb_area_struct *entry, *mba = NULL;
+ struct mb_struct *mb = NULL;
+ struct mb_user *mbu = NULL;
+ unsigned long flags, addr;
+ unsigned int pages;
+ size_t ns; /* name size */
+
+ if (!name) {
+ MB_WARN("mb_alloc fail. null user tgid %x\n", tgid);
+ return 0;
+ }
+
+ size = PAGE_ALIGN(size);
+ pages = size >> PAGE_SHIFT;
+
+ if (!pages) {
+ MB_WARN("%s alloc fail. unavailable size %x(KB)\n",
+ name, 4*pages);
+ return 0;
+ }
+
+ mbu = kmem_cache_alloc(wmt_mbah->mbu_cachep, GFP_ATOMIC);
+ if (!mbu) {
+ MB_WARN("%s alloc fail. mbu_cachep out of memory.\n", name);
+ return 0;
+ }
+ memset(mbu, 0x0, sizeof(struct mb_user));
+
+ MB_DBG("IN, TGID %x size %ld(KB) name %s\n", tgid, size/1024, name);
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+
+ if (pages > wmt_mbah->max_available_pages) {
+#ifdef CONFIG_WMT_MB_DYNAMIC_ALLOCATE_SUPPORT
+ mba = mb_alloc_mba(pages);
+ if (mba == NULL)
+ MB_WARN("%s alloc %u page fail. [MBA %lu/%lu/%lu]\n",
+ name, pages, wmt_mbah->max_available_pages,
+ wmt_mbah->tot_free_pages, wmt_mbah->tot_pages);
+#else
+ MB_DBG("DYNAMIC ALLOCATED(%u) unsuported. [MBA %lu/%lu/%lu]\n",
+ pages, wmt_mbah->max_available_pages,
+ wmt_mbah->tot_free_pages, wmt_mbah->tot_pages);
+ goto error;
+#endif
+ } else {
+ list_loop(entry, &wmt_mbah->mba_list, mba_list) {
+ if (entry->max_available_pages >= pages) {
+ mba = entry;
+ break;
+ }
+ }
+ }
+
+ if (!mba) {
+ MB_WARN("%s alloc fail. dedicated MBA not found\n", name);
+ goto error;
+ }
+
+ mb = mb_alloc_mb(mba, size);
+ if (mb == NULL) {
+ MB_WARN("%s alloc fail. create MB\n", name);
+ goto error;
+ }
+
+ INIT_LIST_HEAD(&mbu->mbu_list);
+ mbu->mb = mb;
+ mbu->tgid = tgid;
+ mbu->size = mb->size;
+ /* mbu->addr = 0; // address of creator is 0 */
+ ns = strlen(name);
+ if (ns > TASK_COMM_LEN)
+ ns = TASK_COMM_LEN;
+ strncpy(mbu->the_user, name, ns);
+ mbu->the_user[ns] = 0;
+ list_add_tail(&mbu->mbu_list, &mb->mbu_list);
+ atomic_inc(&mb->count);
+ mb->creator = mbu;
+ addr = mb->phys;
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ MB_DBG("OUT, Addr %lx TGID %x size %ld(KB) name %s\n",
+ addr, tgid, size/1024, name);
+
+ return addr;
+
+error:
+
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ return 0;
+}
+EXPORT_SYMBOL(mb_do_alloc);
+
+/* physical address */
+int mb_do_free(unsigned long addr, pid_t tgid, char *name)
+{
+ struct mb_struct *mb;
+ struct mb_user *mbu;
+ unsigned long flags;
+ size_t ns; /* name size */
+ int ret;
+
+ if (!addr || !name) {
+ MB_WARN("mb_free fail. invalid args %lx/%s\n",
+ addr, name);
+ return -EINVAL;
+ }
+
+ MB_DBG("IN, TGID %x addr %#lx name %s\n", tgid, addr, name);
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mbu = mb_find_mbu((void *)addr, tgid, MBUFIND_PHYS|MBUFIND_CREATOR);
+ if (!mbu || !mbu->mb) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("%s free fail. unknow addr %lx\n", name, addr);
+ return -EINVAL;
+ }
+
+ ns = strlen(name);
+ if (ns > TASK_COMM_LEN)
+ ns = TASK_COMM_LEN;
+ if (strncmp(mbu->the_user, name, ns))
+ MB_DBG("Owner no match. (%s/%s)\n", mbu->the_user, name);
+
+ mb = mbu->mb;
+
+ list_del(&mbu->mbu_list);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ mb->creator = NULL;
+ atomic_dec(&mb->count);
+ if (MB_IN_USE(mb)) {
+ MB_DBG("<mb_free, MB %8lx still in use. Cnt %d>\n",
+ mb->phys, MB_COUNT(mb));
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ return 0;
+ }
+
+ ret = mb_free_mb(mb);
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ MB_DBG("OUT\n");
+
+ return ret;
+}
+EXPORT_SYMBOL(mb_do_free);
+
+int mb_do_get(unsigned long addr, pid_t tgid, char *name)
+{
+ struct mb_user *mbu = NULL, *newmbu;
+ struct mb_struct *mb = NULL;
+ unsigned long flags;
+ size_t ns; /* name size */
+ void *virt = (void *)addr;
+
+ if (!addr || !name) {
+ MB_WARN("mb_do_get fail. invalid args %lx/%s\n",
+ addr, name);
+ return -EINVAL;
+ }
+
+ newmbu = kmem_cache_alloc(wmt_mbah->mbu_cachep, GFP_ATOMIC);
+ if (!newmbu) {
+ MB_DBG("<mbu_cachep out of memory.>\n");
+ return -ENOMEM;
+ }
+
+ MB_DBG("IN, TGID %x addr %#lx name %s\n", tgid, addr, name);
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ if (tgid != MB_DEF_TGID) { /* find user address, only exist on MMAP */
+ mbu = mb_find_mbu(virt, tgid, MBUFIND_USER|MBUFIND_MMAP);
+ mb = (mbu) ? mbu->mb : NULL;
+ } else
+ mb = mb_find_mb(virt, MBFIND_PHYS);
+
+ if (!mb) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ kmem_cache_free(wmt_mbah->mbu_cachep, newmbu);
+ MB_WARN("%s mb_get fail. unknown addr %8lx\n", name, addr);
+ return -EFAULT;
+ }
+
+ memset(newmbu, 0x0, sizeof(struct mb_user));
+ INIT_LIST_HEAD(&newmbu->mbu_list);
+ newmbu->addr = addr;
+ newmbu->mb = mb;
+ newmbu->tgid = tgid;
+ ns = strlen(name);
+ if (ns > TASK_COMM_LEN)
+ ns = TASK_COMM_LEN;
+ strncpy(newmbu->the_user, name, ns);
+ newmbu->the_user[ns] = 0;
+ atomic_inc(&mb->count);
+ list_add_tail(&newmbu->mbu_list, &mb->mbu_list);
+
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ MB_DBG("out, [mbu] addr %lx %s [mb] addr %lx cnt %d\n",
+ addr, newmbu->the_user, mb->phys, MB_COUNT(mb));
+
+ return 0;
+}
+EXPORT_SYMBOL(mb_do_get);
+
+int mb_do_put(unsigned long addr, pid_t tgid, char *name)
+{
+ struct mb_struct *mb;
+ struct mb_user *mbu;
+ unsigned long flags;
+ void *virt = (void *)addr;
+
+ if (!addr || !name) {
+ MB_WARN("mb_do_put fail. invalid args %lx/%s\n", addr, name);
+ return -EINVAL;
+ }
+
+ MB_DBG("IN, TGID %x addr %#lx name %s\n", tgid, addr, name);
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ if (tgid == MB_DEF_TGID)
+ mbu = mb_find_mbu(virt, tgid, MBUFIND_PHYS|MBUFIND_GETPUT);
+ else
+ mbu = mb_find_mbu(virt, tgid, MBUFIND_USER|MBUFIND_GETPUT);
+
+ if (!mbu) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("%s mb_put fail. unknow addr %8lx\n", name, addr);
+ return -EINVAL;
+ }
+
+ mb = mbu->mb;
+ if (!MB_IN_USE(mb)) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("%s mb_put fail. block %p unbalance.\n", name, mb);
+ return -EPERM;
+ }
+
+ list_del_init(&mbu->mbu_list);
+ atomic_dec(&mb->count);
+ /* retrieve unused block */
+ if (!MB_IN_USE(mb))
+ mb_free_mb(mb);
+
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+
+ MB_DBG("out, [mbu] addr %lx %s [mb] addr %lx cnt %d\n",
+ addr, mbu->the_user, mb->phys, MB_COUNT(mb));
+
+ return 0;
+}
+EXPORT_SYMBOL(mb_do_put);
+
+#define DEVICE_NAME "Memory Block"
+
+static int mb_dev_major;
+static int mb_dev_minor;
+static int mb_dev_nr;
+static struct cdev *mb_cdev;
+static struct class *mb_dev_class;
+
+static int mb_open(struct inode *inode, struct file *filp)
+{
+ struct mb_task_info *mbti = NULL;
+ unsigned long flags;
+ int task_exist = 0;
+ size_t ns;
+
+ if (filp->private_data) {
+ MB_ERROR("none empty private data.\n");
+ return -EFAULT;
+ }
+
+ spin_lock_irqsave(&mb_task_lock, flags);
+ list_loop(mbti, &wmt_mbti.mbti_list, mbti_list) {
+ if (mbti->task && mbti->tgid == current->tgid) {
+ task_exist = 1;
+ break;
+ }
+ }
+ if (!task_exist) {
+ mbti = kmem_cache_alloc(wmt_mbah->mbti_cachep, GFP_ATOMIC);
+ if (!mbti) {
+ spin_unlock_irqrestore(&mb_task_lock, flags);
+ MB_ERROR("out of memory (mb_task_info).\n");
+ return -EFAULT;
+ }
+ memset(mbti, 0x0, sizeof(struct mb_task_info));
+ INIT_LIST_HEAD(&mbti->mbti_list);
+ mbti->task = current;
+ mbti->tgid = current->tgid;
+ ns = strlen(current->comm);
+ if (ns > TASK_COMM_LEN)
+ ns = TASK_COMM_LEN;
+ strncpy(mbti->task_name, current->comm, ns);
+ mbti->task_name[ns] = 0;
+ atomic_set(&mbti->count, 0);
+ list_add_tail(&mbti->mbti_list, &wmt_mbti.mbti_list);
+ }
+ atomic_inc(&mbti->count);
+ MB_DBG("mb driver is opened by task(%p) %s TGID %x count %d.\n",
+ current, current->comm, current->tgid,
+ atomic_read(&(mbti->count)));
+ filp->private_data = (void *)mbti;
+ spin_unlock_irqrestore(&mb_task_lock, flags);
+
+ return 0;
+}
+
+static int mb_release(struct inode *inode, struct file *filp)
+{
+ struct mb_area_struct *mba;
+ struct mb_struct *mb;
+ struct mb_user *mbu;
+ struct mb_task_info *cmbti = NULL, *mbti = NULL;
+ unsigned long flags, tflags;
+ int task_exist = 0;
+
+ cmbti = (struct mb_task_info *)(filp->private_data);
+ if (!cmbti) {
+ MB_ERROR("none empty private data.\n");
+ return -EFAULT;
+ }
+
+ spin_lock_irqsave(&mb_task_lock, tflags);
+ list_loop(mbti, &wmt_mbti.mbti_list, mbti_list) {
+ if (mbti->task && mbti->tgid == cmbti->tgid) {
+ atomic_dec(&mbti->count);
+ task_exist = 1;
+ break;
+ }
+ }
+ if (!task_exist) {
+ spin_unlock_irqrestore(&mb_task_lock, tflags);
+ MB_INFO("mb driver is closed by unknown task %s TGID %x.\n",
+ current->comm, current->tgid);
+ return 0;
+ }
+
+ MB_DBG("mb driver is closed by task %s TGID %x cnt %d.\n",
+ mbti->task_name, mbti->tgid, atomic_read(&mbti->count));
+ if (atomic_read(&mbti->count)) {
+ spin_unlock_irqrestore(&mb_task_lock, tflags);
+ return 0;
+ }
+
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+RESCAN_MBA:
+ /* munmap virtual memroy and retrieve unused MB */
+ list_loop(mba, &wmt_mbah->mba_list, mba_list) {
+#ifdef CONFIG_WMT_MB_DYNAMIC_ALLOCATE_SUPPORT
+ if (mba->tgid == mbti->tgid) { /* free prefetched mba marker */
+ wmt_mbah->tot_static_pages -= mba->pages;
+ mba->flags &= ~(MBAFLAG_STATIC);
+ mba->tgid = MB_DEF_TGID;
+ if (!mba->nr_mb) {
+ mb_free_mba(mba);
+ goto RESCAN_MBA;
+ }
+ }
+#endif
+ list_loop(mb, &mba->mb_list, mb_list) {
+RESCAN_MBU:
+ list_loop(mbu, &mb->mbu_list, mbu_list) {
+ if (mbu->tgid == mbti->tgid) {
+ char msg[128], *p = msg;
+ const char *opStr[3] = {
+ "\"ALLOC\"",
+ "\"MAP\"",
+ "\"GET\""};
+ int op = (mbu->size) ? 1 : 2;
+ if (mb->creator == mbu) {
+ op = 0;
+ mb->creator = NULL;
+ }
+ p += sprintf(p, "Stop mb operation %s",
+ opStr[op]);
+ p += sprintf(p, "(addr %lx # %lu(KB) ",
+ mb->phys, mbu->size/1024);
+ p += sprintf(p, "by %s) (Cnt %d)\n",
+ mbu->the_user, MB_COUNT(mb)-1);
+ MB_INFO("%s", msg);
+ atomic_dec(&mb->count);
+ list_del_init(&mbu->mbu_list);
+ kmem_cache_free(
+ wmt_mbah->mbu_cachep, mbu);
+ if (MB_COUNT(mb))
+ goto RESCAN_MBU;
+ MB_INFO("Recycle mb addr %lx # %lu\n",
+ mb->phys, mb->size);
+ mb_free_mb(mb);
+ goto RESCAN_MBA;
+ }
+ }
+ }
+ }
+
+ mb_update_mbah();
+ list_del(&mbti->mbti_list);
+ kmem_cache_free(wmt_mbah->mbti_cachep, mbti);
+ filp->private_data = NULL;
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ spin_unlock_irqrestore(&mb_task_lock, tflags);
+
+ return 0;
+}
+
+static long mb_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
+{
+ struct mb_task_info *mbti = NULL;
+ struct mb_area_struct *mba = NULL;
+ struct mb_struct *mb = NULL;
+ struct mb_user *mbu = NULL;
+ unsigned long value, flags, phys, virt, size;
+ pid_t tgid = MB_DEF_TGID;
+ int count, ret = 0, cmdSize;
+ size_t len;
+
+ /* check type and number, if fail return ENOTTY */
+#ifdef CONFIG_WMT_MB_RESERVE_FROM_IO
+ if (_IOC_TYPE(cmd) != MB_IOC_MAGIC && cmd != FBIOGET_FSCREENINFO)
+ return -ENOTTY;
+#else
+ if (_IOC_TYPE(cmd) != MB_IOC_MAGIC)
+ return -ENOTTY;
+#endif
+
+ /* check argument area */
+ cmdSize = _IOC_SIZE(cmd);
+ if (_IOC_DIR(cmd) & _IOC_READ)
+ ret = !access_ok(VERIFY_WRITE, (void __user *) arg, cmdSize);
+ else if (_IOC_DIR(cmd) & _IOC_WRITE)
+ ret = !access_ok(VERIFY_READ, (void __user *) arg, cmdSize);
+
+ if (ret)
+ return -EFAULT;
+
+ /* check task validation */
+ if (!filp || !filp->private_data) {
+ MB_ERROR("mmap with null file info or task.\n");
+ return -EFAULT;
+ }
+
+ mbti = (struct mb_task_info *)(filp->private_data);
+ tgid = mbti->tgid;
+ if (tgid != current->tgid) {
+ MB_WARN("ioctl within diff task.\n");
+ MB_WARN("Open: task %s TGID %x VS Curr: task %s TGID %x\n",
+ mbti->task_name, tgid, current->comm, current->tgid);
+ }
+
+ switch (cmd) {
+ /* _IOR (MB_IOC_MAGIC, 16, unsigned long) O: mb driver version */
+ case MBIO_GET_VERSION:
+ value = MB_VERSION_MAJOR << 24 |
+ MB_VERSION_MINOR << 16 |
+ MB_VERSION_MICRO << 8 |
+ MB_VERSION_BUILD;
+ put_user(value, (unsigned long *)arg);
+ break;
+
+ /* _IOWR(MB_IOC_MAGIC, 0, unsigned long)
+ * I: size
+ * O: physical address */
+ case MBIO_MALLOC:
+ get_user(value, (unsigned long *)arg);
+ phys = mb_do_alloc(value, tgid, mbti->task_name);
+ if (!phys) {
+ MB_WARN("MALLOC: size %ld fail.\n\n", value);
+ return -EFAULT;
+ }
+ MB_OPDBG("MALLOC: addr %lx size %ld\n\n", phys, value);
+ put_user(phys, (unsigned long *)arg);
+ break;
+
+ /* _IOW (MB_IOC_MAGIC, 17, unsigned long) O: property of MB */
+ case MBIO_SET_CACHABLE:
+ get_user(value, (unsigned long *)arg);
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mb = mb_find_mb((void *)value, MBFIND_PHYS);
+ if (!mb) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("SET_CACHABLE: phys %8lx (name %s) fail\n",
+ value, mbti->task_name);
+ return -EFAULT;
+ }
+ mb->flags |= MBFLAG_CACHED;
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_OPDBG("SET_CACHABLE: phys %8lx\n\n", value);
+ break;
+
+ /* _IOW (MB_IOC_MAGIC,18, unsigned long) O: property of MB */
+ case MBIO_CLR_CACHABLE:
+ get_user(value, (unsigned long *)arg);
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mb = mb_find_mb((void *)value, MBFIND_PHYS);
+ if (!mb) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("CLR_CACHABLE: phys %8lx (name %s) fail\n",
+ value, mbti->task_name);
+ return -EFAULT;
+ }
+ mb->flags &= ~MBFLAG_CACHED;
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_OPDBG("CLR_CACHABLE: phys %8lx\n\n", value);
+ break;
+
+ /* _IOWR(MB_IOC_MAGIC, 1, unsigned long)
+ * I: user address if map, physical address if not map
+ * O: ummap size */
+ case MBIO_FREE: /* if not unmap, unmap first */
+ get_user(value, (unsigned long *)arg);
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ size = value;
+ mbu = mb_find_mbu((void *)value, tgid,
+ MBUFIND_USER|MBUFIND_MMAP);
+ if (mbu != NULL) {
+ size = mbu->size;
+ value = mbu->mb->phys;
+ list_del(&mbu->mbu_list);
+ atomic_dec(&mbu->mb->count);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ } else
+ MB_DBG("FREE: unknown MMAP addr %lx\n", value);
+ ret = mb_do_free(value, tgid, mbti->task_name);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ if (ret < 0)
+ MB_WARN("FREE: addr %lx fail\n", value);
+ /* return mmap size for ummap */
+ put_user(size, (unsigned long *)arg);
+ MB_OPDBG("FREE: addr %lx out %ld\n\n", value, size);
+ break;
+
+ /* _IOWR(MB_IOC_MAGIC, 2, unsigned long)
+ * I: user address
+ * O: ummap size */
+ case MBIO_UNMAP:
+ get_user(value, (unsigned long *)arg);
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mbu = mb_find_mbu((void *)value, tgid,
+ MBUFIND_USER|MBUFIND_MMAP);
+ if (mbu == NULL) {
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_WARN("UNMAP: addr %lx fail\n\n", value);
+ return -EFAULT;
+ }
+ mb = mbu->mb;
+ size = mbu->size;
+ list_del(&mbu->mbu_list);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ atomic_dec(&mb->count);
+ count = MB_COUNT(mb);
+ if (count <= 0)
+ MB_WARN("UNMAP: MB %8lx count %d weird\n\n",
+ mb->phys, count);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ /* return mmap size for ummap */
+ put_user(size, (unsigned long *)arg);
+ return 0;
+
+ /* _IOWR(MB_IOC_MAGIC, 3, unsigned long)
+ * I: phys address
+ * O: mb size */
+ case MBIO_MBSIZE:
+ get_user(value, (unsigned long *)arg);
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mb = mb_find_mb((void *)value, MBFIND_PHYS);
+ if (mb == NULL) {
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_WARN("MBSIZE: user %lx fail.\n\n", value);
+ return -EFAULT;
+ }
+ size = mb->size;
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_OPDBG("MBSIZE: addr %lx size %ld\n\n", value, size);
+ put_user(size, (unsigned long *)arg);
+ break;
+
+ /* _IOR (MB_IOC_MAGIC, 4, unsigned long)
+ * O: max free mba size */
+ case MBIO_MAX_AVAILABLE_SIZE:
+ put_user(wmt_mbah->max_available_pages*PAGE_SIZE,
+ (unsigned long *)arg);
+ MB_OPDBG("MAX_MB_SIZE: size %ld KB\n\n",
+ PAGE_KB(wmt_mbah->max_available_pages));
+ break;
+
+ /* _IOW (MB_IOC_MAGIC, 5, unsigned long)
+ * I: user address */
+ case MBIO_GET:
+ get_user(value, (unsigned long *)arg);
+ ret = mb_do_get(value, tgid, mbti->task_name);
+ if (MBMSG_LEVEL) {
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mbu = mb_find_mbu((void *)value, tgid,
+ MBUFIND_USER|MBUFIND_GETPUT);
+ count = mbu ? MB_COUNT(mbu->mb) : 0xffffffff;
+ MB_OPDBG("GET: addr %lx cnt %x\n\n", value, count);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ }
+ break;
+
+ /* _IOW (MB_IOC_MAGIC, 6, unsigned long)
+ * I: user address */
+ case MBIO_PUT:
+ get_user(value, (unsigned long *)arg);
+ if (MBMSG_LEVEL) {
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mbu = mb_find_mbu((void *)value, tgid,
+ MBUFIND_USER|MBUFIND_GETPUT);
+ count = mbu ? MB_COUNT(mbu->mb) - 1 : 0xffffffff;
+ MB_OPDBG("PUT: addr %lx cnt %x\n\n", value, count);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ }
+ ret = mb_do_put(value, tgid, mbti->task_name);
+ break;
+
+ /* _IOWR(MB_IOC_MAGIC, 7, unsigned long)
+ * I: user address
+ * O: virt address */
+ case MBIO_USER_TO_VIRT:
+ get_user(value, (unsigned long *)arg);
+ virt = mb_do_user_to_virt(value, tgid, mbti->task_name);
+ put_user(virt, (unsigned long *)arg);
+ MB_OPDBG("USER_TO_VERT: user %8lx virt %lx\n\n", value, virt);
+ break;
+
+ /* _IOWR(MB_IOC_MAGIC, 8, unsigned long)
+ * I: user address
+ * O: phys address */
+ case MBIO_USER_TO_PHYS:
+ get_user(value, (unsigned long *)arg);
+ phys = mb_do_user_to_phys(value, tgid, mbti->task_name);
+ put_user(phys, (unsigned long *)arg);
+ MB_OPDBG("USER_TO_PHYS: user %8lx phys %lx\n\n", value, phys);
+ break;
+
+#ifdef CONFIG_WMT_MB_RESERVE_FROM_IO
+ /* This IOCTRL is provide AP to recognize MB region as framebuffer,
+ then can access mb memory directly. */
+ case MBIO_FSCREENINFO:
+ {
+ struct fb_fix_screeninfo ffs;
+
+ if (!wmt_mbah || list_empty(&wmt_mbah->mba_list)) {
+ MB_WARN("MBIO_FSCREENINFO: MB driver does not init.\n");
+ return -EFAULT;
+ }
+
+ mba = (struct mb_area_struct *)wmt_mbah->mba_list.next;
+ ffs.smem_start = mba->phys;
+ ffs.smem_len = wmt_mbah->tot_static_pages * PAGE_SIZE;
+ if (copy_to_user((void __user *)arg, &ffs, sizeof(ffs)))
+ ret = -EFAULT;
+ MB_OPDBG("FSCREENINFO: phys %lx len %x\n\n",
+ ffs.smem_start, ffs.smem_len);
+ break;
+ }
+#endif
+
+ /* _IOW (MB_IOC_MAGIC, 9, unsigned long)
+ * I: size */
+ case MBIO_PREFETCH:
+ {
+ unsigned long fetch_size = 0;
+ get_user(value, (unsigned long *)arg);
+#ifdef CONFIG_WMT_MB_DYNAMIC_ALLOCATE_SUPPORT
+{
+ unsigned long try_size = 0;
+
+ MB_OPDBG("PREFETCH: size %ld KB\n\n", value/1024);
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ while (try_size < value) {
+ int order;
+ order = get_order(value-fetch_size);
+ if (order > MBMAX_ORDER)
+ order = MBMAX_ORDER;
+ try_size += (1 << order) * PAGE_SIZE;
+ mba = mb_alloc_mba(1 << order);
+ if (mba) {
+ mba->flags |= MBAFLAG_STATIC;
+ mba->tgid = current->tgid;
+ wmt_mbah->tot_static_pages += mba->pages;
+ fetch_size += mba->pages * PAGE_SIZE;
+ }
+ }
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_INFO("PREFETCH: SIZE %ld / %ld kB by %s\n",
+ fetch_size/1024, value/1024, current->comm);
+}
+#else
+ MB_WARN("PREFETCH: SIZE %ld / %ld kB by %s fail\n",
+ fetch_size/1024, value/1024, current->comm);
+ MB_WARN("PREFETCH: Dynamic Allocated Not Support\n");
+ ret = -ENOTTY;
+#endif
+ break;
+ }
+
+ /* _IOW (MB_IOC_MAGIC, 10, unsigned long)
+ * O: static mba size */
+ case MBIO_STATIC_SIZE:
+ put_user(wmt_mbah->tot_static_pages * PAGE_SIZE,
+ (unsigned long *)arg);
+ MB_OPDBG("STATIC_SIZE: size %ld KB\n\n",
+ PAGE_KB(wmt_mbah->tot_static_pages));
+ break;
+
+ /* _IOWR(MB_IOC_MAGIC,11, unsigned long)
+ * I: physical address
+ * O: use counter */
+ case MBIO_MB_USER_COUNT:
+ get_user(value, (unsigned long *)arg);
+
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mb = mb_find_mb((void *)value, MBFIND_PHYS);
+ if (mb == NULL)
+ size = 0;
+ else
+ size = MB_COUNT(mb);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+
+ put_user(size, (unsigned long *)arg);
+ MB_OPDBG("USER_COUNT: addr %lx count %ld\n\n", value, size);
+ break;
+
+ /* _IO (MB_IOC_MAGIC,12) */
+ case MBIO_FORCE_RESET:
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+RESCAN_MBA:
+ list_loop(mba, &wmt_mbah->mba_list, mba_list) {
+ list_loop(mb, &mba->mb_list, mb_list) {
+ list_loop(
+ mbu, &mb->mbu_list, mbu_list) {
+ list_del(&mbu->mbu_list);
+ kmem_cache_free(
+ wmt_mbah->mbu_cachep, mbu);
+ }
+ mb->creator = NULL;
+ atomic_set(&mb->count, 0);
+ mb_free_mb(mb);
+ /* because mba link maybe breaken */
+ goto RESCAN_MBA;
+ }
+ }
+ mb_update_mbah();
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_OPDBG("RESET: OK\n\n");
+ break;
+
+ /* _IOW (MB_IOC_MAGIC,13, unsigned long) // I: phys address */
+ case MBIO_GET_BY_PHYS:
+ get_user(value, (unsigned long *)arg);
+ if (!value || !mbti->task_name || tgid == MB_DEF_TGID) {
+ MB_WARN("GET(phys): invalid args %lx/%d/%s\n",
+ value, tgid, mbti->task_name);
+ return -EINVAL;
+ }
+
+ mbu = kmem_cache_alloc(wmt_mbah->mbu_cachep, GFP_ATOMIC);
+ if (!mbu) {
+ MB_WARN("GET(phys): mbu_cachep out of memory\n");
+ return -ENOMEM;
+ }
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mb = mb_find_mb((void *)value, MBFIND_PHYS);
+ if (!mb) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ MB_WARN("GET(phys): unknown phys %8lx name %s\n",
+ value, mbti->task_name);
+ return -EFAULT;
+ }
+
+ memset(mbu, 0x0, sizeof(struct mb_user));
+ INIT_LIST_HEAD(&mbu->mbu_list);
+ mbu->addr = value;
+ mbu->mb = mb;
+ mbu->tgid = tgid;
+ len = strlen(mbti->task_name);
+ if (len > TASK_COMM_LEN)
+ len = TASK_COMM_LEN;
+ strncpy(mbu->the_user, mbti->task_name, len);
+ mbu->the_user[len] = 0;
+ atomic_inc(&mb->count);
+ list_add_tail(&mbu->mbu_list, &mb->mbu_list);
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_OPDBG("GET(phys): phys %8lx cnt %x\n\n",
+ value, MB_COUNT(mb));
+ break;
+
+ /* _IOW (MB_IOC_MAGIC,14, unsigned long) // I: phys address */
+ case MBIO_PUT_BY_PHYS:
+ get_user(value, (unsigned long *)arg);
+ if (!value || !mbti->task_name || tgid == MB_DEF_TGID) {
+ MB_WARN("PUT(phys): invalid args %lx/%d/%s\n",
+ value, tgid, mbti->task_name);
+ return -EINVAL;
+ }
+
+ spin_lock_irqsave(&mb_do_lock, flags);
+ mbu = mb_find_mbu((void *)value, tgid,
+ MBUFIND_PHYS|MBUFIND_GETPUT);
+ if (!mbu) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("PUT(phys): unknow phys %8lx name %s\n",
+ value, mbti->task_name);
+ return -EINVAL;
+ }
+
+ mb = mbu->mb;
+ if (!MB_IN_USE(mb)) {
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ MB_WARN("PUT(phys): phys %8lx unbalance.\n", value);
+ return -EPERM;
+ }
+
+ list_del_init(&mbu->mbu_list);
+ atomic_dec(&mb->count);
+ size = MB_COUNT(mb);
+ /* retrieve unused block */
+ if (!size)
+ mb_free_mb(mb);
+
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+
+ MB_OPDBG("PUT(phys): phys %8lx cnt %lx\n\n", value, size);
+
+ break;
+
+ /* _IOW (MB_IOC_MAGIC,15, unsigned long) // I: user address */
+ case MBIO_SYNC_CACHE:
+ get_user(value, (unsigned long *)arg);
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mbu = mb_find_mbu((void *)value, tgid,
+ MBUFIND_USER|MBUFIND_MMAP);
+ if (mbu == NULL) {
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_WARN("SYNC_CACHE: invalid addr %lx\n\n", value);
+ return -EFAULT;
+ }
+ mb = mbu->mb;
+ size = mbu->size;
+ dmac_flush_range(mb->virt, mb->virt + size);
+ outer_flush_range(mb->phys, mb->phys + size);
+ MB_OPDBG("SYNC_CACHE: (%lx # %lx) V %p ~ %p P %lx ~ %lx\n\n",
+ value, size, mb->virt, mb->virt + size,
+ mb->phys, mb->phys + size);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ return 0;
+
+ default:
+ ret = -ENOTTY;
+ break;
+ }
+
+ return ret;
+}
+
+static int mb_mmap(struct file *filp, struct vm_area_struct *vma)
+{
+ struct mb_user *mbu = NULL;
+ struct mb_task_info *mbti;
+ struct mb_struct *mb;
+ unsigned long off, flags;
+ unsigned int len;
+ pid_t tgid;
+ size_t ns;
+
+ if (!filp || !filp->private_data) {
+ MB_ERROR("mmap with null file info or task.\n");
+ return -EFAULT;
+ }
+
+ mbti = (struct mb_task_info *)(filp->private_data);
+ tgid = mbti->tgid;
+ if (tgid != current->tgid) {
+ MB_WARN("mmap within diff task.\n");
+ MB_WARN("Open: task %s TGID %x VS Curr: task %s TGID %x\n",
+ mbti->task_name, tgid, current->comm, current->tgid);
+ }
+
+ off = vma->vm_pgoff << PAGE_SHIFT;
+ MB_DBG("IN, offset %lx TGID %x name %s\n", off, tgid, mbti->task_name);
+#ifdef CONFIG_WMT_MB_RESERVE_FROM_IO
+ /* if page offset under MB total size, it means page offset
+ * is not physical address from MB_alloc and then map function
+ * works like framebuffer.
+ * Suppose MB total size will not cross MBAH phys start address */
+ if (vma->vm_pgoff <= wmt_mbah->tot_static_pages) {
+ struct mb_area_struct *mba = NULL;
+ unsigned long mbtotal =
+ wmt_mbah->tot_static_pages << PAGE_SHIFT;
+ if ((vma->vm_end - vma->vm_start + off) > mbtotal) {
+ MB_ERROR("mmap io fail. range %lx @ [%lx, %lx] s %lx\n",
+ off, vma->vm_start, vma->vm_end, mbtotal);
+ return -EINVAL;
+ }
+ mba = (struct mb_area_struct *)wmt_mbah->mba_list.next;
+ vma->vm_pgoff += mba->phys >> PAGE_SHIFT;
+ vma->vm_flags |= VM_IO | VM_RESERVED;
+ if (io_remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff,
+ vma->vm_end - vma->vm_start, vma->vm_page_prot)) {
+ MB_ERROR("mmap io fail. (io_remap pfn %lx len %lx)\n",
+ vma->vm_pgoff, vma->vm_end - vma->vm_start);
+ return -EAGAIN;
+ }
+ MB_OPDBG("MAP: IO from %lx to %lx/%lx size %ld\n\n",
+ vma->vm_pgoff << PAGE_SHIFT,
+ vma->vm_start, vma->vm_end,
+ vma->vm_end - vma->vm_start);
+ return 0;
+ }
+#endif
+
+ mbu = kmem_cache_alloc(wmt_mbah->mbu_cachep, GFP_ATOMIC);
+ if (!mbu) {
+ MB_ERROR("mbu_cachep out of memory.\n");
+ return -ENOMEM;
+ }
+
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ mb = mb_find_mb((void *)off, MBFIND_PHYS);
+ if (!mb) {
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ MB_ERROR("map addr %lx not exist in MB.\n", off);
+ return -EINVAL;
+ }
+
+ len = PAGE_ALIGN(mb->phys + mb->size);
+ if ((vma->vm_end - vma->vm_start + off) > len) {
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ return -EINVAL;
+ }
+
+ memset(mbu, 0x0, sizeof(struct mb_user));
+ INIT_LIST_HEAD(&mbu->mbu_list);
+ mbu->tgid = tgid;
+ mbu->mb = mb;
+ mbu->addr = vma->vm_start;
+ mbu->size = vma->vm_end - vma->vm_start;
+ ns = strlen(mbti->task_name);
+ if (ns > TASK_COMM_LEN)
+ ns = TASK_COMM_LEN;
+ strncpy(mbu->the_user, mbti->task_name, ns);
+ mbu->the_user[ns] = 0;
+ list_add_tail(&mbu->mbu_list, &mb->mbu_list);
+ atomic_inc(&mb->count);
+
+ off = mb->phys;
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+
+ vma->vm_flags |= VM_IO | VM_RESERVED;
+ if (!(mb->flags & MBFLAG_CACHED))
+ vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
+ if (io_remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff,
+ vma->vm_end - vma->vm_start, vma->vm_page_prot)) {
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ list_del(&mbu->mbu_list);
+ atomic_dec(&mb->count);
+ kmem_cache_free(wmt_mbah->mbu_cachep, mbu);
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ MB_ERROR("virtual address memory map fail.\n");
+ return -EAGAIN;
+ }
+
+ MB_OPDBG("MAP: IO from %lx to %lx/%lx size %ld mbcnt %d\n\n",
+ off, vma->vm_start, vma->vm_end,
+ (vma->vm_end - vma->vm_start), MB_COUNT(mb));
+
+ return 0;
+}
+
+/*!*************************************************************************
+ driver file operations struct define
+****************************************************************************/
+const struct file_operations mb_fops = {
+ .owner = THIS_MODULE,
+ .open = mb_open,
+ .release = mb_release,
+ .unlocked_ioctl = mb_ioctl,
+ .mmap = mb_mmap,
+};
+
+static int mb_probe(struct platform_device *dev)
+{
+ struct mb_area_struct *mba = NULL, *bind = NULL;
+ unsigned long flags;
+ int ret = 0;
+ dev_t dev_no;
+
+ dev_no = MKDEV(mb_dev_major, mb_dev_minor);
+
+ if (wmt_mbah) {
+ MB_ERROR("%s dirty.\n", DEVICE_NAME);
+ return -EINVAL;
+ }
+
+ /* register char device */
+ mb_cdev = cdev_alloc();
+ if (!mb_cdev) {
+ MB_ERROR("alloc dev error.\n");
+ return -ENOMEM;
+ }
+
+ cdev_init(mb_cdev, &mb_fops);
+ ret = cdev_add(mb_cdev, dev_no, 1);
+
+ if (ret) {
+ MB_ERROR("reg char dev error(%d).\n", ret);
+ cdev_del(mb_cdev);
+ return ret;
+ }
+
+ wmt_mbah = kmalloc(sizeof(struct mba_host_struct), GFP_ATOMIC);
+ if (!wmt_mbah) {
+ MB_ERROR("out of memory (mb_area_struct).\n");
+ cdev_del(mb_cdev);
+ return -ENOMEM;
+ }
+ memset(wmt_mbah, 0x0, sizeof(struct mba_host_struct));
+
+ wmt_mbah->mba_cachep = kmem_cache_create("mb_area_struct",
+ sizeof(struct mb_area_struct),
+ 0,
+ SLAB_HWCACHE_ALIGN,
+ NULL);
+ if (!wmt_mbah->mba_cachep) {
+ MB_ERROR("out of memory (mba_cachep).\n");
+ kfree(wmt_mbah);
+ cdev_del(mb_cdev);
+ return -ENOMEM;
+ }
+
+ wmt_mbah->mb_cachep = kmem_cache_create("mb_struct",
+ sizeof(struct mb_struct),
+ 0,
+ SLAB_HWCACHE_ALIGN,
+ NULL);
+ if (!wmt_mbah->mb_cachep) {
+ MB_ERROR("out of memory (mb_cachep).\n");
+ kmem_cache_destroy(wmt_mbah->mba_cachep);
+ kfree(wmt_mbah);
+ cdev_del(mb_cdev);
+ return -ENOMEM;
+ }
+
+ wmt_mbah->mbu_cachep = kmem_cache_create("mb_user",
+ sizeof(struct mb_user),
+ 0,
+ SLAB_HWCACHE_ALIGN,
+ NULL);
+ if (!wmt_mbah->mbu_cachep) {
+ MB_ERROR("out of memory (mbu_cachep).\n");
+ kmem_cache_destroy(wmt_mbah->mb_cachep);
+ kmem_cache_destroy(wmt_mbah->mba_cachep);
+ kfree(wmt_mbah);
+ cdev_del(mb_cdev);
+ return -ENOMEM;
+ }
+
+ wmt_mbah->mbti_cachep = kmem_cache_create("mb_task_info",
+ sizeof(struct mb_task_info),
+ 0,
+ SLAB_HWCACHE_ALIGN,
+ NULL);
+ if (!wmt_mbah->mbti_cachep) {
+ MB_ERROR("out of memory (mbti_cachep).\n");
+ kmem_cache_destroy(wmt_mbah->mbu_cachep);
+ kmem_cache_destroy(wmt_mbah->mb_cachep);
+ kmem_cache_destroy(wmt_mbah->mba_cachep);
+ kfree(wmt_mbah);
+ cdev_del(mb_cdev);
+ return -ENOMEM;
+ }
+
+ MBMAX_ORDER = MBA_MAX_ORDER;
+ MBMIN_ORDER = MBA_MIN_ORDER;
+ INIT_LIST_HEAD(&wmt_mbah->mba_list);
+ INIT_LIST_HEAD(&wmt_mbti.mbti_list);
+
+ spin_lock_init(&mb_do_lock);
+ spin_lock_init(&mb_search_lock);
+ spin_lock_init(&mb_ioctl_lock);
+ spin_lock_init(&mb_task_mm_lock);
+ spin_lock_init(&mb_task_lock);
+
+ MB_INFO("Preparing VIDEO BUFFER (SIZE %d kB) ...\n", MB_TOTAL_SIZE);
+ spin_lock_irqsave(&mb_do_lock, flags);
+ while (PAGE_KB(wmt_mbah->tot_static_pages) < MB_TOTAL_SIZE) {
+ int pages = MB_TOTAL_SIZE*1024 -
+ wmt_mbah->tot_static_pages * PAGE_SIZE;
+ pages = min((pages >> PAGE_SHIFT), (int)(1 << MBMAX_ORDER));
+ mba = mb_alloc_mba(pages);
+ if (mba) {
+ mba->flags |= MBAFLAG_STATIC;
+ mba->tgid = MB_DEF_TGID;
+ wmt_mbah->tot_static_pages += mba->pages;
+ /* conbine to continue mba if possible */
+ if (bind && bind->pgi.pfn_end == mba->pgi.pfn_start) {
+ MB_COMBIND_MBA(bind, mba);
+ continue;
+ }
+ if (bind && bind->pgi.pfn_start == mba->pgi.pfn_end)
+ MB_COMBIND_MBA(mba, bind);
+ bind = mba;
+ }
+ }
+ spin_unlock_irqrestore(&mb_do_lock, flags);
+
+ mb_update_mbah();
+ MB_INFO("MB version: %d.%d.%d.%d\n",
+ MB_VERSION_MAJOR, MB_VERSION_MINOR,
+ MB_VERSION_MICRO, MB_VERSION_BUILD);
+ MB_INFO("MAX MB Area size: Max %ld Kb Min %ld Kb\n",
+ PAGE_KB(1 << MBMAX_ORDER), PAGE_KB(1 << MBMIN_ORDER));
+ MB_INFO("prob /dev/%s major %d, minor %d\n",
+ DEVICE_NAME, mb_dev_major, mb_dev_minor);
+
+ return ret;
+}
+
+static int mb_remove(struct platform_device *dev)
+{
+ if (mb_cdev)
+ cdev_del(mb_cdev);
+
+ if (wmt_mbah) {
+ if (wmt_mbah->mba_cachep)
+ kmem_cache_destroy(wmt_mbah->mba_cachep);
+ if (wmt_mbah->mb_cachep)
+ kmem_cache_destroy(wmt_mbah->mb_cachep);
+ if (wmt_mbah->mbu_cachep)
+ kmem_cache_destroy(wmt_mbah->mbu_cachep);
+ if (wmt_mbah->mbti_cachep)
+ kmem_cache_destroy(wmt_mbah->mbti_cachep);
+ kfree(wmt_mbah);
+ }
+
+ MB_INFO("MB dev remove\n");
+ return 0;
+}
+
+static int mb_suspend(struct platform_device *dev, pm_message_t state)
+{
+ MB_DBG("MB get suspend, Event %d.\n", state.event);
+ return 0;
+}
+
+static int mb_resume(struct platform_device *dev)
+{
+ MB_DBG("MB get resume.\n");
+ return 0;
+}
+
+int mb_area_read_proc(
+ char *buf, char **start, off_t off,
+ int count, int *eof, void *data
+)
+{
+ struct mb_area_struct *mba;
+ struct free_area *fa;
+ struct zone *zone;
+ unsigned long flags;
+ unsigned int idx = 1;
+ char *p = buf, *base = (char *)data;
+ int datalen = 0, len;
+
+ if (!data || !count) {
+ p += sprintf(p, "no resource for mb_area read proc. (%p,%d)\n",
+ data, count);
+ return p-buf;
+ }
+
+ if (!wmt_mbah) {
+ p += sprintf(p, "no MB area host existed.\n");
+ return p-buf;
+ }
+
+ spin_lock_irqsave(&mb_ioctl_lock, flags);
+ if (!off) { /* re read mb area information */
+ p = base;
+ memset(p, 0x0, MBPROC_BUFSIZE);
+
+ /* show block information */
+ p += sprintf(p, "VERSION: %3d.%3d.%3d.%3d\n",
+ MB_VERSION_MAJOR, MB_VERSION_MINOR,
+ MB_VERSION_MICRO, MB_VERSION_BUILD);
+ p += sprintf(p, "MESSAGE LEVEL: %8d /%8lx\n",
+ MBMSG_LEVEL, __pa(&MBMSG_LEVEL));
+ p += sprintf(p, "STATIC MB SIZE: %8ld MB /%5d MB\n",
+ PAGE_KB(wmt_mbah->tot_static_pages)/1024,
+ MB_TOTAL_SIZE/1024);
+ p += sprintf(p, "MAX MBA ORDER: %8d /%8lx\n",
+ MBMAX_ORDER, __pa(&MBMAX_ORDER));
+ p += sprintf(p, "MIN MBA ORDER: %8d /%8lx\n\n",
+ MBMIN_ORDER, __pa(&MBMIN_ORDER));
+ p += sprintf(p, "USER TO PRDT METHOD: %8d /%8lx\n\n",
+ USR2PRDT_METHOD, __pa(&USR2PRDT_METHOD));
+ p += sprintf(p, "total MB areas: %8d\n",
+ wmt_mbah->nr_mba);
+ p += sprintf(p, "total size: %8ld kB\n",
+ PAGE_KB(wmt_mbah->tot_pages));
+ p += sprintf(p, "total free size: %8ld kB\n",
+ PAGE_KB(wmt_mbah->tot_free_pages));
+ p += sprintf(p, "max MB size: %8ld kB\n\n",
+ PAGE_KB(wmt_mbah->max_available_pages));
+
+ list_loop(mba, &wmt_mbah->mba_list, mba_list) {
+ p += sprintf(p, "(ID) [MB Area] VirtAddr PhysAddr ");
+ p += sprintf(p, "size [ zs, ze] MBs Max/");
+ p += sprintf(p, " Free Ste\n");
+ len = (int)(p-base);
+ if ((MBPROC_BUFSIZE - len) < 12) {
+ p += sprintf(p, " more ...\n");
+ break;
+ }
+ p += sprintf(p, "(%2d)", idx++);
+ len = (int)(p-base);
+ /* show all MBs */
+ /* -2 is for \n and zero */
+ p += mb_show_mba(mba, NULL, p,
+ MBPROC_BUFSIZE - len - 2, 1);
+ p += sprintf(p, "\n");
+ }
+ /* show memory fragment */
+ zone = (first_online_pgdat())->node_zones;
+ fa = zone->free_area;
+ p += sprintf(p, "DMA ZONE:\n");
+ for (idx = 0; idx < MAX_ORDER; idx++)
+ p += sprintf(p, " %5ld * %5ldkB = %5ld kB\n",
+ fa[idx].nr_free, PAGE_KB((1<<idx)),
+ fa[idx].nr_free * PAGE_KB((1<<idx)));
+ p += sprintf(p, " ------------------------------\n");
+ p += sprintf(p, " + = %ld kB\n\n",
+ PAGE_KB((unsigned long)nr_free_pages()));
+ }
+ spin_unlock_irqrestore(&mb_ioctl_lock, flags);
+ datalen = strlen(base);
+ if (off >= datalen) {
+ *eof = 1;
+ return 0;
+ }
+ len = min((int)(datalen - off), count);
+ memcpy(buf, &base[off], len);
+ *start = (char *)len; /* for case1: *start < page, mass read data */
+
+ return len;
+}
+
+static struct platform_driver mb_driver = {
+ .driver.name = "wmt_mb",
+ .probe = mb_probe,
+ .remove = mb_remove,
+ .suspend = mb_suspend,
+ .resume = mb_resume
+};
+
+static void mb_platform_release(struct device *device)
+{
+ return;
+}
+
+static struct platform_device mb_device = {
+ .name = "wmt_mb",
+ .id = 0,
+ .dev = { .release = mb_platform_release, },
+ .num_resources = 0,
+ .resource = NULL,
+};
+
+static int __init mb_init(void)
+{
+ int varlen = 32, ret = -1, mbtotal = 0;
+ unsigned char buf[32];
+ dev_t dev_no;
+
+ mb_dev_major = MB_MAJOR;
+ mb_dev_minor = 0;
+ mb_dev_nr = 1;
+
+ dev_no = MKDEV(mb_dev_major, mb_dev_minor);
+ ret = register_chrdev_region(dev_no, mb_dev_nr, "wmt_mb");
+ if (ret < 0) {
+ MB_ERROR("can't get %s device major %d\n",
+ DEVICE_NAME, mb_dev_major);
+ return ret;
+ }
+
+ if (wmt_getsyspara("mbsize", buf, &varlen) == 0) {
+ sscanf(buf, "%dM", &mbtotal);
+ mbtotal *= 1024;
+ MB_INFO("Set MB total size %d KB\n", mbtotal);
+ if (mbtotal > 0)
+ MB_TOTAL_SIZE = mbtotal;
+ }
+
+ wmt_mbah = NULL;
+ MBMSG_LEVEL = MBMAX_ORDER = MBMIN_ORDER = USR2PRDT_METHOD = 0;
+ mb_dev_class = class_create(THIS_MODULE, "mbdev");
+ device_create(mb_dev_class, NULL, dev_no, NULL, "mbdev");
+
+#ifdef CONFIG_PROC_FS
+ {
+ void *proc_buffer = kmalloc(MBPROC_BUFSIZE, GFP_KERNEL);
+ struct proc_dir_entry *res = NULL;
+ if (proc_buffer) {
+ memset(proc_buffer, 0x0, MBPROC_BUFSIZE);
+ res = create_proc_entry("mbinfo", 0, NULL);
+ if (res) {
+ res->read_proc = mb_area_read_proc;
+ res->data = proc_buffer;
+ MB_DBG("create MB proc\n");
+ } else
+ kfree(proc_buffer);
+ }
+ }
+#endif
+
+ ret = platform_driver_register(&mb_driver);
+ if (!ret) {
+ ret = platform_device_register(&mb_device);
+ if (ret)
+ platform_driver_unregister(&mb_driver);
+ }
+
+ return ret;
+}
+
+/* because mb will not exit from system, not need to release proc resource */
+static void __exit mb_exit(void)
+{
+ dev_t dev_no;
+
+ platform_driver_unregister(&mb_driver);
+ platform_device_unregister(&mb_device);
+ dev_no = MKDEV(mb_dev_major, mb_dev_minor);
+ device_destroy(mb_dev_class, dev_no);
+ class_destroy(mb_dev_class);
+ unregister_chrdev_region(dev_no, mb_dev_nr);
+
+ return;
+}
+
+fs_initcall(mb_init);
+module_exit(mb_exit);
+
+MODULE_AUTHOR("WonderMedia Technologies, Inc.");
+MODULE_DESCRIPTION("memory block driver");
+MODULE_LICENSE("GPL");
diff --git a/drivers/video/wmt/wmt-sync.c b/drivers/video/wmt/wmt-sync.c
new file mode 100644
index 00000000..cc4136dc
--- /dev/null
+++ b/drivers/video/wmt/wmt-sync.c
@@ -0,0 +1,346 @@
+/*++
+ * linux/drivers/video/wmt/wmt-sync.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define DEV_SYNC_C
+
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/fs.h>
+#include <linux/kernel.h>
+#include <linux/device.h>
+#include <linux/errno.h>
+#include <asm/uaccess.h>
+#include <linux/kdev_t.h>
+#include <linux/cdev.h>
+#include <linux/dma-mapping.h>
+#include <linux/major.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/spinlock.h>
+#include <mach/irqs.h>
+#include <mach/hardware.h>
+#include <linux/sched.h>
+#include <linux/wmt-mb.h>
+#include <linux/wmt-se.h>
+#include <linux/poll.h>
+
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+#include "vpp.h"
+
+#define DEVICE_NAME "wmt-sync"
+
+static DEFINE_SEMAPHORE(wmt_sync_sem);
+static DECLARE_WAIT_QUEUE_HEAD(wmt_sync_wait);
+static struct class *wmt_sync_class;
+static int wmt_sync_major;
+unsigned int wmt_vsync_flag;
+
+int wmt_sync_set_vsync(void *arg)
+{
+ wmt_vsync_flag++;
+ wake_up_interruptible(&wmt_sync_wait);
+ vpp_mb_irqproc_sync(0);
+ return 0;
+}
+
+static int wmt_sync_open(
+ struct inode *inode,
+ struct file *filp
+)
+{
+ int ret = 0;
+ vpp_int_t type;
+
+ DBG_MSG("\n");
+
+ down(&wmt_sync_sem);
+ if (g_vpp.dual_display) {
+ type = (vout_info[0].govr_mod == VPP_MOD_GOVRH) ?
+ VPP_INT_GOVRH_VBIS : VPP_INT_GOVRH2_VBIS;
+ } else {
+ type = VPP_INT_GOVRH2_VBIS; /* HDMI should slow down */
+ }
+ vpp_irqproc_work(type, wmt_sync_set_vsync, 0, 0, 0);
+ up(&wmt_sync_sem);
+ return ret;
+} /* End of wmt_sync_open() */
+
+static int wmt_sync_release(
+ struct inode *inode,
+ struct file *filp
+)
+{
+ int ret = 0;
+ vpp_int_t type;
+
+ DBG_MSG("\n");
+
+ down(&wmt_sync_sem);
+ type = (vout_info[0].govr_mod == VPP_MOD_GOVRH) ?
+ VPP_INT_GOVRH_VBIS : VPP_INT_GOVRH2_VBIS;
+ vpp_irqproc_del_work(type, wmt_sync_set_vsync);
+ up(&wmt_sync_sem);
+ return ret;
+} /* End of wmt_sync_release() */
+
+static long wmt_sync_ioctl(struct file *filp, unsigned int cmd,
+ unsigned long arg)
+{
+ int ret = -EINVAL;
+
+ DBG_MSG("0x%x,0x%x\n", cmd, arg);
+
+ /* check type and number, if fail return ENOTTY */
+ if (_IOC_TYPE(cmd) != WMT_CEC_IOC_MAGIC)
+ return -ENOTTY;
+ if (_IOC_NR(cmd) > WMT_CEC_IOC_MAXNR)
+ return -ENOTTY;
+
+ /* check argument area */
+ if (_IOC_DIR(cmd) & _IOC_READ)
+ ret = !access_ok(VERIFY_WRITE,
+ (void __user *) arg, _IOC_SIZE(cmd));
+ else if (_IOC_DIR(cmd) & _IOC_WRITE)
+ ret = !access_ok(VERIFY_READ,
+ (void __user *) arg, _IOC_SIZE(cmd));
+ else
+ ret = 0;
+
+ if (ret)
+ return -EFAULT;
+
+ down(&wmt_sync_sem);
+ switch (cmd) {
+ default:
+ DBG_ERR("*W* cmd 0x%x\n", cmd);
+ break;
+ }
+ up(&wmt_sync_sem);
+ return ret;
+} /* End of wmt_sync_ioctl() */
+
+static ssize_t wmt_sync_read(
+ struct file *filp,
+ char __user *buf,
+ size_t count,
+ loff_t *f_pos
+)
+{
+ unsigned long data;
+ ssize_t retval = 0;
+
+ down(&wmt_sync_sem);
+ if (filp->f_flags & O_NONBLOCK && !wmt_vsync_flag) {
+ retval = -EAGAIN;
+ goto read_end;
+ }
+ data = xchg(&wmt_vsync_flag, 0);
+ retval = wait_event_interruptible(wmt_sync_wait, data);
+ if (retval == 0)
+ retval = put_user(data, (unsigned int __user *)buf);
+read_end:
+ up(&wmt_sync_sem);
+ return retval;
+} /* wmt_sync_read */
+
+static ssize_t wmt_sync_write(
+ struct file *filp,
+ const char __user *buf,
+ size_t count,
+ loff_t *f_pos
+)
+{
+ ssize_t ret = 0;
+ down(&wmt_sync_sem);
+ /* TODO */
+ up(&wmt_sync_sem);
+ return ret;
+} /* End of wmt_sync_write() */
+
+static unsigned int wmt_sync_poll(
+ struct file *filp,
+ struct poll_table_struct *wait
+)
+{
+ unsigned int mask = 0;
+
+ down(&wmt_sync_sem);
+ poll_wait(filp, &wmt_sync_wait, wait);
+ if (wmt_vsync_flag != 0)
+ mask |= POLLIN | POLLRDNORM;
+ up(&wmt_sync_sem);
+ return mask;
+}
+
+const struct file_operations wmt_sync_fops = {
+ .owner = THIS_MODULE,
+ .open = wmt_sync_open,
+ .release = wmt_sync_release,
+ .read = wmt_sync_read,
+ .write = wmt_sync_write,
+ .unlocked_ioctl = wmt_sync_ioctl,
+ .poll = wmt_sync_poll,
+};
+
+static int wmt_sync_probe
+(
+ struct platform_device *dev /*!<; // a pointer to struct device */
+)
+{
+ return 0;
+} /* End of wmt_sync_probe */
+
+static int wmt_sync_remove
+(
+ struct platform_device *dev /*!<; // a pointer to struct device */
+)
+{
+ return 0;
+} /* End of wmt_sync_remove */
+
+#ifdef CONFIG_PM
+static int wmt_sync_suspend
+(
+ struct platform_device *pDev, /*!<; // a pointer to struct device */
+ pm_message_t state /*!<; // suspend state */
+)
+{
+ DPRINT("Enter wmt_sync_suspend\n");
+ return 0;
+} /* End of wmt_sync_suspend */
+
+static int wmt_sync_resume
+(
+ struct platform_device *pDev /*!<; // a pointer to struct device */
+)
+{
+ DPRINT("Enter wmt_sync_resume\n");
+ return 0;
+} /* End of wmt_sync_resume */
+#else
+#define wmt_sync_suspend NULL
+#define wmt_sync_resume NULL
+#endif
+
+/***************************************************************************
+ device driver struct define
+****************************************************************************/
+static struct platform_driver wmt_sync_driver = {
+ .driver.name = DEVICE_NAME, /* equal to platform device name. */
+/* .bus = &platform_bus_type, */
+ .probe = wmt_sync_probe,
+ .remove = wmt_sync_remove,
+ .suspend = wmt_sync_suspend,
+ .resume = wmt_sync_resume,
+};
+
+static void wmt_sync_platform_release(struct device *device)
+{
+} /* End of wmt_sync_platform_release() */
+
+/***************************************************************************
+ platform device struct define
+****************************************************************************/
+/* static u64 wmt_sync_dma_mask = 0xffffffffUL; */
+static struct platform_device wmt_sync_device = {
+ .name = DEVICE_NAME,
+ .id = 0,
+ .dev = {
+ .release = wmt_sync_platform_release,
+#if 0
+ .dma_mask = &wmt_sync_dma_mask,
+ .coherent_dma_mask = ~0,
+#endif
+ },
+ .num_resources = 0, /* ARRAY_SIZE(cipher_resources), */
+ .resource = NULL, /* cipher_resources, */
+};
+
+static int __init wmt_sync_init(void)
+{
+ int ret;
+ char buf[100];
+ int varlen = 100;
+ unsigned int sync_enable = 1; /* default enable */
+ dev_t dev_no;
+
+ if (wmt_getsyspara("wmt.display.sync", buf, &varlen) == 0)
+ vpp_parse_param(buf, &sync_enable, 1, 0);
+
+ if (sync_enable == 0)
+ return 0;
+
+ wmt_sync_major = register_chrdev(0, DEVICE_NAME, &wmt_sync_fops);
+ if (wmt_sync_major < 0) {
+ DBG_ERR("get gpio_dev_major failed\n");
+ return -EFAULT;
+ }
+
+ wmt_sync_class = class_create(THIS_MODULE, DEVICE_NAME);
+ if (IS_ERR(wmt_sync_class)) {
+ ret = PTR_ERR(wmt_sync_class);
+ DBG_ERR("Can't create class : %s !!\n", DEVICE_NAME);
+ return ret;
+ }
+
+ dev_no = MKDEV(wmt_sync_major, 0);
+ device_create(wmt_sync_class, NULL, dev_no, NULL, DEVICE_NAME);
+ ret = platform_device_register(&wmt_sync_device);
+ if (ret != 0)
+ return -ENODEV;
+
+ ret = platform_driver_register(&wmt_sync_driver);
+ if (ret != 0) {
+ platform_device_unregister(&wmt_sync_device);
+ return -ENODEV;
+ }
+ return 0;
+}
+
+static void __exit wmt_sync_exit(void)
+{
+ dev_t dev_no;
+
+ DBG_MSG("\n");
+
+ platform_driver_unregister(&wmt_sync_driver);
+ platform_device_unregister(&wmt_sync_device);
+ dev_no = MKDEV(wmt_sync_major, 0);
+ device_destroy(wmt_sync_class, dev_no);
+ class_destroy(wmt_sync_class);
+ unregister_chrdev(wmt_sync_major, DEVICE_NAME);
+ return;
+}
+
+module_init(wmt_sync_init);
+module_exit(wmt_sync_exit);
+
+MODULE_LICENSE("GPL");
+MODULE_DESCRIPTION("WMT SYNC driver");
+MODULE_AUTHOR("WMT TECH");
+MODULE_VERSION("1.0.0");
+
diff --git a/drivers/video/wmt/wmt-vpp.c b/drivers/video/wmt/wmt-vpp.c
new file mode 100755
index 00000000..4415477b
--- /dev/null
+++ b/drivers/video/wmt/wmt-vpp.c
@@ -0,0 +1,1538 @@
+/*++
+ * linux/drivers/video/wmt/wmt-vpp.c
+ * WonderMedia video post processor (VPP) driver
+ *
+ * Copyright c 2014 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define DEV_VPP_C
+
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/fs.h>
+#include <linux/kernel.h>
+#include <linux/device.h>
+#include <linux/errno.h>
+#include <asm/uaccess.h>
+#include <linux/kdev_t.h>
+#include <linux/cdev.h>
+#include <linux/dma-mapping.h>
+#include <linux/major.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/spinlock.h>
+#include <mach/irqs.h>
+#include <mach/hardware.h>
+#include <linux/sched.h>
+#include <linux/wmt-mb.h>
+#include <linux/wmt-se.h>
+#include <linux/poll.h>
+
+#undef DEBUG
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+#include "vpp.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define WMT_VPP_XXXX 1 *//*Example*/
+#define DEVICE_NAME "wmt-vpp"
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx wmt_vpp_xxx_t; *//*Example*/
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int wmt_vpp_xxx; *//*Example*/
+static DEFINE_SEMAPHORE(wmt_vpp_sem);
+static struct class *wmt_vpp_class;
+static int wmt_vpp_major;
+
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void wmt_vpp_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+#ifdef CONFIG_PROC_FS
+#define CONFIG_VPP_PROC
+#ifdef CONFIG_VPP_PROC
+unsigned int vpp_proc_value;
+char vpp_proc_str[16];
+static ctl_table vpp_table[];
+static int vpp_do_proc(ctl_table *ctl, int write,
+ void *buffer, size_t *len, loff_t *ppos)
+{
+ int ret;
+ int ctl_name;
+
+ ctl_name = (((int)ctl - (int)vpp_table) / sizeof(ctl_table)) + 1;
+ if (!write) {
+ switch (ctl_name) {
+ case 1:
+ vpp_proc_value = g_vpp.dbg_msg_level;
+ break;
+ case 2:
+ vpp_proc_value = g_vpp.dbg_wait;
+ break;
+ case 3:
+ vpp_proc_value = g_vpp.dbg_flag;
+ break;
+ case 8:
+ case 9:
+ vpp_proc_value = vpp_get_base_clock((ctl_name == 8) ?
+ VPP_MOD_GOVRH : VPP_MOD_GOVRH2);
+ break;
+ case 10:
+ vpp_proc_value = p_scl->scale_mode;
+ break;
+ case 11:
+ vpp_proc_value = p_scl->filter_mode;
+ break;
+#ifdef CONFIG_WMT_HDMI
+ case 12:
+ vpp_proc_value = g_vpp.hdmi_cp_enable;
+ break;
+ case 13:
+ vpp_proc_value = g_vpp.hdmi_3d_type;
+ break;
+ case 14:
+ vpp_proc_value = g_vpp.hdmi_certify_flag;
+ break;
+#endif
+ case 15:
+ vpp_proc_value = govrh_get_brightness(p_govrh);
+ break;
+ case 16:
+ vpp_proc_value = govrh_get_contrast(p_govrh);
+ break;
+ case 17:
+ vpp_proc_value = govrh_get_saturation(p_govrh);
+ break;
+ case 18:
+ vpp_proc_value = g_vpp.fb_manual;
+ break;
+ default:
+ break;
+ }
+ }
+
+ ret = proc_dointvec(ctl, write, buffer, len, ppos);
+ if (write) {
+ switch (ctl_name) {
+ case 1:
+ DPRINT("---------- VPP debug level ----------\n");
+ DPRINT("0-disable,255-show all\n");
+ DPRINT("1-scale,2-disp fb,3-interrupt,4-timer\n");
+ DPRINT("5-ioctl,6-diag,7-stream\n");
+ DPRINT("-------------------------------------\n");
+ g_vpp.dbg_msg_level = vpp_proc_value;
+ break;
+ case 2:
+ g_vpp.dbg_wait = vpp_proc_value;
+ vpp_dbg_wake_up();
+ break;
+ case 3:
+ g_vpp.dbg_flag = vpp_proc_value;
+ break;
+#ifdef CONFIG_WMT_EDID
+ case 6:
+ {
+ struct vout_t *vo;
+
+ vo = vout_get_entry_adapter(vpp_proc_value);
+ if ((vo->inf) && (vo->inf->get_edid)) {
+ vo->status &= ~VPP_VOUT_STS_EDID;
+ if (vout_get_edid(vo->num)) {
+ int i;
+
+ vo->edid_info.option = 0;
+ edid_dump(vo->edid);
+ for (i = 1; i <= vo->edid[126]; i++)
+ edid_dump(vo->edid + 128 * i);
+ if (!edid_parse(vo->edid,
+ &vo->edid_info))
+ DBG_ERR("parse EDID\n");
+ } else {
+ DBG_ERR("read EDID\n");
+ }
+ }
+ }
+ break;
+#endif
+ case 8:
+ case 9:
+ govrh_set_clock((ctl_name == 8) ? p_govrh : p_govrh2,
+ vpp_proc_value);
+ DPRINT("set govr pixclk %d\n", vpp_proc_value);
+ break;
+ case 10:
+ DPRINT("---------- scale mode ----------\n");
+ DPRINT("0-recursive normal\n");
+ DPRINT("1-recursive sw bilinear\n");
+ DPRINT("2-recursive hw bilinear\n");
+ DPRINT("3-realtime noraml (quality but x/32 limit)\n");
+ DPRINT("4-realtime bilinear (fit edge w skip line)\n");
+ DPRINT("-------------------------------------\n");
+ p_scl->scale_mode = vpp_proc_value;
+ break;
+ case 11:
+ p_scl->filter_mode = vpp_proc_value;
+ break;
+#ifdef CONFIG_WMT_HDMI
+ case 12:
+ g_vpp.hdmi_cp_enable = vpp_proc_value;
+ hdmi_set_cp_enable(vpp_proc_value);
+ break;
+ case 13:
+ g_vpp.hdmi_3d_type = vpp_proc_value;
+ hdmi_tx_vendor_specific_infoframe_packet();
+ break;
+ case 14:
+ g_vpp.hdmi_certify_flag = vpp_proc_value;
+ break;
+#endif
+ case 15:
+ govrh_set_brightness(p_govrh, vpp_proc_value);
+ break;
+ case 16:
+ govrh_set_contrast(p_govrh, vpp_proc_value);
+ break;
+ case 17:
+ govrh_set_saturation(p_govrh, vpp_proc_value);
+ break;
+ case 18:
+ g_vpp.fb_manual = vpp_proc_value;
+ break;
+ default:
+ break;
+ }
+ }
+ return ret;
+}
+
+struct proc_dir_entry *vpp_proc_dir;
+static ctl_table vpp_table[] = {
+ { /* .ctl_name = 1, */
+ .procname = "dbg_msg",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 2, */
+ .procname = "dbg_wait",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 3, */
+ .procname = "dbg_flag",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 4, */
+ .procname = "edid_disable",
+ .data = &edid_disable,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 5, */
+ .procname = "edid_msg",
+ .data = &edid_msg_enable,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 6, */
+ .procname = "vout_edid",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 7, */
+ .procname = "vo_mode",
+ .data = vpp_proc_str,
+ .maxlen = 12,
+ .mode = 0666,
+ .proc_handler = &proc_dostring,
+ },
+ { /* .ctl_name = 8, */
+ .procname = "govr1_pixclk",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 9, */
+ .procname = "govr2_pixclk",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 10, */
+ .procname = "scl_scale_mode",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 11, */
+ .procname = "scl_filter",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 12 */
+ .procname = "hdmi_cp_enable",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 13 */
+ .procname = "hdmi_3d",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 14 */
+ .procname = "hdmi_certify",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 15 */
+ .procname = "brightness",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 16 */
+ .procname = "contrast",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 17 */
+ .procname = "saturation",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 18 */
+ .procname = "fb_manual",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* end of table */
+ }
+};
+
+static ctl_table vpp_root_table[] = {
+ {
+ .procname = "vpp", /* create path ==> /proc/sys/vpp */
+ .mode = 0555,
+ .child = vpp_table
+ },
+ { /* end of table */
+ }
+};
+static struct ctl_table_header *vpp_table_header;
+#endif
+
+static int vpp_sts_read_proc(char *buf, char **start, off_t offset,
+ int len, int *eof, void *data)
+{
+ struct govrh_regs *regs = (struct govrh_regs *) REG_GOVRH_BASE1_BEGIN;
+ unsigned int yaddr, caddr;
+ char *p = buf;
+ unsigned int reg;
+
+ p += sprintf(p, "--- VPP HW status ---\n");
+#ifdef WMT_FTBLK_GOVRH
+ p += sprintf(p, "GOVRH memory read underrun error %d,cnt %d,cnt2 %d\n",
+ (regs->interrupt.val & 0x200) ? 1 : 0,
+ p_govrh->underrun_cnt, p_govrh2->underrun_cnt);
+ p_govrh->clr_sts(VPP_INT_ALL);
+#endif
+
+#ifdef WMT_FTBLK_SCL
+ p += sprintf(p, "---------------------------------------\n");
+ p += sprintf(p, "SCL TG error %d\n", scl_regs1->tg_sts.b.tgerr);
+ p += sprintf(p, "SCLR MIF1 read error %d\n",
+ scl_regs1->r_fifo_ctl.b.r1_mif_err);
+ p += sprintf(p, "SCLR MIF2 read error %d\n",
+ scl_regs1->r_fifo_ctl.b.r2_mif_err);
+ p += sprintf(p, "SCLW RGB fifo overflow %d\n",
+ scl_regs1->w_ff_ctl.b.mif_rgb_err);
+ p += sprintf(p, "SCLW Y fifo overflow %d\n",
+ scl_regs1->w_ff_ctl.b.mif_y_err);
+ p += sprintf(p, "SCLW C fifo overflow %d\n",
+ scl_regs1->w_ff_ctl.b.mif_c_err);
+ p_scl->clr_sts(VPP_INT_ALL);
+#endif
+
+ p += sprintf(p, "---------------------------------------\n");
+ p += sprintf(p, "(880.0)GOVRH Enable %d,(900.0)TG %d\n",
+ regs->mif.b.enable, regs->tg_enable.b.enable);
+
+ reg = inl(PM_CTRL_BASE_ADDR + 0x258);
+ p += sprintf(p, "--- POWER CONTROL ---\n");
+ p += sprintf(p, "0x%x = 0x%x\n", PM_CTRL_BASE_ADDR + 0x258, reg);
+ p += sprintf(p, "HDCP %d,VPP %d,SCL %d,HDMI I2C %d\n",
+ (reg & BIT7) ? 1 : 0, (reg & BIT18) ? 1 : 0,
+ (reg & BIT21) ? 1 : 0, (reg & BIT22) ? 1 : 0);
+ p += sprintf(p, "HDMI %d,GOVR %d,NA12 %d\n",
+ (reg & BIT23) ? 1 : 0, (reg & BIT25) ? 1 : 0,
+ (reg & BIT16) ? 1 : 0);
+ p += sprintf(p, "DVO %d,HDMI OUT %d,LVDS %d\n", (reg & BIT29) ? 1 : 0,
+ (reg & BIT30) ? 1 : 0, (reg & BIT14) ? 1 : 0);
+
+ p += sprintf(p, "--- VPP fb Address ---\n");
+
+#ifdef WMT_FTBLK_GOVRH
+ govrh_get_fb_addr(p_govrh, &yaddr, &caddr);
+ p += sprintf(p, "GOVRH fb addr Y(0x%x) 0x%x, C(0x%x) 0x%x\n",
+ REG_GOVRH_YSA, yaddr, REG_GOVRH_CSA, caddr);
+ govrh_get_fb_addr(p_govrh2, &yaddr, &caddr);
+ p += sprintf(p, "GOVRH2 fb addr Y(0x%x) 0x%x, C(0x%x) 0x%x\n",
+ REG_GOVRH2_YSA, yaddr, REG_GOVRH2_CSA, caddr);
+#endif
+ p_govrh->underrun_cnt = 0;
+ p_govrh2->underrun_cnt = 0;
+ return p - buf;
+}
+
+static int vpp_reg_read_proc(char *buf, char **start, off_t offset,
+ int len, int *eof, void *data)
+{
+ char *p = buf;
+ struct vpp_mod_base_t *mod_p;
+ int i;
+
+ DPRINT("Product ID:0x%x\n", vpp_get_chipid());
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->dump_reg)
+ mod_p->dump_reg();
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_reg_dump();
+#endif
+#ifdef WMT_FTBLK_LVDS
+ lvds_reg_dump();
+#endif
+ return p - buf;
+}
+#endif
+
+irqreturn_t vpp_interrupt_routine(int irq, void *dev_id)
+{
+ enum vpp_int_t int_sts;
+
+ switch (irq) {
+ case VPP_IRQ_VPPM: /* VPP */
+ int_sts = p_vppm->get_sts();
+ p_vppm->clr_sts(int_sts);
+
+ vpp_dbg_show_val1(VPP_DBGLVL_INT, 0, "[VPP] VPPM INT", int_sts);
+ {
+ int i;
+ unsigned int mask;
+ struct vpp_irqproc_t *irqproc;
+
+ for (i = 0, mask = 0x1; (i < 32) && int_sts; i++, mask <<= 1) {
+ if ((int_sts & mask) == 0)
+ continue;
+
+ irqproc = vpp_irqproc_get_entry(mask);
+ if (irqproc) {
+ if (list_empty(&irqproc->list) == 0)
+ tasklet_schedule(&irqproc->tasklet);
+ } else {
+ irqproc = vpp_irqproc_get_entry(VPP_INT_MAX);
+ if (list_empty(&irqproc->list) == 0) {
+ struct vpp_proc_t *entry;
+ struct list_head *ptr;
+
+ ptr = (&irqproc->list)->next;
+ entry = list_entry(ptr,
+ struct vpp_proc_t, list);
+ if (entry->type == mask)
+ tasklet_schedule(
+ &irqproc->tasklet);
+ }
+ }
+ int_sts &= ~mask;
+ }
+ }
+ break;
+#ifdef WMT_FTBLK_SCL
+ case VPP_IRQ_SCL: /* SCL */
+ int_sts = p_scl->get_sts();
+ p_scl->clr_sts(int_sts);
+ vpp_dbg_show_val1(VPP_DBGLVL_INT, 0, "[VPP] SCL INT", int_sts);
+ break;
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ case VPP_IRQ_GOVR: /* GOVR */
+ case VPP_IRQ_GOVR2:
+ {
+ struct govrh_mod_t *govr;
+
+ govr = (irq == VPP_IRQ_GOVR) ? p_govrh : p_govrh2;
+ int_sts = govr->get_sts();
+ govr->clr_sts(int_sts);
+ vpp_dbg_show_val1(VPP_DBGLVL_INT, 0, "[VPP] GOVR INT", int_sts);
+ govr->underrun_cnt++;
+#ifdef VPP_DBG_DIAG_NUM
+ vpp_dbg_show(VPP_DBGLVL_DIAG, 3, "GOVR MIF Err");
+ vpp_dbg_diag_delay = 10;
+#endif
+ }
+ break;
+#endif
+ default:
+ DPRINT("*E* invalid vpp isr\n");
+ break;
+ }
+ return IRQ_HANDLED;
+}
+
+void vpp_wait_vsync(int no, int cnt)
+{
+ struct govrh_mod_t *govr;
+
+ govr = vout_info_get_govr(no);
+ if (govr) {
+ if (govrh_get_MIF_enable(govr)) {
+ vpp_irqproc_work((govr->mod == VPP_MOD_GOVRH) ?
+ VPP_INT_GOVRH_VBIS : VPP_INT_GOVRH2_VBIS,
+ 0, 0, 100 * cnt, cnt);
+ if (vpp_check_dbg_level(VPP_DBGLVL_DISPFB))
+ MSG("wait vsync(%d,%d)\n", no, cnt);
+ }
+ }
+}
+
+/*----------------------- vpp ioctl --------------------------------------*/
+int vpp_common_ioctl(unsigned int cmd, unsigned long arg)
+{
+ struct vpp_mod_base_t *mod_p;
+ struct vpp_fb_base_t *mod_fb_p;
+ int retval = 0;
+
+ switch (cmd) {
+ case VPPIO_VPPGET_INFO:
+ {
+ int i;
+ vpp_cap_t parm;
+
+ parm.chip_id = vpp_get_chipid();
+ parm.version = 0x01;
+ parm.resx_max = VPP_HD_MAX_RESX;
+ parm.resy_max = VPP_HD_MAX_RESY;
+ parm.pixel_clk = 400000000;
+ parm.module = 0x0;
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p)
+ parm.module |= (0x01 << i);
+ }
+ parm.option = VPP_CAP_DUAL_DISPLAY;
+ copy_to_user((void *)arg, (void *) &parm, sizeof(vpp_cap_t));
+ }
+ break;
+ case VPPIO_VPPSET_INFO:
+ {
+ vpp_cap_t parm;
+
+ copy_from_user((void *)&parm, (const void *)arg,
+ sizeof(vpp_cap_t));
+ }
+ break;
+ case VPPIO_I2CSET_BYTE:
+ {
+ vpp_i2c_t parm;
+ unsigned int id;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_i2c_t));
+ id = (parm.addr & 0x0000FF00) >> 8;
+ vpp_i2c_write(id, (parm.addr & 0xFF), parm.index,
+ (char *)&parm.val, 1);
+ }
+ break;
+ case VPPIO_I2CGET_BYTE:
+ {
+ vpp_i2c_t parm;
+ unsigned int id;
+ int len;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_i2c_t));
+ id = (parm.addr & 0x0000FF00) >> 8;
+ len = parm.val;
+ {
+ unsigned char buf[len];
+
+ vpp_i2c_read(id, (parm.addr & 0xFF), parm.index,
+ buf, len);
+ parm.val = buf[0];
+ }
+ copy_to_user((void *)arg, (void *) &parm, sizeof(vpp_i2c_t));
+ }
+ break;
+ case VPPIO_MODULE_FBINFO:
+ {
+ vpp_mod_fbinfo_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_fbinfo_t));
+
+ if (g_vpp.virtual_display)
+ parm.mod = (hdmi_get_plugin()) ?
+ VPP_MOD_GOVRH : VPP_MOD_GOVRH2;
+ mod_fb_p = vpp_mod_get_fb_base(parm.mod);
+ if (!mod_fb_p)
+ break;
+ mod_p = vpp_mod_get_base(parm.mod);
+ if (parm.read) {
+ parm.fb = mod_fb_p->fb;
+ switch (parm.mod) {
+ case VPP_MOD_GOVRH:
+ case VPP_MOD_GOVRH2:
+ govrh_get_framebuffer(
+ (struct govrh_mod_t *)mod_p,
+ &parm.fb);
+ break;
+ default:
+ break;
+ }
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_mod_fbinfo_t));
+ } else {
+ mod_fb_p->fb = parm.fb;
+ mod_fb_p->set_framebuf(&parm.fb);
+ }
+ }
+ break;
+#ifdef CONFIG_VPP_STREAM_CAPTURE
+ case VPPIO_STREAM_ENABLE:
+ g_vpp.stream_enable = arg;
+ g_vpp.stream_mb_sync_flag = 0;
+ g_vpp.stream_mb_lock = 0;
+ g_vpp.stream_mb_index = 0xFF;
+ MSG("VPPIO_STREAM_ENABLE %d\n", g_vpp.stream_enable);
+ vpp_netlink_notify(WP_PID, DEVICE_STREAM,
+ (g_vpp.stream_enable) ? 1 : 0);
+ wmt_enable_mmfreq(WMT_MMFREQ_MIRACAST, g_vpp.stream_enable);
+ break;
+ case VPPIO_STREAM_GETFB:
+ {
+ vdo_framebuf_t fb;
+
+ vpp_lock();
+ if (g_vpp.stream_mb_index == 0xFF) { /* not avail */
+ retval = -1;
+ vpp_unlock();
+ break;
+ }
+ fb = vout_info[g_vpp.stream_fb]->fb;
+ fb.fb_h = vpp_calc_align(fb.img_h, 16);
+ copy_to_user((void *)arg, (void *) &fb, sizeof(vdo_framebuf_t));
+ retval = vpp_mb_get(fb.y_addr);
+ vpp_unlock();
+ }
+ break;
+ case VPPIO_STREAM_PUTFB:
+ {
+ vdo_framebuf_t fb;
+ copy_from_user((void *) &fb, (const void *)arg,
+ sizeof(vdo_framebuf_t));
+ vpp_lock();
+ vpp_mb_put(fb.y_addr);
+ vpp_unlock();
+ }
+ break;
+#endif
+ case VPPIO_MULTIVD_ENABLE:
+ wmt_enable_mmfreq(WMT_MMFREQ_MULTI_VD, arg);
+ break;
+ case VPPIO_MODSET_CSC:
+ {
+ struct vpp_mod_parm_t parm;
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(struct vpp_mod_parm_t));
+ switch (parm.mod) {
+ case VPP_MOD_GOVRH:
+ p_govrh->csc_mode_force = parm.parm;
+ break;
+ case VPP_MOD_GOVRH2:
+ if (p_govrh2->csc_mode_force != parm.parm)
+ MSG("GOVRH2 CSC %d-->%d\n",
+ p_govrh2->csc_mode_force, parm.parm);
+ p_govrh2->csc_mode_force = parm.parm;
+ break;
+ default:
+ break;
+ }
+ }
+ break;
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+}
+
+int vpp_vout_ioctl(unsigned int cmd, unsigned long arg)
+{
+ int retval = 0;
+
+ switch (cmd) {
+ case VPPIO_VOGET_INFO:
+ {
+ vpp_vout_info_t parm;
+ struct vout_info_t *info;
+ int fb_no, i, idx;
+ unsigned int status;
+
+ memset(&parm, 0, sizeof(vpp_vout_info_t));
+ for (fb_no = 0, idx = 0; ; fb_no++) {
+ info = vout_info_get_entry(fb_no);
+ if (!info)
+ break;
+
+ for (i = 0; i < VPP_VOUT_NUM; i++, idx++) {
+ if (info->vout[i] == 0)
+ break;
+ parm.mode[idx] = vout_get_mode_adapter(
+ info->vout[i]);
+ status = info->vout[i]->status |
+ (fb_no << 8);
+ DBGMSG("GET_INFO(fb %d,mode %d,sts 0x%x)",
+ fb_no, parm.mode[idx], status);
+ DBGMSG("plug %d\n",
+ (status & VPP_VOUT_STS_PLUGIN) ? 1 : 0);
+ parm.status[idx] = status;
+ }
+ }
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_info_t));
+ }
+ break;
+ case VPPIO_VOUT_VMODE:
+ {
+ vpp_vout_vmode_t parm;
+ int i;
+ struct fb_videomode *vmode;
+ unsigned int resx, resy, fps;
+ unsigned int pre_resx, pre_resy, pre_fps;
+ int index, from_index;
+ int support;
+ unsigned int option, pre_option;
+#ifdef CONFIG_WMT_EDID
+ struct edid_info_t *edid_info;
+#endif
+ struct fb_videomode *edid_vmode;
+ vpp_vout_t mode;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_vmode_t));
+ from_index = parm.num;
+ parm.num = 0;
+ mode = parm.mode & 0xF;
+#ifdef CONFIG_VPP_DEMO
+ parm.parm[parm.num].resx = 1920;
+ parm.parm[parm.num].resy = 1080;
+ parm.parm[parm.num].fps = 60;
+ parm.parm[parm.num].option = 0;
+ parm.num++;
+#else
+#ifdef CONFIG_WMT_EDID
+ {
+ struct vout_t *vo;
+
+ vo = vout_get_entry_adapter(mode);
+ if (!vo)
+ goto vout_vmode_end;
+
+ if (!(vo->status & VPP_VOUT_STS_PLUGIN)) {
+ DPRINT("*W* not plugin\n");
+ goto vout_vmode_end;
+ }
+
+ if (vout_get_edid(vo->num) == 0) {
+ DPRINT("*W* read EDID fail\n");
+ goto vout_vmode_end;
+ }
+ if (edid_parse(vo->edid, &vo->edid_info) == 0) {
+ DPRINT("*W* parse EDID fail\n");
+ goto vout_vmode_end;
+ }
+ edid_info = &vo->edid_info;
+ }
+#endif
+ index = 0;
+ resx = resy = fps = option = 0;
+ pre_resx = pre_resy = pre_fps = pre_option = 0;
+ for (i = 0; ; i++) {
+ int ret = 0;
+ vmode = (struct fb_videomode *) &vpp_videomode[i];
+ if (vmode->pixclock == 0)
+ break;
+ resx = vmode->xres;
+ resy = vmode->yres;
+ fps = vmode->refresh;
+ option = fps & EDID_TMR_FREQ;
+ option |= (vmode->vmode & FB_VMODE_INTERLACED) ?
+ EDID_TMR_INTERLACE : 0;
+ if ((pre_resx == resx) && (pre_resy == resy) &&
+ (pre_fps == fps) && (pre_option == option))
+ continue;
+ pre_resx = resx;
+ pre_resy = resy;
+ pre_fps = fps;
+ pre_option = option;
+ support = 0;
+#ifdef CONFIG_WMT_EDID
+ if (ret = edid_find_support(edid_info, resx, resy,
+ option, &edid_vmode))
+#else
+ if (1)
+#endif
+ support = 1;
+
+ if (support) {
+ if (index >= from_index) {
+ parm.parm[parm.num].resx = resx;
+ parm.parm[parm.num].resy = resy;
+ parm.parm[parm.num].fps = fps;
+ parm.parm[parm.num].option =
+ vmode->vmode;
+#ifdef CONFIG_WMT_EDID
+ if(ret == 3)
+ parm.parm[parm.num].option |= 0x80;
+#endif
+ parm.num++;
+ }
+ index++;
+ if (parm.num >= VPP_VOUT_VMODE_NUM)
+ break;
+ }
+ }
+#ifdef CONFIG_WMT_EDID
+vout_vmode_end:
+#endif
+#endif
+#if 0
+ if (parm.num == 0) { /* if no EDID*/
+ enum vout_tvformat_t tvformat = vout_get_tvformat();
+ if (g_vpp.virtual_display ||
+ (g_vpp.dual_display == 0)) {
+ if (mode == VPP_VOUT_DVI) {
+ switch (tvformat) {
+ case TV_PAL:
+ parm.parm[0].resx = 720;
+ parm.parm[0].resy = 576;
+ parm.parm[0].fps = 50;
+ parm.num = 1;
+ parm.parm[0].option = 0;
+ break;
+ case TV_NTSC:
+ parm.parm[0].resx = 720;
+ parm.parm[0].resy = 480;
+ parm.parm[0].fps = 60;
+ parm.num = 1;
+ parm.parm[0].option = 0;
+ break;
+ default:
+ break;
+ }
+ }
+ }
+ }
+#endif
+ DBG_MSG("[VPP] get support vmode %d\n", parm.num);
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_vmode_t));
+ }
+ break;
+ case VPPIO_VOGET_EDID:
+ {
+ vpp_vout_edid_t parm;
+ char *edid;
+ struct vout_t *vo;
+ int size;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_edid_t));
+ size = 0;
+#ifdef CONFIG_WMT_EDID
+ vo = vout_get_entry_adapter(parm.mode);
+ if (!vo)
+ goto vout_edid_end;
+
+ if (!(vo->status & VPP_VOUT_STS_PLUGIN)) {
+ DBG_ERR("*W* not plugin\n");
+ goto vout_edid_end;
+ }
+
+ edid = vout_get_edid(vo->num);
+ if (edid == 0) {
+ DBG_ERR("*W* read EDID fail\n");
+ goto vout_edid_end;
+ }
+ size = (edid[0x7E] + 1) * 128;
+ if (size > parm.size)
+ size = parm.size;
+ copy_to_user((void *) parm.buf, (void *) edid, size);
+vout_edid_end:
+#endif
+ parm.size = size;
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_edid_t));
+ }
+ break;
+ case VPPIO_VOGET_CP_INFO:
+ {
+ vpp_vout_cp_info_t parm;
+ int num;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_cp_info_t));
+ num = parm.num;
+ if (num >= VOUT_MODE_MAX) {
+ retval = -ENOTTY;
+ break;
+ }
+ memset(&parm, 0, sizeof(vpp_vout_cp_info_t));
+ switch (num) {
+ case VOUT_HDMI:
+ if (!g_vpp.hdmi_certify_flag) {
+ if (g_vpp.hdmi_bksv[0] == 0) {
+ hdmi_get_bksv(&g_vpp.hdmi_bksv[0]);
+ DBG_MSG("get BKSV 0x%x 0x%x\n",
+ g_vpp.hdmi_bksv[0],
+ g_vpp.hdmi_bksv[1]);
+ }
+ }
+ case VOUT_DVO2HDMI:
+ parm.bksv[0] = g_vpp.hdmi_bksv[0];
+ parm.bksv[1] = g_vpp.hdmi_bksv[1];
+ break;
+ default:
+ parm.bksv[0] = parm.bksv[1] = 0;
+ break;
+ }
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_cp_info_t));
+ }
+ break;
+ case VPPIO_VOSET_CP_KEY:
+ if (g_vpp.hdmi_cp_p == 0) {
+ g_vpp.hdmi_cp_p = kmalloc(sizeof(vpp_vout_cp_key_t),
+ GFP_KERNEL);
+ }
+ if (g_vpp.hdmi_cp_p) {
+ copy_from_user((void *) g_vpp.hdmi_cp_p,
+ (const void *)arg, sizeof(vpp_vout_cp_key_t));
+ if (hdmi_cp)
+ hdmi_cp->init();
+ }
+ break;
+#ifdef WMT_FTBLK_HDMI
+ case VPPIO_VOSET_AUDIO_PASSTHRU:
+ hdmi_regs1->aud_mode.b.sub_packet = (arg) ? 0xF : 0x0;
+ break;
+#endif
+#ifdef CONFIG_VPP_VIRTUAL_DISPLAY
+ case VPPIO_VOSET_VIRTUAL_FBDEV:
+ {
+ struct vout_info_t *info;
+ int i;
+
+ g_vpp.fb0_bitblit = (arg) ? 0 : 1;
+ for (i = 1; ; i++) {
+ info = vout_info_get_entry(i);
+ if (!info)
+ break;
+ if (info->alloc_mode != VOUT_ALLOC_GE_OVERSCAN)
+ continue;
+ if (info->mb == 0)
+ continue;
+ MSG("fb%d mb free 0x%x\n", i, info->mb);
+ mb_free(info->mb);
+ info->mb = 0;
+ }
+ MSG("[VPP] virtual display %d\n", (int)arg);
+ }
+ break;
+#endif
+ case VPPIO_VOGET_HDMI_3D:
+ {
+ struct vpp_vout_parm_s parm;
+ struct vout_t *vo;
+
+ vo = vout_get_entry(VPP_VOUT_NUM_HDMI);
+ parm.arg = edid_get_hdmi_3d_mask(&vo->edid_info, hdmi_info.vic);
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(struct vpp_vout_parm_s));
+ }
+ break;
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+}
+
+int vpp_scl_ioctl(unsigned int cmd, unsigned long arg)
+{
+ int retval = 0;
+
+ switch (cmd) {
+ case VPPIO_SCL_SCALE_OVERLAP:
+ {
+ vpp_scale_overlap_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_scale_overlap_t));
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->overlap_timer, 0, 1);
+
+ p_scl->scale_sync = 1;
+ retval = scl_set_scale_overlap(&parm.src_fb,
+ &parm.src2_fb, &parm.dst_fb);
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->overlap_timer, "overlap", 2);
+ }
+ break;
+ case VPPIO_SCL_SCALE_ASYNC:
+ case VPPIO_SCL_SCALE:
+ {
+ vpp_scale_t parm;
+
+ p_scl->scale_sync = (cmd == VPPIO_SCL_SCALE) ? 1 : 0;
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_scale_t));
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->scale_timer, 0, 1);
+
+ vpp_set_NA12_hiprio(1);
+ retval = vpp_set_recursive_scale(&parm.src_fb, &parm.dst_fb);
+ vpp_set_NA12_hiprio(0);
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->scale_timer, "scale", 2);
+ copy_to_user((void *)arg, (void *)&parm, sizeof(vpp_scale_t));
+ }
+ break;
+ case VPPIO_SCL_SCALE_FINISH:
+ retval = p_scl->scale_finish();
+ break;
+ case VPPIO_SCLSET_OVERLAP:
+ {
+ vpp_overlap_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_overlap_t));
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_ENABLE, 0);
+ scl_set_overlap(&parm);
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_DISABLE, 0);
+ }
+ break;
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+}
+
+static int wmt_vpp_open(struct inode *inode, struct file *filp)
+{
+ DBG_MSG("\n");
+ return 0;
+}
+
+static int wmt_vpp_release(struct inode *inode, struct file *filp)
+{
+ DBG_MSG("\n");
+ return 0;
+}
+
+static long wmt_vpp_ioctl(struct file *filp, unsigned int cmd,
+ unsigned long arg)
+{
+ int ret = -EINVAL;
+ int skip_mutex = 0;
+
+/* DBG_MSG("0x%x,0x%x\n", cmd, (int)arg); */
+
+ if (_IOC_TYPE(cmd) != VPPIO_MAGIC)
+ return -ENOTTY;
+ if (_IOC_NR(cmd) > VPPIO_MAX)
+ return -ENOTTY;
+
+ /* check argument area */
+ if (_IOC_DIR(cmd) & _IOC_READ)
+ ret = !access_ok(VERIFY_WRITE,
+ (void __user *) arg, _IOC_SIZE(cmd));
+ else if (_IOC_DIR(cmd) & _IOC_WRITE)
+ ret = !access_ok(VERIFY_READ,
+ (void __user *) arg, _IOC_SIZE(cmd));
+ else
+ ret = 0;
+
+ if (ret)
+ return -EFAULT;
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_IOCTL)) {
+ switch (cmd) {
+ default:
+ DPRINT("[VPP] ioctl cmd 0x%x,arg 0x%x\n",
+ _IOC_NR(cmd), (int)arg);
+ break;
+ }
+ }
+
+ switch (cmd) {
+ case VPPIO_STREAM_GETFB: /* block mode should wait vsync */
+ skip_mutex = 1;
+ break;
+ default:
+ /* scale should wait complete */
+ if ((_IOC_NR(cmd) >= VPPIO_SCL_BASE) &&
+ (_IOC_NR(cmd) < VPPIO_MAX))
+ skip_mutex = 1;
+ break;
+ }
+
+ if (!skip_mutex)
+ down(&wmt_vpp_sem);
+
+ switch (_IOC_NR(cmd)) {
+ case VPPIO_VPP_BASE ... (VPPIO_VOUT_BASE-1):
+ /* DBGMSG("VPP command ioctl\n"); */
+ ret = vpp_common_ioctl(cmd, arg);
+ break;
+ case VPPIO_VOUT_BASE ... (VPPIO_SCL_BASE - 1):
+ ret = vpp_vout_ioctl(cmd, arg);
+ break;
+ case VPPIO_SCL_BASE ... (VPPIO_MAX-1):
+ /* DBGMSG("SCL ioctl\n"); */
+ ret = vpp_scl_ioctl(cmd, arg);
+ break;
+ default:
+ DBG_ERR("*W* cmd 0x%x\n", cmd);
+ break;
+ }
+
+ if (!skip_mutex)
+ up(&wmt_vpp_sem);
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_IOCTL)) {
+ switch (cmd) {
+ default:
+ DPRINT("[VPP] ioctl cmd 0x%x,ret 0x%x\n",
+ _IOC_NR(cmd), (int)ret);
+ break;
+ }
+ }
+ return ret;
+}
+
+const struct file_operations wmt_vpp_fops = {
+ .owner = THIS_MODULE,
+ .open = wmt_vpp_open,
+ .release = wmt_vpp_release,
+ .unlocked_ioctl = wmt_vpp_ioctl,
+};
+
+static int wmt_vpp_probe(struct platform_device *dev)
+{
+ DBG_MSG("\n");
+
+ vpp_irqproc_init();
+ vpp_netlink_init();
+ vpp_init();
+#ifdef CONFIG_VPP_PROC
+ /* init system proc */
+ if (vpp_proc_dir == 0) {
+ struct proc_dir_entry *res;
+
+ vpp_proc_dir = proc_mkdir("driver/vpp", NULL);
+ res = create_proc_entry("sts", 0, vpp_proc_dir);
+ if (res)
+ res->read_proc = vpp_sts_read_proc;
+ res = create_proc_entry("reg", 0, vpp_proc_dir);
+ if (res)
+ res->read_proc = vpp_reg_read_proc;
+ vpp_table_header = register_sysctl_table(vpp_root_table);
+ }
+#endif
+
+ /* init interrupt service routine */
+#ifdef WMT_FTBLK_SCL
+ if (vpp_request_irq(VPP_IRQ_SCL, vpp_interrupt_routine,
+ SA_INTERRUPT, "scl", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+#endif
+ if (vpp_request_irq(VPP_IRQ_VPPM, vpp_interrupt_routine,
+ SA_INTERRUPT, "vpp", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+#ifdef WMT_FTBLK_GOVRH
+ if (vpp_request_irq(VPP_IRQ_GOVR, vpp_interrupt_routine,
+ SA_INTERRUPT, "govr", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+
+ if (vpp_request_irq(VPP_IRQ_GOVR2, vpp_interrupt_routine,
+ SA_INTERRUPT, "govr2", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+#endif
+ vpp_switch_state_init();
+ return 0;
+}
+
+static int wmt_vpp_remove(struct platform_device *dev)
+{
+ return 0;
+}
+
+#ifdef CONFIG_PM
+unsigned int wmt_vpp_vout_blank_mask;
+static int wmt_vpp_suspend(struct device *dev)
+{
+
+ struct vpp_mod_base_t *mod_p;
+ struct vout_t *vo;
+ int i;
+
+ DPRINT("Enter wmt_vpp_suspend\n");
+
+ wmt_vpp_vout_blank_mask = 0;
+ for (i = 0; i <= VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo && !(vo->status & VPP_VOUT_STS_BLANK))
+ wmt_vpp_vout_blank_mask |= (0x1 << i);
+ vout_set_blank(i, VOUT_BLANK_POWERDOWN);
+ if (vo && vo->dev && vo->dev->suspend)
+ vo->dev->suspend();
+ }
+
+ if (vout_check_plugin(1))
+ vpp_netlink_notify_plug(VPP_VOUT_ALL, 0);
+ else
+ wmt_set_mmfreq(0);
+
+ /* disable module */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->suspend)
+ mod_p->suspend(0);
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_suspend(0);
+#endif
+ wmt_suspend_mmfreq();
+#ifdef WMT_FTBLK_LVDS
+ lvds_suspend(0);
+#endif
+ /* disable tg */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->suspend)
+ mod_p->suspend(1);
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_suspend(1);
+#endif
+#ifdef WMT_FTBLK_LVDS
+ lvds_suspend(1);
+#endif
+ /* backup registers */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->suspend)
+ mod_p->suspend(2);
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_suspend(2);
+#endif
+#ifdef WMT_FTBLK_LVDS
+ lvds_suspend(2);
+#endif
+#if 0
+ if (lcd_get_lvds_id() == LCD_LVDS_1024x600) {
+ mdelay(5);
+ /* GPIO10 off 8ms -> clock -> off */
+ outl(inl(GPIO_BASE_ADDR + 0xC0) & ~BIT10,
+ GPIO_BASE_ADDR + 0xC0);
+ }
+#endif
+ return 0;
+}
+
+static int wmt_vpp_freeze(struct device *dev)
+{
+ return 0;
+}
+
+#ifdef CONFIG_VPP_SHENZHEN
+extern int lcd_spi_resume(void);
+#endif
+
+static int wmt_vpp_resume(struct device *dev)
+{
+
+ struct vpp_mod_base_t *mod_p;
+ int i;
+
+#if 0
+ if (lcd_get_lvds_id() == LCD_LVDS_1024x600) {
+ /* GPIO10 6ms -> clock r0.02.04 */
+ outl(inl(GPIO_BASE_ADDR + 0x80) | BIT10,
+ GPIO_BASE_ADDR + 0x80);
+ outl(inl(GPIO_BASE_ADDR + 0xC0) | BIT10,
+ GPIO_BASE_ADDR + 0xC0);
+ }
+#endif
+
+ DPRINT("Enter wmt_vpp_resume\n");
+
+#ifdef CONFIG_VPP_SHENZHEN
+ lcd_spi_resume();
+#endif
+
+ /* restore registers */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->resume)
+ mod_p->resume(0);
+ }
+#ifdef WMT_FTBLK_LVDS
+ lvds_resume(0);
+#endif
+#ifdef WMT_FTBLK_HDMI
+ hdmi_check_plugin(0);
+ hdmi_resume(0);
+#endif
+ /* enable tg */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->resume)
+ mod_p->resume(1);
+ }
+#ifdef WMT_FTBLK_LVDS
+ lvds_resume(1);
+#endif
+#ifdef WMT_FTBLK_HDMI
+ hdmi_resume(1);
+#endif
+ /* wait */
+ msleep(150);
+
+ /* enable module */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->resume)
+ mod_p->resume(2);
+ }
+#ifdef WMT_FTBLK_LVDS
+ lvds_resume(2);
+#endif
+ if (lcd_get_lvds_id() != LCD_LVDS_1024x600) {
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ struct vout_t *vo;
+
+ vo = vout_get_entry(i);
+ if (vo && vo->dev) {
+ if (lcd_get_type() == 0) /* DVI type */
+ govrh_set_dvo_enable(vo->govr, 1);
+ vo->dev->init(vo);
+ vo->dev->set_mode(&vo->option[0]);
+ vo->inf->init(vo, 0);
+ }
+
+ if (wmt_vpp_vout_blank_mask & (0x1 << i))
+ vout_set_blank(i, VOUT_BLANK_UNBLANK);
+ if (vo && vo->dev && vo->dev->resume)
+ vo->dev->resume();
+ }
+ }
+ wmt_resume_mmfreq();
+ if (vout_check_plugin(0))
+ vpp_netlink_notify_plug(VPP_VOUT_ALL, 1);
+ else
+ wmt_set_mmfreq(0);
+#ifdef WMT_FTBLK_HDMI
+ hdmi_resume(2);
+#endif
+ return 0;
+}
+
+static int wmt_vpp_restore(struct device *dev)
+{
+ return 0;
+}
+
+static void wmt_vpp_shutdown
+(
+ struct platform_device *pDev /*!<; // a pointer struct device */
+)
+{
+ DPRINT("wmt_vpp_shutdown\n");
+ hdmi_set_power_down(1);
+ lvds_set_power_down(1);
+}
+
+#else
+#define wmt_vpp_suspend NULL
+#define wmt_vpp_resume NULL
+#define wmt_vpp_shutdown NULL
+#endif
+
+/********************************************************************
+ device driver struct define
+**********************************************************************/
+static struct dev_pm_ops wmt_vpp_pm_ops = {
+ .suspend = wmt_vpp_suspend,
+ .resume = wmt_vpp_resume,
+ .freeze = wmt_vpp_freeze,
+ .thaw = wmt_vpp_restore,
+ .restore = wmt_vpp_restore,
+};
+
+static struct platform_driver wmt_vpp_driver = {
+ .driver.name = DEVICE_NAME, /* equal to platform device name. */
+/* .bus = &platform_bus_type, */
+ .probe = wmt_vpp_probe,
+ .remove = wmt_vpp_remove,
+ /* .suspend = wmt_vpp_suspend, */
+ /* .resume = wmt_vpp_resume, */
+ .shutdown = wmt_vpp_shutdown,
+ .driver.pm = &wmt_vpp_pm_ops
+};
+
+static void wmt_vpp_platform_release(struct device *device)
+{
+}
+
+/********************************************************************
+ platform device struct define
+*********************************************************************/
+/* static u64 wmt_vpp_dma_mask = 0xffffffffUL; */
+static struct platform_device wmt_vpp_device = {
+ .name = DEVICE_NAME,
+ .id = 0,
+ .dev = {
+ .release = wmt_vpp_platform_release,
+#if 0
+ .dma_mask = &wmt_vpp_dma_mask,
+ .coherent_dma_mask = ~0,
+#endif
+ },
+ .num_resources = 0, /* ARRAY_SIZE(cipher_resources), */
+ .resource = NULL, /* cipher_resources, */
+};
+
+static int __init wmt_vpp_init(void)
+{
+ int ret;
+ dev_t dev_no;
+
+ wmt_vpp_major = register_chrdev(0, DEVICE_NAME, &wmt_vpp_fops);
+ if (wmt_vpp_major < 0) {
+ DBG_ERR("get major failed\n");
+ return -EFAULT;
+ }
+
+ wmt_vpp_class = class_create(THIS_MODULE, DEVICE_NAME);
+ if (IS_ERR(wmt_vpp_class)) {
+ ret = PTR_ERR(wmt_vpp_class);
+ DBG_ERR("Can't create class : %s !!\n", DEVICE_NAME);
+ return ret;
+ }
+
+ dev_no = MKDEV(wmt_vpp_major, 0);
+ device_create(wmt_vpp_class, NULL, dev_no, NULL, DEVICE_NAME);
+ ret = platform_device_register(&wmt_vpp_device);
+ if (ret != 0)
+ return -ENODEV;
+
+ ret = platform_driver_register(&wmt_vpp_driver);
+ if (ret != 0) {
+ platform_device_unregister(&wmt_vpp_device);
+ return -ENODEV;
+ }
+ return 0;
+}
+
+static void __exit wmt_vpp_exit(void)
+{
+ dev_t dev_no;
+
+ DBG_MSG("\n");
+
+ vout_exit();
+#ifdef CONFIG_VPP_PROC
+ unregister_sysctl_table(vpp_table_header);
+#endif
+ platform_driver_unregister(&wmt_vpp_driver);
+ platform_device_unregister(&wmt_vpp_device);
+ dev_no = MKDEV(wmt_vpp_major, 0);
+ device_destroy(wmt_vpp_class, dev_no);
+ class_destroy(wmt_vpp_class);
+ unregister_chrdev(wmt_vpp_major, DEVICE_NAME);
+ return;
+}
+
+module_init(wmt_vpp_init);
+module_exit(wmt_vpp_exit);
+
+MODULE_LICENSE("GPL");
+MODULE_DESCRIPTION("WMT VPP driver");
+MODULE_AUTHOR("WMT TECH");
+MODULE_VERSION("1.0.0");
+
diff --git a/drivers/video/wmt/wmtfb.c b/drivers/video/wmt/wmtfb.c
new file mode 100644
index 00000000..bac0973f
--- /dev/null
+++ b/drivers/video/wmt/wmtfb.c
@@ -0,0 +1,2764 @@
+/*++
+ * linux/drivers/video/wmt/wmtfb.c
+ * WonderMedia frame buffer driver
+ *
+ * Copyright c 2013 WonderMedia Technologies, Inc.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ * WonderMedia Technologies, Inc.
+ * 4F, 533, Chung-Cheng Road, Hsin-Tien, Taipei 231, R.O.C
+--*/
+
+#define WMTFB_C
+/* #define DEBUG */
+/* #define DEBUG_DETAIL */
+
+/*----------------------- DEPENDENCE -----------------------------------------*/
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/types.h>
+#include <linux/fs.h>
+#include <linux/kernel.h>
+#include <linux/platform_device.h>
+#include <linux/errno.h>
+#include <asm/uaccess.h>
+#include <linux/kdev_t.h>
+#include <linux/cdev.h>
+#include <linux/fb.h>
+#include <linux/dma-mapping.h>
+#include <asm/page.h>
+#include <linux/mm.h>
+#include <linux/proc_fs.h>
+
+#include "vpp.h"
+
+/*----------------------- PRIVATE MACRO --------------------------------------*/
+
+/*----------------------- PRIVATE CONSTANTS ----------------------------------*/
+/* #define FBUT_XXXX 1 *//*Example*/
+
+/*----------------------- PRIVATE TYPE --------------------------------------*/
+/* typedef xxxx fbut_xxx_t; *//*Example*/
+
+/*----------------------- INTERNAL PRIVATE VARIABLES - -----------------------*/
+/* int fbut_xxx; *//*Example*/
+
+static struct fb_fix_screeninfo wmtfb_fix = {
+ .id = "wmtfb",
+ .smem_start = 0,
+ .smem_len = 0,
+ .type = FB_TYPE_PACKED_PIXELS,
+ .type_aux = 0,
+ .visual = FB_VISUAL_TRUECOLOR,
+ .xpanstep = 1,
+ .ypanstep = 1,
+ .ywrapstep = 1,
+ .line_length = 0,
+ .mmio_start = 0xD8050000,
+ .mmio_len = 0x0700,
+ .accel = FB_ACCEL_NONE
+};
+
+static struct fb_var_screeninfo wmtfb_var = {
+ .xres = VPP_HD_DISP_RESX,
+ .yres = VPP_HD_DISP_RESY,
+ .xres_virtual = VPP_HD_DISP_RESX,
+ .yres_virtual = VPP_HD_DISP_RESY,
+#if 0
+ .bits_per_pixel = 32,
+ .red = {16, 8, 0},
+ .green = {8, 8, 0},
+ .blue = {0, 8, 0},
+ .transp = {24, 8, 0},
+#else
+ .bits_per_pixel = 16,
+ .red = {11, 5, 0},
+ .green = {5, 6, 0},
+ .blue = {0, 5, 0},
+ .transp = {0, 0, 0},
+#endif
+ .activate = FB_ACTIVATE_NOW | FB_ACTIVATE_FORCE,
+ .height = -1,
+ .width = -1,
+ .pixclock = (VPP_HD_DISP_RESX*VPP_HD_DISP_RESY*VPP_HD_DISP_FPS),
+ .left_margin = 40,
+ .right_margin = 24,
+ .upper_margin = 32,
+ .lower_margin = 11,
+ .hsync_len = 96,
+ .vsync_len = 2,
+ .vmode = FB_VMODE_NONINTERLACED
+};
+
+int wmtfb_fb1_probe;
+/*--------------------- INTERNAL PRIVATE FUNCTIONS ---------------------------*/
+/* void fbut_xxx(void); *//*Example*/
+
+/*----------------------- Function Body --------------------------------------*/
+/*----------------------- Linux Proc --------------------------------------*/
+#ifdef DEBUG
+void wmtfb_show_var(char *str, struct fb_var_screeninfo *var)
+{
+ DPRINT("----- %s ------------------------\n", str);
+ DPRINT("res(%d,%d),vir(%d,%d),offset(%d,%d)\n",
+ var->xres, var->yres, var->xres_virtual,
+ var->yres_virtual, var->xoffset, var->yoffset);
+ DPRINT("pixclk %d(%d),hsync %d,vsync %d\n", var->pixclock,
+ (int)(PICOS2KHZ(var->pixclock) * 1000),
+ var->hsync_len, var->vsync_len);
+ DPRINT("left %d,right %d,upper %d,lower %d\n", var->left_margin,
+ var->right_margin, var->upper_margin, var->lower_margin);
+ DPRINT("bpp %d, grayscale %d\n", var->bits_per_pixel, var->grayscale);
+ DPRINT("nonstd %d, activate %d, height %d, width %d\n",
+ var->nonstd, var->activate, var->height, var->width);
+ DPRINT("vmode 0x%x,sync 0x%x,rotate %d,accel %d\n",
+ var->vmode, var->sync, var->rotate, var->accel_flags);
+ DPRINT("-----------------------------\n");
+ return;
+}
+#endif
+
+#ifdef CONFIG_PROC_FS
+#define CONFIG_VPP_PROC
+#ifdef CONFIG_VPP_PROC
+unsigned int vpp_proc_value;
+char vpp_proc_str[16];
+static ctl_table vpp_table[];
+static int vpp_do_proc(ctl_table *ctl, int write,
+ void *buffer, size_t *len, loff_t *ppos)
+{
+ int ret;
+ int ctl_name;
+
+ ctl_name = (((int)ctl - (int)vpp_table) / sizeof(ctl_table)) + 1;
+ if (!write) {
+ switch (ctl_name) {
+ case 1:
+ vpp_proc_value = g_vpp.dbg_msg_level;
+ break;
+ case 2:
+ vpp_proc_value = g_vpp.dbg_wait;
+ break;
+ case 3:
+ vpp_proc_value = g_vpp.dbg_flag;
+ break;
+ case 8:
+ case 9:
+ vpp_proc_value = vpp_get_base_clock((ctl_name == 8) ?
+ VPP_MOD_GOVRH : VPP_MOD_GOVRH2);
+ break;
+ case 10:
+ vpp_proc_value = p_scl->scale_mode;
+ break;
+ case 11:
+ vpp_proc_value = p_scl->filter_mode;
+ break;
+#ifdef CONFIG_WMT_HDMI
+ case 12:
+ vpp_proc_value = g_vpp.hdmi_cp_enable;
+ break;
+ case 13:
+ vpp_proc_value = g_vpp.hdmi_3d_type;
+ break;
+ case 14:
+ vpp_proc_value = g_vpp.hdmi_certify_flag;
+ break;
+#endif
+ case 15:
+ vpp_proc_value = govrh_get_brightness(p_govrh);
+ break;
+ case 16:
+ vpp_proc_value = govrh_get_contrast(p_govrh);
+ break;
+ case 17:
+ vpp_proc_value = govrh_get_saturation(p_govrh);
+ break;
+ default:
+ break;
+ }
+ }
+
+ ret = proc_dointvec(ctl, write, buffer, len, ppos);
+ if (write) {
+ switch (ctl_name) {
+ case 1:
+ DPRINT("---------- VPP debug level ----------\n");
+ DPRINT("0-disable,255-show all\n");
+ DPRINT("1-scale,2-disp fb,3-interrupt,4-timer\n");
+ DPRINT("5-ioctl,6-diag,7-stream\n");
+ DPRINT("-------------------------------------\n");
+ g_vpp.dbg_msg_level = vpp_proc_value;
+ break;
+ case 2:
+ g_vpp.dbg_wait = vpp_proc_value;
+ vpp_dbg_wake_up();
+ break;
+ case 3:
+ g_vpp.dbg_flag = vpp_proc_value;
+ break;
+#ifdef CONFIG_WMT_EDID
+ case 6:
+ {
+ vout_t *vo;
+
+ vo = vout_get_entry_adapter(vpp_proc_value);
+ if ((vo->inf) && (vo->inf->get_edid)) {
+ vo->status &= ~VPP_VOUT_STS_EDID;
+ if (vout_get_edid(vo->num)) {
+ int i;
+
+ vo->edid_info.option = 0;
+ edid_dump(vo->edid);
+ for (i = 1; i <= vo->edid[126]; i++)
+ edid_dump(vo->edid + 128 * i);
+ if (!edid_parse(vo->edid,
+ &vo->edid_info))
+ DBG_ERR("parse EDID\n");
+ } else {
+ DBG_ERR("read EDID\n");
+ }
+ }
+ }
+ break;
+#endif
+ case 8:
+ case 9:
+ govrh_set_clock((ctl_name == 8) ? p_govrh : p_govrh2,
+ vpp_proc_value);
+ DPRINT("set govr pixclk %d\n", vpp_proc_value);
+ break;
+ case 10:
+ DPRINT("---------- scale mode ----------\n");
+ DPRINT("0-recursive normal\n");
+ DPRINT("1-recursive sw bilinear\n");
+ DPRINT("2-recursive hw bilinear\n");
+ DPRINT("3-realtime noraml (quality but x/32 limit)\n");
+ DPRINT("4-realtime bilinear (fit edge w skip line)\n");
+ DPRINT("-------------------------------------\n");
+ p_scl->scale_mode = vpp_proc_value;
+ break;
+ case 11:
+ p_scl->filter_mode = vpp_proc_value;
+ break;
+#ifdef CONFIG_WMT_HDMI
+ case 12:
+ g_vpp.hdmi_cp_enable = vpp_proc_value;
+ hdmi_set_cp_enable(vpp_proc_value);
+ break;
+ case 13:
+ g_vpp.hdmi_3d_type = vpp_proc_value;
+ hdmi_tx_vendor_specific_infoframe_packet();
+ break;
+ case 14:
+ g_vpp.hdmi_certify_flag = vpp_proc_value;
+ break;
+#endif
+ case 15:
+ govrh_set_brightness(p_govrh, vpp_proc_value);
+ break;
+ case 16:
+ govrh_set_contrast(p_govrh, vpp_proc_value);
+ break;
+ case 17:
+ govrh_set_saturation(p_govrh, vpp_proc_value);
+ break;
+ default:
+ break;
+ }
+ }
+ return ret;
+}
+
+struct proc_dir_entry *vpp_proc_dir;
+static ctl_table vpp_table[] = {
+ { /* .ctl_name = 1, */
+ .procname = "dbg_msg",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 2, */
+ .procname = "dbg_wait",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 3, */
+ .procname = "dbg_flag",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 4, */
+ .procname = "edid_disable",
+ .data = &edid_disable,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 5, */
+ .procname = "edid_msg",
+ .data = &edid_msg_enable,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 6, */
+ .procname = "vout_edid",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 7, */
+ .procname = "vo_mode",
+ .data = vpp_proc_str,
+ .maxlen = 12,
+ .mode = 0666,
+ .proc_handler = &proc_dostring,
+ },
+ { /* .ctl_name = 8, */
+ .procname = "govr1_pixclk",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 9, */
+ .procname = "govr2_pixclk",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 10, */
+ .procname = "scl_scale_mode",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 11, */
+ .procname = "scl_filter",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 12 */
+ .procname = "hdmi_cp_enable",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 13 */
+ .procname = "hdmi_3d",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 14 */
+ .procname = "hdmi_certify",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 15 */
+ .procname = "brightness",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 16 */
+ .procname = "contrast",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* .ctl_name = 17 */
+ .procname = "saturation",
+ .data = &vpp_proc_value,
+ .maxlen = sizeof(int),
+ .mode = 0666,
+ .proc_handler = &vpp_do_proc,
+ },
+ { /* end of table */
+ }
+};
+
+static ctl_table vpp_root_table[] = {
+ {
+ .procname = "vpp", /* create path ==> /proc/sys/vpp */
+ .mode = 0555,
+ .child = vpp_table
+ },
+ { /* end of table */
+ }
+};
+static struct ctl_table_header *vpp_table_header;
+#endif
+
+static int vpp_sts_read_proc(char *buf, char **start, off_t offset,
+ int len, int *eof, void *data)
+{
+ volatile struct govrh_regs *regs =
+ (volatile struct govrh_regs *) REG_GOVRH_BASE1_BEGIN;
+ unsigned int yaddr, caddr;
+ char *p = buf;
+ unsigned int reg;
+
+ p += sprintf(p, "--- VPP HW status ---\n");
+#ifdef WMT_FTBLK_GOVRH
+ p += sprintf(p, "GOVRH memory read underrun error %d,cnt %d,cnt2 %d\n",
+ (regs->interrupt.val & 0x200) ? 1 : 0,
+ p_govrh->underrun_cnt, p_govrh2->underrun_cnt);
+ p_govrh->clr_sts(VPP_INT_ALL);
+#endif
+
+#ifdef WMT_FTBLK_SCL
+ p += sprintf(p, "---------------------------------------\n");
+ p += sprintf(p, "SCL TG error %d\n",
+ vppif_reg32_read(SCL_INTSTS_TGERR));
+ p += sprintf(p, "SCLR MIF1 read error %d\n",
+ vppif_reg32_read(SCLR_INTSTS_R1MIFERR));
+ p += sprintf(p, "SCLR MIF2 read error %d\n",
+ vppif_reg32_read(SCLR_INTSTS_R2MIFERR));
+ p += sprintf(p, "SCLW RGB fifo overflow %d\n",
+ vppif_reg32_read(SCLW_INTSTS_MIFRGBERR));
+ p += sprintf(p, "SCLW Y fifo overflow %d\n",
+ vppif_reg32_read(SCLW_INTSTS_MIFYERR));
+ p += sprintf(p, "SCLW C fifo overflow %d\n",
+ vppif_reg32_read(SCLW_INTSTS_MIFCERR));
+ p_scl->clr_sts(VPP_INT_ALL);
+#endif
+
+ p += sprintf(p, "---------------------------------------\n");
+ p += sprintf(p, "(880.0)GOVRH Enable %d,(900.0)TG %d\n",
+ regs->mif.b.enable, regs->tg_enable.b.enable);
+
+ reg = REG32_VAL(PM_CTRL_BASE_ADDR + 0x258);
+ p += sprintf(p, "--- POWER CONTROL ---\n");
+ p += sprintf(p, "0x%x = 0x%x\n", PM_CTRL_BASE_ADDR + 0x258, reg);
+ p += sprintf(p, "HDCP %d,VPP %d,SCL %d,HDMI I2C %d\n",
+ (reg & BIT7) ? 1 : 0, (reg & BIT18) ? 1 : 0,
+ (reg & BIT21) ? 1 : 0, (reg & BIT22) ? 1 : 0);
+ p += sprintf(p, "HDMI %d,GOVR %d,NA12 %d\n",
+ (reg & BIT23) ? 1 : 0, (reg & BIT25) ? 1 : 0,
+ (reg & BIT16) ? 1 : 0);
+ p += sprintf(p, "DVO %d,HDMI OUT %d,LVDS %d\n", (reg & BIT29) ? 1 : 0,
+ (reg & BIT30) ? 1 : 0, (reg & BIT14) ? 1 : 0);
+
+ p += sprintf(p, "--- VPP fb Address ---\n");
+ p += sprintf(p, "GOV mb 0x%x 0x%x\n", g_vpp.mb[0], g_vpp.mb[1]);
+
+#ifdef WMT_FTBLK_GOVRH
+ govrh_get_fb_addr(p_govrh, &yaddr, &caddr);
+ p += sprintf(p, "GOVRH fb addr Y(0x%x) 0x%x, C(0x%x) 0x%x\n",
+ REG_GOVRH_YSA, yaddr, REG_GOVRH_CSA, caddr);
+ govrh_get_fb_addr(p_govrh2, &yaddr, &caddr);
+ p += sprintf(p, "GOVRH2 fb addr Y(0x%x) 0x%x, C(0x%x) 0x%x\n",
+ REG_GOVRH2_YSA, yaddr, REG_GOVRH2_CSA, caddr);
+#endif
+ p_govrh->underrun_cnt = 0;
+ p_govrh2->underrun_cnt = 0;
+ return p - buf;
+} /* End of vpp_sts_read_proc */
+
+static int vpp_reg_read_proc(char *buf, char **start, off_t offset,
+ int len, int *eof, void *data)
+{
+ char *p = buf;
+ vpp_mod_base_t *mod_p;
+ int i;
+
+ DPRINT("Product ID:0x%x\n", vpp_get_chipid());
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->dump_reg)
+ mod_p->dump_reg();
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_reg_dump();
+#endif
+#ifdef WMT_FTBLK_LVDS
+ lvds_reg_dump();
+#endif
+ return p - buf;
+} /* End of vpp_reg_read_proc */
+#endif
+
+/*----------------------- fb ioctl --------------------------------------*/
+int vpp_common_ioctl(unsigned int cmd, unsigned long arg)
+{
+ vpp_mod_base_t *mod_p;
+ vpp_fb_base_t *mod_fb_p;
+ int retval = 0;
+
+ switch (cmd) {
+ case VPPIO_VPPGET_INFO:
+ {
+ int i;
+ vpp_cap_t parm;
+
+ parm.chip_id = vpp_get_chipid();
+ parm.version = 0x01;
+ parm.resx_max = VPP_HD_MAX_RESX;
+ parm.resy_max = VPP_HD_MAX_RESY;
+ parm.pixel_clk = 400000000;
+ parm.module = 0x0;
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p)
+ parm.module |= (0x01 << i);
+ }
+ parm.option = VPP_CAP_DUAL_DISPLAY;
+ copy_to_user((void *)arg, (void *) &parm, sizeof(vpp_cap_t));
+ }
+ break;
+ case VPPIO_VPPSET_INFO:
+ {
+ vpp_cap_t parm;
+
+ copy_from_user((void *)&parm, (const void *)arg,
+ sizeof(vpp_cap_t));
+ }
+ break;
+ case VPPIO_I2CSET_BYTE:
+ {
+ vpp_i2c_t parm;
+ unsigned int id;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_i2c_t));
+ id = (parm.addr & 0x0000FF00) >> 8;
+ vpp_i2c_write(id, (parm.addr & 0xFF), parm.index,
+ (char *)&parm.val, 1);
+ }
+ break;
+ case VPPIO_I2CGET_BYTE:
+ {
+ vpp_i2c_t parm;
+ unsigned int id;
+ int len;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_i2c_t));
+ id = (parm.addr & 0x0000FF00) >> 8;
+ len = parm.val;
+ {
+ unsigned char buf[len];
+
+ vpp_i2c_read(id, (parm.addr & 0xFF), parm.index,
+ buf, len);
+ parm.val = buf[0];
+ }
+ copy_to_user((void *)arg, (void *) &parm, sizeof(vpp_i2c_t));
+ }
+ break;
+ case VPPIO_VPPSET_FBDISP:
+ break;
+ case VPPIO_VPPGET_FBDISP:
+ {
+ vpp_dispfb_info_t parm;
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_dispfb_info_t));
+ }
+ break;
+ case VPPIO_WAIT_FRAME:
+ vpp_wait_vsync(0, arg);
+ break;
+ case VPPIO_MODULE_FRAMERATE:
+ {
+ vpp_mod_arg_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_arg_t));
+ mod_fb_p = vpp_mod_get_fb_base(parm.mod);
+ if (!mod_fb_p)
+ break;
+ if (parm.read) {
+ parm.arg1 = mod_fb_p->framerate;
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_mod_arg_t));
+ } else {
+ mod_fb_p->framerate = parm.arg1;
+ }
+ }
+ break;
+ case VPPIO_MODULE_ENABLE:
+ {
+ vpp_mod_arg_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_arg_t));
+ mod_p = vpp_mod_get_base(parm.mod);
+ if (!mod_p)
+ break;
+
+ if (parm.read) {
+ /* not used */
+ } else {
+ mod_p->set_enable(parm.arg1);
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ if (parm.mod == VPP_MOD_CURSOR) {
+ p_cursor->enable = parm.arg1;
+ if (p_cursor->enable) {
+ vpp_irqproc_work(VPP_INT_GOVRH_PVBI,
+ (void *)govrh_CUR_irqproc, 0,
+ 0, 0);
+ } else {
+ vpp_irqproc_del_work(VPP_INT_GOVRH_PVBI,
+ (void *)govrh_CUR_irqproc);
+ }
+ }
+#endif
+ }
+ }
+ break;
+ case VPPIO_MODULE_TIMING:
+ DPRINT("remove for new arch\n");
+ break;
+ case VPPIO_MODULE_FBADDR:
+ {
+ vpp_mod_arg_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_arg_t));
+ mod_fb_p = vpp_mod_get_fb_base(parm.mod);
+ if (!mod_fb_p)
+ break;
+ mod_p = vpp_mod_get_base(parm.mod);
+ if (parm.read) {
+ mod_fb_p->get_addr(&parm.arg1, &parm.arg2);
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_mod_arg_t));
+ } else {
+ mod_fb_p->set_addr(parm.arg1, parm.arg2);
+ }
+ }
+ break;
+ case VPPIO_MODULE_FBINFO:
+ {
+ vpp_mod_fbinfo_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_fbinfo_t));
+
+ if (g_vpp.virtual_display)
+ parm.mod = (hdmi_get_plugin()) ?
+ VPP_MOD_GOVRH : VPP_MOD_GOVRH2;
+ mod_fb_p = vpp_mod_get_fb_base(parm.mod);
+ if (!mod_fb_p)
+ break;
+ mod_p = vpp_mod_get_base(parm.mod);
+ if (parm.read) {
+ parm.fb = mod_fb_p->fb;
+ switch (parm.mod) {
+ case VPP_MOD_GOVRH:
+ case VPP_MOD_GOVRH2:
+ govrh_get_framebuffer((govrh_mod_t *)mod_p,
+ &parm.fb);
+ break;
+ default:
+ break;
+ }
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_mod_fbinfo_t));
+ } else {
+ mod_fb_p->fb = parm.fb;
+ mod_fb_p->set_framebuf(&parm.fb);
+
+ if (g_vpp.virtual_display) {
+#ifdef CONFIG_VPP_STREAM_CAPTURE
+ if (g_vpp.stream_enable)
+ vpp_mb_scale_bitblit(&mod_fb_p->fb);
+ else
+#endif
+ {
+ if(g_vpp.mb[0] != 0)
+ vpp_free_framebuffer();
+ }
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(
+ &vout_info[1].pandisp_timer,
+ "fbinfo", 2);
+ }
+ }
+ }
+ break;
+ case VPPIO_MODULE_VIEW:
+ {
+ vpp_mod_view_t parm;
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_view_t));
+
+ mod_fb_p = vpp_mod_get_fb_base(parm.mod);
+ if (!mod_fb_p)
+ break;
+ mod_p = vpp_mod_get_base(parm.mod);
+ if (parm.read) {
+ mod_fb_p->fn_view(VPP_FLAG_RD, &parm.view);
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_mod_view_t));
+ } else {
+ mod_fb_p->fn_view(0, &parm.view);
+ }
+ }
+ break;
+#ifdef CONFIG_VPP_STREAM_CAPTURE
+ case VPPIO_STREAM_ENABLE:
+ g_vpp.stream_enable = arg;
+ g_vpp.stream_mb_sync_flag = 0;
+
+ MSG("[VPP] VPPIO_STREAM_ENABLE %d\n", g_vpp.stream_enable);
+#ifdef CONFIG_VPP_STREAM_FIX_RESOLUTION
+ memset(&g_vpp.stream_fb, 0, sizeof(vdo_framebuf_t));
+ g_vpp.stream_fb.img_w = 1280;
+ g_vpp.stream_fb.img_h = 720;
+ g_vpp.stream_fb.fb_w =
+ vpp_calc_align(g_vpp.stream_fb.img_w, 64);
+ g_vpp.stream_fb.fb_h = g_vpp.stream_fb.img_h;
+ g_vpp.stream_fb.bpp = 16;
+ g_vpp.stream_fb.col_fmt = VDO_COL_FMT_YUV422H;
+ g_vpp.stream_fb.y_size = g_vpp.stream_fb.fb_w *
+ g_vpp.stream_fb.fb_h;
+ g_vpp.stream_fb.c_size = g_vpp.stream_fb.fb_w *
+ g_vpp.stream_fb.fb_h;
+#endif
+#ifdef CONFIG_VPP_STREAM_ROTATE
+ g_vpp.stream_mb_lock= 0;
+ g_vpp.stream_mb_index = 0xFF;
+ vpp_netlink_notify(WP_PID, DEVICE_STREAM,
+ (g_vpp.stream_enable) ? 1 : 0);
+#else
+#ifndef CONFIG_VPP_DYNAMIC_ALLOC
+ if (g_vpp.alloc_framebuf == 0) {
+#endif
+ {
+ if (arg) {
+#ifdef CONFIG_VPP_STREAM_FIX_RESOLUTION
+ vpp_alloc_framebuffer(g_vpp.stream_fb.fb_w,
+ g_vpp.stream_fb.fb_h);
+#else
+ vout_info_t *info;
+
+ info = vout_info_get_entry(0);
+ vpp_alloc_framebuffer(info->resx, info->resy);
+#endif
+ } else {
+ vpp_free_framebuffer();
+ }
+ }
+ vpp_mod_set_clock(VPP_MOD_SCL, g_vpp.stream_enable, 0);
+ vpp_pan_display(0, 0);
+ if (!g_vpp.stream_enable) {
+ vpp_lock();
+ vpp_mb_put(0);
+ vpp_unlock();
+ }
+#endif
+ {
+#if 0
+ vpp_int_t type;
+
+ type = (vout_info[0].govr_mod == VPP_MOD_GOVRH) ?
+ VPP_INT_GOVRH_VBIS : VPP_INT_GOVRH2_VBIS;
+ if (g_vpp.stream_enable) {
+ vpp_irqproc_work(type, (void *)vpp_mb_irqproc_sync,
+ vout_info[0].govr, 0, 0);
+ } else {
+ vpp_irqproc_del_work(type, (void *)vpp_mb_irqproc_sync);
+ }
+#endif
+ wmt_enable_mmfreq(WMT_MMFREQ_MIRACAST, g_vpp.stream_enable);
+ }
+ break;
+ case VPPIO_STREAM_GETFB:
+ {
+ vdo_framebuf_t fb;
+
+ vpp_lock();
+#ifdef CONFIG_VPP_STREAM_ROTATE
+ if (g_vpp.stream_mb_index == 0xFF) { /* not avail */
+ retval = -1;
+ vpp_unlock();
+ break;
+ }
+ fb = vout_info[1].fb;
+#else
+#ifdef CONFIG_VPP_STREAM_FIX_RESOLUTION
+ fb = g_vpp.stream_fb;
+#else
+ fb = vout_info[0].fb;
+#endif
+#endif
+ fb.fb_w = vpp_calc_align(fb.fb_w, 64);
+ fb.y_addr = g_vpp.stream_mb[g_vpp.stream_mb_index];
+ fb.y_size = fb.fb_w * fb.img_h;
+ fb.c_addr = fb.y_addr + fb.y_size;
+ fb.c_size = fb.y_size;
+ fb.col_fmt = VDO_COL_FMT_YUV422H;
+
+ copy_to_user((void *)arg, (void *) &fb, sizeof(vdo_framebuf_t));
+ retval = vpp_mb_get(fb.y_addr);
+ vpp_unlock();
+ }
+ break;
+ case VPPIO_STREAM_PUTFB:
+ {
+ vdo_framebuf_t fb;
+ copy_from_user((void *) &fb, (const void *)arg,
+ sizeof(vdo_framebuf_t));
+ vpp_lock();
+ vpp_mb_put(fb.y_addr);
+ vpp_unlock();
+ }
+ break;
+#endif
+ case VPPIO_MODULE_CSC:
+ {
+ vpp_mod_arg_t parm;
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_mod_arg_t));
+ mod_fb_p = vpp_mod_get_fb_base(parm.mod);
+ if (!mod_fb_p)
+ break;
+ mod_p = vpp_mod_get_base(parm.mod);
+ if (parm.read) {
+ parm.arg1 = mod_fb_p->csc_mode;
+ copy_to_user((void *)arg, (void *) &parm,
+ sizeof(vpp_mod_arg_t));
+ } else {
+ mod_fb_p->csc_mode = parm.arg1;
+ mod_fb_p->set_csc(mod_fb_p->csc_mode);
+ }
+ }
+ break;
+ case VPPIO_MULTIVD_ENABLE:
+ wmt_enable_mmfreq(WMT_MMFREQ_MULTI_VD, arg);
+ break;
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+} /* End of vpp_common_ioctl */
+
+int vout_ioctl(unsigned int cmd, unsigned long arg)
+{
+ int retval = 0;
+
+ switch (cmd) {
+ case VPPIO_VOGET_INFO:
+ {
+ vpp_vout_info_t parm;
+ vout_t *vo;
+ int num;
+ vout_inf_t *inf;
+ int fb_num;
+ vout_info_t *vo_info;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_info_t));
+ fb_num = (parm.num & 0xF0) >> 4;
+ num = parm.num & 0xF;
+ DBGMSG("VPPIO_VOGET_INFO %d,%d\n", fb_num, num);
+ if ((fb_num >= VPP_VOUT_INFO_NUM) || (num >= VOUT_MODE_MAX)) {
+ retval = -ENOTTY;
+ DBG_MSG("fb_num or num invalid\n");
+ break;
+ }
+ memset(&parm, 0, sizeof(vpp_vout_info_t));
+ vo = vout_get_entry_adapter(num);
+ if (vo == 0) {
+ retval = -ENOTTY;
+ DBG_MSG("vo null\n");
+ break;
+ }
+ inf = vout_get_inf_entry_adapter(num);
+ if (inf == 0) {
+ retval = -ENOTTY;
+ DBG_MSG("inf null\n");
+ break;
+ }
+ vo_info = vout_get_info_entry(vo->num);
+ if (vo_info == 0) {
+ retval = -ENOTTY;
+ DBG_MSG("vo_info null\n");
+ break;
+ }
+ if ((g_vpp.dual_display == 1) && (g_vpp.virtual_display == 0)
+ && (fb_num != vo_info->num)) {
+ retval = -ENOTTY;
+ DBG_MSG("fb num not match\n");
+ break;
+ }
+ DBGMSG("VPPIO_VOGET_INFO %d,fb%d,0x%x,0x%x\n", num,
+ vo_info->num, (int)vo, (int)inf);
+ if ((vo == 0) || (inf == 0))
+ goto label_get_info;
+
+ parm.num = num;
+ parm.status = vo->status | VPP_VOUT_STS_REGISTER;
+ strncpy(parm.name, vout_adpt_str[num], 10);
+ switch (inf->mode) {
+ case VOUT_INF_DVI:
+ parm.status &= ~VPP_VOUT_STS_ACTIVE;
+ if (vo->dev) {
+ /* check current DVI is dvi/dvo2hdmi/lcd */
+ switch (num) {
+ case VOUT_DVI:
+ if (strcmp("VT1632",
+ vo->dev->name) == 0)
+ parm.status |=
+ VPP_VOUT_STS_ACTIVE;
+ parm.status |= VPP_VOUT_STS_ACTIVE;
+ break;
+ case VOUT_LCD:
+ if (strcmp("LCD",
+ vo->dev->name) == 0)
+ parm.status |=
+ VPP_VOUT_STS_ACTIVE;
+ break;
+ case VOUT_DVO2HDMI:
+ if (strcmp("SIL902X",
+ vo->dev->name) == 0)
+ parm.status |=
+ VPP_VOUT_STS_ACTIVE;
+ break;
+ default:
+ break;
+ }
+ } else { /* dvi hw mode */
+ if (num == VOUT_DVI)
+ parm.status |= VPP_VOUT_STS_ACTIVE;
+ }
+
+ if (g_vpp.virtual_display) {
+ if (vout_chkplug(VPP_VOUT_NUM_HDMI))
+ parm.status &= ~VPP_VOUT_STS_ACTIVE;
+ }
+ break;
+ case VOUT_INF_HDMI:
+ case VOUT_INF_LVDS:
+ /* check current HDMI is HDMI or LVDS */
+ if (vo->inf != inf)
+ parm.status = VPP_VOUT_STS_REGISTER;
+ break;
+ default:
+ break;
+ }
+
+ if (parm.status & VPP_VOUT_STS_ACTIVE) {
+ if (vout_chkplug(vo->num))
+ parm.status |= VPP_VOUT_STS_PLUGIN;
+ else
+ parm.status &= ~VPP_VOUT_STS_PLUGIN;
+ } else {
+ parm.status = VPP_VOUT_STS_REGISTER;
+ }
+label_get_info:
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_info_t));
+ }
+ break;
+ case VPPIO_VOSET_MODE:
+ {
+ vpp_vout_parm_t parm;
+ vout_t *vo;
+ vout_inf_t *inf;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_parm_t));
+ vo = vout_get_entry_adapter(parm.num);
+ inf = vout_get_inf_entry_adapter(parm.num);
+ if (vo && inf) {
+ DBG_DETAIL("VPPIO_VOSET_MODE %d %d\n",
+ vo->num, parm.arg);
+ if (g_vpp.dual_display == 0) {
+ int plugin;
+
+ plugin = vout_chkplug(VPP_VOUT_NUM_HDMI);
+ vout_set_blank((0x1 << VPP_VOUT_NUM_HDMI),
+ (plugin) ? 0 : 1);
+ vout_set_blank((0x1 << VPP_VOUT_NUM_DVI),
+ (plugin) ? 1 : 0);
+ break;
+ }
+ vout_set_blank((0x1 << vo->num), (parm.arg) ? 0 : 1);
+ retval = vout_set_mode(vo->num, inf->mode);
+ }
+ }
+ break;
+ case VPPIO_VOSET_BLANK:
+ {
+ vpp_vout_parm_t parm;
+ vout_t *vo;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_parm_t));
+ vo = vout_get_entry_adapter(parm.num);
+ if (vo)
+ retval = vout_set_blank((0x1 << vo->num), parm.arg);
+ }
+ break;
+ case VPPIO_VOSET_OPTION:
+ {
+ vpp_vout_option_t option;
+ vout_t *vo;
+ int num;
+ vout_inf_t *inf;
+
+ copy_from_user((void *) &option, (const void *)arg,
+ sizeof(vpp_vout_option_t));
+ num = option.num;
+ if (num >= VOUT_MODE_MAX) {
+ retval = -ENOTTY;
+ break;
+ }
+ vo = vout_get_entry_adapter(num);
+ inf = vout_get_inf_entry_adapter(num);
+ if (vo && inf) {
+ vo->option[0] = option.option[0];
+ vo->option[1] = option.option[1];
+ vo->option[2] = option.option[2];
+ vout_set_mode(vo->num, inf->mode);
+ }
+ }
+ break;
+ case VPPIO_VOGET_OPTION:
+ {
+ vpp_vout_option_t option;
+ vout_t *vo;
+ int num;
+
+ copy_from_user((void *) &option, (const void *)arg,
+ sizeof(vpp_vout_option_t));
+ num = option.num;
+ if (num >= VOUT_MODE_MAX) {
+ retval = -ENOTTY;
+ break;
+ }
+ memset(&option, 0, sizeof(vpp_vout_info_t));
+ vo = vout_get_entry_adapter(num);
+ if (vo) {
+ option.num = num;
+ option.option[0] = vo->option[0];
+ option.option[1] = vo->option[1];
+ option.option[2] = vo->option[2];
+ }
+ copy_to_user((void *)arg, (const void *) &option,
+ sizeof(vpp_vout_option_t));
+ }
+ break;
+ case VPPIO_VOUT_VMODE:
+ {
+ vpp_vout_vmode_t parm;
+ int i;
+ struct fb_videomode *vmode;
+ unsigned int resx, resy, fps;
+ unsigned int pre_resx, pre_resy, pre_fps;
+ int index, from_index;
+ int support;
+ unsigned int option, pre_option;
+#ifdef CONFIG_WMT_EDID
+ edid_info_t *edid_info;
+#endif
+ struct fb_videomode *edid_vmode;
+ vpp_vout_t mode;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_vmode_t));
+ from_index = parm.num;
+ parm.num = 0;
+ mode = parm.mode & 0xF;
+#ifdef CONFIG_VPP_DEMO
+ parm.parm[parm.num].resx = 1920;
+ parm.parm[parm.num].resy = 1080;
+ parm.parm[parm.num].fps = 60;
+ parm.parm[parm.num].option = 0;
+ parm.num++;
+#else
+#ifdef CONFIG_WMT_EDID
+ {
+ vout_t *vo;
+
+ vo = vout_get_entry_adapter(mode);
+ if (!vo)
+ goto vout_vmode_end;
+
+ if (!(vo->status & VPP_VOUT_STS_PLUGIN)) {
+ DPRINT("*W* not plugin\n");
+ goto vout_vmode_end;
+ }
+
+ if (vout_get_edid(vo->num) == 0) {
+ DPRINT("*W* read EDID fail\n");
+ goto vout_vmode_end;
+ }
+ if (edid_parse(vo->edid, &vo->edid_info) == 0) {
+ DPRINT("*W* parse EDID fail\n");
+ goto vout_vmode_end;
+ }
+ edid_info = &vo->edid_info;
+ }
+#endif
+ index = 0;
+ resx = resy = fps = option = 0;
+ pre_resx = pre_resy = pre_fps = pre_option = 0;
+ for (i = 0; ; i++) {
+ vmode = (struct fb_videomode *) &vpp_videomode[i];
+ if (vmode->pixclock == 0)
+ break;
+ resx = vmode->xres;
+ resy = vmode->yres;
+ fps = vmode->refresh;
+ option = fps & EDID_TMR_FREQ;
+ option |= (vmode->vmode & FB_VMODE_INTERLACED) ?
+ EDID_TMR_INTERLACE : 0;
+ if ((pre_resx == resx) && (pre_resy == resy) &&
+ (pre_fps == fps) && (pre_option == option))
+ continue;
+ pre_resx = resx;
+ pre_resy = resy;
+ pre_fps = fps;
+ pre_option = option;
+ support = 0;
+#ifdef CONFIG_WMT_EDID
+ if (edid_find_support(edid_info, resx, resy,
+ option, &edid_vmode))
+#else
+ if (1)
+#endif
+ support = 1;
+
+ switch (mode) {
+ case VPP_VOUT_HDMI:
+ case VPP_VOUT_DVO2HDMI:
+ if (g_vpp.hdmi_video_mode) {
+ if (resy > g_vpp.hdmi_video_mode)
+ support = 0;
+ }
+ break;
+ default:
+ break;
+ }
+
+ if (support) {
+ if (index >= from_index) {
+ parm.parm[parm.num].resx = resx;
+ parm.parm[parm.num].resy = resy;
+ parm.parm[parm.num].fps = fps;
+ parm.parm[parm.num].option =
+ vmode->vmode;
+ parm.num++;
+ }
+ index++;
+ if (parm.num >= VPP_VOUT_VMODE_NUM)
+ break;
+ }
+ }
+#ifdef CONFIG_WMT_EDID
+vout_vmode_end:
+#endif
+#endif
+ if(parm.num == 0) { /* if no EDID*/
+ if(g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ if(mode == VPP_VOUT_DVI) {
+ char buf[40] = {0};
+ int varlen = 40;
+
+ if(wmt_getsyspara("wmt.display.tvformat", buf, &varlen) == 0) {
+ if(!strnicmp(buf, "PAL", 3) || !strnicmp(buf, "NTSC", 4)) {
+ parm.parm[0].resx = 720;
+ parm.parm[0].resy = 576;
+ parm.parm[0].fps = 50;
+ parm.parm[0].option = 0;
+
+ parm.parm[1].resx = 720;
+ parm.parm[1].resy = 480;
+ parm.parm[1].fps = 60;
+ parm.parm[1].option = 0;
+
+ parm.num = 2;
+ }
+ }
+ }
+ }
+ }
+
+ DBG_MSG("[VPP] get support vmode %d\n", parm.num);
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_vmode_t));
+ }
+ break;
+ case VPPIO_VOGET_EDID:
+ {
+ vpp_vout_edid_t parm;
+ char *edid;
+ vout_t *vo;
+ int size;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_edid_t));
+ size = 0;
+#ifdef CONFIG_WMT_EDID
+ vo = vout_get_entry_adapter(parm.mode);
+ if (!vo)
+ goto vout_edid_end;
+
+ if (!(vo->status & VPP_VOUT_STS_PLUGIN)) {
+ DBG_ERR("*W* not plugin\n");
+ goto vout_edid_end;
+ }
+
+ edid = vout_get_edid(vo->num);
+ if (edid == 0) {
+ DBG_ERR("*W* read EDID fail\n");
+ goto vout_edid_end;
+ }
+ size = (edid[0x7E] + 1) * 128;
+ if (size > parm.size)
+ size = parm.size;
+ copy_to_user((void *) parm.buf, (void *) edid, size);
+vout_edid_end:
+#endif
+ parm.size = size;
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_edid_t));
+ }
+ break;
+ case VPPIO_VOGET_CP_INFO:
+ {
+ vpp_vout_cp_info_t parm;
+ int num;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_vout_cp_info_t));
+ num = parm.num;
+ if (num >= VOUT_MODE_MAX) {
+ retval = -ENOTTY;
+ break;
+ }
+ memset(&parm, 0, sizeof(vpp_vout_cp_info_t));
+ switch (num) {
+ case VOUT_HDMI:
+ if (!g_vpp.hdmi_certify_flag) {
+ if (g_vpp.hdmi_bksv[0] == 0) {
+ hdmi_get_bksv(&g_vpp.hdmi_bksv[0]);
+ DBG_MSG("get BKSV 0x%x 0x%x\n",
+ g_vpp.hdmi_bksv[0],
+ g_vpp.hdmi_bksv[1]);
+ }
+ }
+ case VOUT_DVO2HDMI:
+ parm.bksv[0] = g_vpp.hdmi_bksv[0];
+ parm.bksv[1] = g_vpp.hdmi_bksv[1];
+ break;
+ default:
+ parm.bksv[0] = parm.bksv[1] = 0;
+ break;
+ }
+ copy_to_user((void *)arg, (const void *) &parm,
+ sizeof(vpp_vout_cp_info_t));
+ }
+ break;
+ case VPPIO_VOSET_CP_KEY:
+ if (g_vpp.hdmi_cp_p == 0) {
+ g_vpp.hdmi_cp_p =
+ (char *)kmalloc(sizeof(vpp_vout_cp_key_t),
+ GFP_KERNEL);
+ }
+ if (g_vpp.hdmi_cp_p) {
+ copy_from_user((void *) g_vpp.hdmi_cp_p,
+ (const void *)arg, sizeof(vpp_vout_cp_key_t));
+ if (hdmi_cp)
+ hdmi_cp->init();
+ }
+ break;
+#ifdef WMT_FTBLK_HDMI
+ case VPPIO_VOSET_AUDIO_PASSTHRU:
+ vppif_reg32_write(HDMI_AUD_SUB_PACKET, (arg) ? 0xF : 0x0);
+ break;
+#endif
+#ifdef CONFIG_VPP_VIRTUAL_DISPLAY
+ case VPPIO_VOSET_VIRTUAL_FBDEV:
+ g_vpp.fb0_bitblit = (arg) ? 0 : 1;
+ MSG("[VPP] virtual display %d\n", (int)arg);
+ break;
+#endif
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+} /* End of vout_ioctl */
+
+int govr_ioctl(unsigned int cmd, unsigned long arg)
+{
+ int retval = 0;
+
+ switch (cmd) {
+#ifdef WMT_FTBLK_GOVRH
+ case VPPIO_GOVRSET_DVO:
+ {
+ vdo_dvo_parm_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vdo_dvo_parm_t));
+ govrh_set_dvo_enable(p_govrh, parm.enable);
+ govrh_set_dvo_color_format(p_govrh, parm.color_fmt);
+ govrh_set_dvo_clock_delay(p_govrh, parm.clk_inv,
+ parm.clk_delay);
+ govrh_set_dvo_outdatw(p_govrh, parm.data_w);
+ govrh_set_dvo_sync_polar(p_govrh, parm.sync_polar,
+ parm.vsync_polar);
+ p_govrh->fb_p->set_csc(p_govrh->fb_p->csc_mode);
+ }
+ break;
+#endif
+#ifdef WMT_FTBLK_GOVRH_CURSOR
+ case VPPIO_GOVRSET_CUR_COLKEY:
+ {
+ vpp_mod_arg_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vdo_dvo_parm_t));
+ govrh_CUR_set_color_key(p_cursor, VPP_FLAG_ENABLE,
+ 0, parm.arg1);
+ }
+ break;
+ case VPPIO_GOVRSET_CUR_HOTSPOT:
+ {
+ vpp_mod_arg_t parm;
+ vdo_view_t view;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vdo_dvo_parm_t));
+ p_cursor->hotspot_x = parm.arg1;
+ p_cursor->hotspot_y = parm.arg2;
+ view.posx = p_cursor->posx;
+ view.posy = p_cursor->posy;
+ p_cursor->fb_p->fn_view(0, &view);
+ }
+ break;
+#endif
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+} /* End of govr_ioctl */
+
+int scl_ioctl(unsigned int cmd, unsigned long arg)
+{
+ int retval = 0;
+ extern int ge_do_alpha_bitblt(vdo_framebuf_t *src,
+ vdo_framebuf_t *src2, vdo_framebuf_t *dst);
+
+ switch (cmd) {
+ case VPPIO_SCL_SCALE_OVERLAP:
+ {
+ vpp_scale_overlap_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_scale_overlap_t));
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->overlap_timer, 0, 1);
+
+ p_scl->scale_sync = 1;
+ retval = scl_set_scale_overlap(&parm.src_fb,
+ &parm.src2_fb, &parm.dst_fb);
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->overlap_timer, "overlap", 2);
+ }
+ break;
+ case VPPIO_SCL_SCALE_ASYNC:
+ case VPPIO_SCL_SCALE:
+ {
+ vpp_scale_t parm;
+
+ p_scl->scale_sync = (cmd == VPPIO_SCL_SCALE) ? 1 : 0;
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_scale_t));
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->scale_timer, 0, 1);
+
+ vpp_set_NA12_hiprio(1);
+ retval = vpp_set_recursive_scale(&parm.src_fb, &parm.dst_fb);
+ vpp_set_NA12_hiprio(0);
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&p_scl->scale_timer, "scale", 2);
+ copy_to_user((void *)arg, (void *)&parm, sizeof(vpp_scale_t));
+ }
+ break;
+#ifdef WMT_FTBLK_SCL
+ case VPPIO_SCL_DROP_LINE_ENABLE:
+ scl_set_drop_line(arg);
+ break;
+#endif
+ case VPPIO_SCL_SCALE_FINISH:
+ retval = p_scl->scale_finish();
+ break;
+ case VPPIO_SCLSET_OVERLAP:
+ {
+ vpp_overlap_t parm;
+
+ copy_from_user((void *) &parm, (const void *)arg,
+ sizeof(vpp_overlap_t));
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_ENABLE, 0);
+ scl_set_overlap(&parm);
+ vpp_mod_set_clock(VPP_MOD_SCL, VPP_FLAG_DISABLE, 0);
+ }
+ break;
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+ return retval;
+} /* End of scl_ioctl */
+
+int vpp_ioctl(unsigned int cmd, unsigned long arg)
+{
+ int retval = 0;
+ int err = 0;
+ int skip_mutex = 0;
+
+/* DBGMSG("vpp_ioctl\n"); */
+ switch (_IOC_TYPE(cmd)) {
+ case VPPIO_MAGIC:
+ break;
+ default:
+ return -ENOTTY;
+ }
+
+ /* check argument area */
+ if (_IOC_DIR(cmd) & _IOC_READ)
+ err = !access_ok(VERIFY_WRITE, (void __user *) arg,
+ _IOC_SIZE(cmd));
+ else if (_IOC_DIR(cmd) & _IOC_WRITE)
+ err = !access_ok(VERIFY_READ, (void __user *) arg,
+ _IOC_SIZE(cmd));
+
+ if (err)
+ return -EFAULT;
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_IOCTL)) {
+ switch (cmd) {
+ case VPPIO_VPPSET_FBDISP:
+ case VPPIO_VPPGET_FBDISP:
+ case VPPIO_MODULE_VIEW: /* cursor pos */
+ break;
+ default:
+ DPRINT("[VPP] ioctl cmd 0x%x,arg 0x%x\n",
+ _IOC_NR(cmd), (int)arg);
+ break;
+ }
+ }
+
+ switch (cmd) {
+ case VPPIO_STREAM_GETFB: /* block mode should wait vsync */
+ skip_mutex = 1;
+ break;
+ default:
+ /* scale should wait complete */
+ if ((_IOC_NR(cmd) >= VPPIO_SCL_BASE) &&
+ (_IOC_NR(cmd) < VPPIO_MAX))
+ skip_mutex = 1;
+ break;
+ }
+
+ if (!skip_mutex)
+ vpp_set_mutex(1, 1);
+
+ switch (_IOC_NR(cmd)) {
+ case VPPIO_VPP_BASE ... (VPPIO_VOUT_BASE-1):
+ /* DBGMSG("VPP command ioctl\n"); */
+ retval = vpp_common_ioctl(cmd, arg);
+ break;
+ case VPPIO_VOUT_BASE ... (VPPIO_GOVR_BASE-1):
+ /* DBGMSG("VOUT ioctl\n"); */
+ retval = vout_ioctl(cmd, arg);
+ break;
+ case VPPIO_GOVR_BASE ... (VPPIO_SCL_BASE-1):
+ /* DBGMSG("GOVR ioctl\n"); */
+ retval = govr_ioctl(cmd, arg);
+ break;
+ case VPPIO_SCL_BASE ... (VPPIO_MAX-1):
+ /* DBGMSG("SCL ioctl\n"); */
+ retval = scl_ioctl(cmd, arg);
+ break;
+ default:
+ retval = -ENOTTY;
+ break;
+ }
+
+ if (!skip_mutex)
+ vpp_set_mutex(1, 0);
+
+ if (vpp_check_dbg_level(VPP_DBGLVL_IOCTL)) {
+ switch (cmd) {
+ case VPPIO_VPPSET_FBDISP:
+ case VPPIO_VPPGET_FBDISP:
+ case VPPIO_MODULE_VIEW:
+ break;
+ default:
+ DPRINT("[VPP] ioctl cmd 0x%x,ret 0x%x\n",
+ _IOC_NR(cmd), (int)retval);
+ break;
+ }
+ }
+ return retval;
+} /* End of vpp_ioctl */
+
+int vpp_set_blank(struct fb_info *info, int blank)
+{
+ vout_info_t *vo_info;
+ unsigned int vout_blank_mask = 0, vout_unblank_mask = 0;
+
+ if ((g_vpp.dual_display == 0) && (info->node == 1))
+ return 0;
+
+ DBG_MSG("(%d,%d)\n", info->node, blank);
+
+ vo_info = vout_info_get_entry(info->node);
+ if (blank)
+ vout_blank_mask = vout_get_mask(vo_info);
+ else
+ vout_unblank_mask = vout_get_mask(vo_info);
+
+ if (g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ if (blank) {
+ vout_blank_mask = ~0;
+ vout_unblank_mask = 0;
+ } else {
+ int plugin;
+
+ plugin = vout_chkplug(VPP_VOUT_NUM_HDMI);
+ vout_blank_mask &= ~((0x1 << VPP_VOUT_NUM_DVI) +
+ (0x1 << VPP_VOUT_NUM_HDMI));
+ vout_unblank_mask &= ~((0x1 << VPP_VOUT_NUM_DVI) +
+ (0x1 << VPP_VOUT_NUM_HDMI));
+ vout_blank_mask |= (plugin) ?
+ (0x1 << VPP_VOUT_NUM_DVI) :
+ (0x1 << VPP_VOUT_NUM_HDMI);
+ vout_unblank_mask |= (plugin) ?
+ (0x1 << VPP_VOUT_NUM_HDMI) :
+ (0x1 << VPP_VOUT_NUM_DVI);
+ }
+ }
+ vout_set_blank(vout_blank_mask, 1);
+ vout_set_blank(vout_unblank_mask, 0);
+ return 0;
+}
+
+void vpp_set_lvds_blank(int blank)
+{
+ if (lcd_get_lvds_id() != LCD_LVDS_1024x600)
+ return;
+
+ if (blank == 0) {
+ REG32_VAL(GPIO_BASE_ADDR+0xC0) |= 0x400;
+ REG32_VAL(GPIO_BASE_ADDR+0x80) |= 0x400;
+ mdelay(6);
+ }
+
+ if (blank)
+ msleep(50);
+
+ vout_set_blank((0x1 << VPP_VOUT_NUM_LVDS),
+ (blank) ? VOUT_BLANK_POWERDOWN : VOUT_BLANK_UNBLANK);
+ lvds_set_power_down((blank) ? 1 : 0);
+
+ if (blank) {
+ mdelay(6);
+ /* GPIO10 off 8ms -> clock -> off */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x400;
+ }
+}
+
+irqreturn_t vpp_interrupt_routine(int irq, void *dev_id)
+{
+ vpp_int_t int_sts;
+
+ switch (irq) {
+ case VPP_IRQ_VPPM: /* VPP */
+ int_sts = p_vppm->get_sts();
+ p_vppm->clr_sts(int_sts);
+
+ vpp_dbg_show_val1(VPP_DBGLVL_INT, 0, "[VPP] VPPM INT", int_sts);
+ {
+ int i;
+ unsigned int mask;
+ vpp_irqproc_t *irqproc;
+
+ for (i = 0, mask = 0x1; (i < 32) && int_sts; i++, mask <<= 1) {
+ if ((int_sts & mask) == 0)
+ continue;
+
+ irqproc = vpp_irqproc_get_entry(mask);
+ if (irqproc) {
+ if (list_empty(&irqproc->list) == 0)
+ tasklet_schedule(&irqproc->tasklet);
+ } else {
+ irqproc = vpp_irqproc_get_entry(VPP_INT_MAX);
+ if (list_empty(&irqproc->list) == 0) {
+ vpp_proc_t *entry;
+ struct list_head *ptr;
+
+ ptr = (&irqproc->list)->next;
+ entry = list_entry(ptr, vpp_proc_t,
+ list);
+ if (entry->type == mask)
+ tasklet_schedule(
+ &irqproc->tasklet);
+ }
+ }
+ int_sts &= ~mask;
+ }
+ }
+ break;
+#ifdef WMT_FTBLK_SCL
+ case VPP_IRQ_SCL: /* SCL */
+ int_sts = p_scl->get_sts();
+ p_scl->clr_sts(int_sts);
+ vpp_dbg_show_val1(VPP_DBGLVL_INT, 0, "[VPP] SCL INT", int_sts);
+ break;
+#endif
+#ifdef WMT_FTBLK_GOVRH
+ case VPP_IRQ_GOVR: /* GOVR */
+ case VPP_IRQ_GOVR2:
+ {
+ govrh_mod_t *govr;
+
+ govr = (irq == VPP_IRQ_GOVR) ? p_govrh : p_govrh2;
+ int_sts = govr->get_sts();
+ govr->clr_sts(int_sts);
+ vpp_dbg_show_val1(VPP_DBGLVL_INT, 0, "[VPP] GOVR INT", int_sts);
+ govr->underrun_cnt++;
+#ifdef VPP_DBG_DIAG_NUM
+ vpp_dbg_show(VPP_DBGLVL_DIAG, 3, "GOVR MIF Err");
+ vpp_dbg_diag_delay = 10;
+#endif
+ }
+ break;
+#endif
+ default:
+ DPRINT("*E* invalid vpp isr\n");
+ break;
+ }
+ return IRQ_HANDLED;
+} /* End of vpp_interrupt_routine */
+
+int vpp_dev_init(void)
+{
+ g_vpp.alloc_framebuf = vpp_alloc_framebuffer;
+ vpp_irqproc_init();
+ vpp_netlink_init();
+ vpp_init();
+#ifdef CONFIG_VPP_PROC
+ /* init system proc */
+ if (vpp_proc_dir == 0) {
+ struct proc_dir_entry *res;
+
+ vpp_proc_dir = proc_mkdir("driver/vpp", NULL);
+ res = create_proc_entry("sts", 0, vpp_proc_dir);
+ if (res)
+ res->read_proc = vpp_sts_read_proc;
+ res = create_proc_entry("reg", 0, vpp_proc_dir);
+ if (res)
+ res->read_proc = vpp_reg_read_proc;
+ vpp_table_header = register_sysctl_table(vpp_root_table);
+ }
+#endif
+
+ /* init interrupt service routine */
+#ifdef WMT_FTBLK_SCL
+ if (vpp_request_irq(VPP_IRQ_SCL, vpp_interrupt_routine,
+ SA_INTERRUPT, "scl", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+#endif
+ if (vpp_request_irq(VPP_IRQ_VPPM, vpp_interrupt_routine,
+ SA_INTERRUPT, "vpp", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+#ifdef WMT_FTBLK_GOVRH
+ if (vpp_request_irq(VPP_IRQ_GOVR, vpp_interrupt_routine,
+ SA_INTERRUPT, "govr", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+
+ if (vpp_request_irq(VPP_IRQ_GOVR2, vpp_interrupt_routine,
+ SA_INTERRUPT, "govr2", (void *)&g_vpp)) {
+ DPRINT("*E* request VPP ISR fail\n");
+ return -1;
+ }
+#endif
+ vpp_switch_state_init();
+ return 0;
+}
+
+int vpp_exit(struct fb_info *info)
+{
+ DBG_MSG("vpp_exit\n");
+
+ vout_exit();
+#ifdef CONFIG_VPP_PROC
+ unregister_sysctl_table(vpp_table_header);
+#endif
+ return 0;
+}
+
+void vpp_wait_vsync(int no, int cnt)
+{
+ int govr_mask = 0;
+
+ if (g_vpp.virtual_display || (g_vpp.dual_display == 0)) {
+ if (govrh_get_MIF_enable(p_govrh))
+ govr_mask |= BIT0;
+ if (govrh_get_MIF_enable(p_govrh2))
+ govr_mask |= BIT1;
+ } else {
+ vout_info_t *vo_info;
+ vout_t *vo;
+ int vo_mask;
+ int i;
+
+ vo_info = vout_info_get_entry(no);
+ vo_mask = vout_get_mask(vo_info);
+ for (i = 0; i < VPP_VOUT_NUM; i++) {
+ if ((vo_mask & (0x1 << i)) == 0)
+ continue;
+ vo = vout_get_entry(i);
+ if (vo) {
+ if (vo->govr == p_govrh)
+ govr_mask |= BIT0;
+ if (vo->govr == p_govrh2)
+ govr_mask |= BIT1;
+ }
+ }
+ }
+
+ if (govr_mask) {
+#if 0
+ if (vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ MSG("[VPP] vpp_wait_vsync %d\n", no);
+#endif
+ vpp_irqproc_work((govr_mask & BIT0) ? VPP_INT_GOVRH_VBIS :
+ VPP_INT_GOVRH2_VBIS, 0, 0, 100 * cnt, cnt);
+ }
+} /* End of vpp_wait_vsync */
+
+/* struct list_head *vpp_modelist; */
+void vpp_get_info(int fbn, struct fb_var_screeninfo *var)
+{
+ static int vpp_init_flag = 1;
+ vout_info_t *info;
+ govrh_mod_t *govr;
+
+ if (vpp_init_flag) {
+#if 0
+ int num;
+
+ INIT_LIST_HEAD(vpp_modelist);
+ for (num = 0; ; num++) {
+ if (vpp_videomode[num].xres == 0)
+ break;
+ }
+ fb_videomode_to_modelist(vpp_videomode, num, vpp_modelist);
+#endif
+ vpp_dev_init();
+ vpp_init_flag = 0;
+ }
+
+ info = vout_info_get_entry(fbn);
+ govr = vout_info_get_govr(fbn);
+ if (govr) {
+ struct fb_videomode vmode;
+
+ govrh_get_framebuffer(govr, &info->fb);
+ govrh_get_videomode(govr, &vmode);
+ fb_videomode_to_var(var, &vmode);
+ } else {
+ var->pixclock =
+ KHZ2PICOS((info->resx * info->resy * 60) / 1000);
+ }
+ var->xres = info->resx;
+ var->yres = info->resy;
+ var->xres_virtual = info->fb.fb_w;
+ var->yres_virtual = var->yres * VPP_MB_ALLOC_NUM;
+ if (g_vpp.mb_colfmt == VDO_COL_FMT_ARGB) {
+ var->bits_per_pixel = 32;
+ var->red.offset = 16;
+ var->red.length = 8;
+ var->red.msb_right = 0;
+ var->green.offset = 8;
+ var->green.length = 8;
+ var->green.msb_right = 0;
+ var->green.offset = 0;
+ var->green.length = 8;
+ var->green.msb_right = 0;
+ var->transp.offset = 24;
+ var->transp.length = 8;
+ var->transp.msb_right = 0;
+ }
+ DBG_MSG("(%d,%dx%d,%dx%d,%d,%d)\n", fbn, var->xres, var->yres,
+ var->xres_virtual, var->yres_virtual,
+ var->pixclock, var->bits_per_pixel);
+#ifdef DEBUG
+ wmtfb_show_var("get_info", var);
+#endif
+}
+
+void vpp_var_to_fb(struct fb_var_screeninfo *var,
+ struct fb_info *info, vdo_framebuf_t *fb)
+{
+ extern unsigned int fb_egl_swap;
+ unsigned int addr;
+ int y_bpp, c_bpp;
+
+ if (var) {
+ fb->col_fmt = WMT_FB_COLFMT(var->nonstd);
+ if (!fb->col_fmt) {
+ switch (var->bits_per_pixel) {
+ case 16:
+ if ((info->var.red.length == 5) &&
+ (info->var.green.length == 6) &&
+ (info->var.blue.length == 5)) {
+ fb->col_fmt = VDO_COL_FMT_RGB_565;
+ } else if ((info->var.red.length == 5) &&
+ (info->var.green.length == 5) &&
+ (info->var.blue.length == 5)) {
+ fb->col_fmt = VDO_COL_FMT_RGB_1555;
+ } else {
+ fb->col_fmt = VDO_COL_FMT_RGB_5551;
+ }
+ break;
+ case 32:
+ fb->col_fmt = VDO_COL_FMT_ARGB;
+ break;
+ default:
+ fb->col_fmt = VDO_COL_FMT_RGB_565;
+ break;
+ }
+ y_bpp = var->bits_per_pixel;
+ c_bpp = 0;
+ } else {
+ y_bpp = 8;
+ c_bpp = 8;
+ }
+ var->xres_virtual = vpp_calc_fb_width(fb->col_fmt, var->xres);
+
+ fb->img_w = var->xres;
+ fb->img_h = var->yres;
+ fb->fb_w = var->xres_virtual;
+ fb->fb_h = var->yres_virtual;
+ fb->h_crop = 0;
+ fb->v_crop = 0;
+ fb->flag = 0;
+ fb->bpp = var->bits_per_pixel;
+
+ addr = info->fix.smem_start +
+ (var->yoffset * var->xres_virtual * ((y_bpp + c_bpp) >> 3));
+ addr += var->xoffset * ((y_bpp) >> 3);
+ fb->y_addr = addr;
+ fb->y_size = var->xres_virtual * var->yres * (y_bpp >> 3);
+ fb->c_addr = fb->y_addr + fb->y_size;
+ fb->c_size = var->xres_virtual * var->yres * (c_bpp >> 3);
+ }
+
+ if (info && (info->node == 0) && (fb_egl_swap != 0)) /* for Android */
+ fb->y_addr = fb_egl_swap;
+
+}
+
+int vpp_pan_display(struct fb_var_screeninfo *var, struct fb_info *info)
+{
+ vout_info_t *vo_info;
+ vdo_framebuf_t *uboot_fb = 0;
+
+ if (g_vpp.hdmi_certify_flag)
+ return 0;
+
+ vo_info = vout_info_get_entry((info) ? info->node : 0);
+
+ DBG_DETAIL("fb %d\n", (info) ? info->node : 0);
+
+ if (wmtfb_fb1_probe == 0) {
+ uboot_fb = kmalloc(sizeof(vdo_framebuf_t), GFP_KERNEL);
+ *uboot_fb = vo_info->fb;
+ uboot_fb->img_h = g_vpp.govrh_init_yres;
+ }
+
+ vpp_var_to_fb(var, info, &vo_info->fb);
+
+ if (uboot_fb) {
+ DMSG("uboot copy fb%d\n", info->node);
+ p_scl->scale_sync = 1;
+ vpp_set_recursive_scale(uboot_fb, &vo_info->fb);
+ kfree(uboot_fb);
+ }
+
+ /* for 8:1:0 mode change resolution garbage frame */
+ if ((g_vpp.virtual_display_mode == 1) && (info->node == 1)) {
+ vdo_framebuf_t fb;
+
+ govrh_get_framebuffer(p_govrh, &fb);
+ vo_info->fb.y_addr = fb.y_addr;
+ vo_info->fb.c_addr = 0;
+ }
+ vout_set_framebuffer(vout_get_mask(vo_info), &vo_info->fb);
+
+ if (g_vpp.fb0_bitblit) {
+ vdo_framebuf_t src, dst;
+ struct fb_videomode vmode;
+ govrh_mod_t *govr;
+
+ src = vo_info->fb;
+ govr = vout_info_get_govr(1);
+ p_scl->scale_sync = 1;
+ govrh_get_framebuffer(govr, &dst);
+ govrh_get_videomode(govr, &vmode);
+ dst.img_w = vmode.xres;
+ dst.fb_w = vpp_calc_fb_width(dst.col_fmt, dst.img_w);
+ dst.img_h = vmode.yres;
+ dst.fb_h = vmode.yres;
+ dst.y_size = dst.fb_w * dst.img_h * dst.bpp / 8;
+#ifdef CONFIG_VPP_DYNAMIC_ALLOC
+ if (g_vpp.mb[0] == 0)
+ vpp_alloc_framebuffer(dst.img_w, dst.img_h);
+#endif
+ g_vpp.stream_mb_index = (g_vpp.stream_mb_index) ? 0 : 1;
+ dst.y_addr = g_vpp.mb[0] + (dst.y_size * g_vpp.stream_mb_index);
+ dst.c_addr = 0;
+ vpp_set_recursive_scale(&src, &dst);
+ vout_set_framebuffer(VPP_VOUT_ALL, &dst);
+ }
+
+#ifdef CONFIG_VPP_STREAM_CAPTURE
+ if (g_vpp.stream_enable) {
+#ifdef CONFIG_VPP_STREAM_ROTATE
+ if (info && (info->node == 1)) {
+ g_vpp.stream_mb_index = var->yoffset / var->yres;
+ vpp_dbg_show_val1(VPP_DBGLVL_STREAM, 0,
+ "stream pan disp", g_vpp.stream_mb_index);
+ }
+#else
+ if ((info && (info->node == 0)) || (info == 0))
+ vpp_mb_scale_bitblit(&vo_info->fb);
+#endif
+ }
+#endif
+ if (vo_info->govr && vpp_check_dbg_level(VPP_DBGLVL_DISPFB)) {
+ char buf[50];
+ unsigned int yaddr, caddr;
+
+ govrh_get_fb_addr(vo_info->govr, &yaddr, &caddr);
+ sprintf(buf, "pan_display %d,%s,0x%x", vo_info->num,
+ vpp_mod_str[vo_info->govr_mod], yaddr);
+ vpp_dbg_show(VPP_DBGLVL_DISPFB, vo_info->num + 1, buf);
+ }
+
+ if (vo_info->govr && vpp_check_dbg_level(VPP_DBGLVL_FPS))
+ vpp_dbg_timer(&vo_info->pandisp_timer,
+ (vo_info->num == 0) ? "fb0" : "fb1", 2);
+ return 0;
+}
+
+int vpp_set_par(struct fb_info *info)
+{
+ vout_info_t *vo_info;
+ unsigned int mask;
+ int i;
+
+ if (g_vpp.hdmi_certify_flag)
+ return 0;
+
+ if ((g_vpp.dual_display == 0) && (info->node == 1))
+ return 0;
+
+ if (g_vpp.govrh_preinit)
+ g_vpp.govrh_preinit = 0;
+
+ vpp_set_mutex(info->node, 1);
+
+for (i = 0; i < VPP_VOUT_INFO_NUM; i++) {
+ vo_info = vout_info_get_entry(i);
+ mask = vout_get_mask(vo_info);
+
+ /* set frame buffer */
+ if (mask) {
+ vdo_framebuf_t fb;
+
+ vpp_var_to_fb(&info->var, info, &fb);
+ vo_info->fb.fb_h = fb.fb_h;
+ if (memcmp(&fb.img_w, &vo_info->fb.img_w, 32)) {
+#ifdef DEBUG
+ DPRINT("[wmtfb_set_par] set_par %d : set framebuf\n",
+ info->node);
+ vpp_show_framebuf("cur", &vo_info->fb);
+ vpp_show_framebuf("new", &fb);
+#endif
+ vo_info->fb = fb;
+ vout_set_framebuffer(mask, &vo_info->fb);
+ }
+ }
+
+ /* set timing */
+ if (vo_info->govr && !(g_vpp.virtual_display && (info->node == 0))) {
+ struct fb_videomode var, cur;
+ govrh_mod_t *govr;
+ unsigned int cur_pixclk, new_pixclk;
+
+ govr = vout_info_get_govr(info->node);
+ fb_var_to_videomode(&var, &info->var);
+ if(g_vpp.virtual_display) {
+ if (vout_find_match_mode(info->node, &var, 1)) {
+ DPRINT("[wmtfb] not support\n");
+ vpp_set_mutex(info->node, 0);
+ return -1;
+ }
+ fb_videomode_to_var(&info->var, &var);
+ var.flag = FB_MODE_IS_FROM_VAR;
+ }
+ govrh_get_videomode(govr, &cur);
+ if ((cur.xres == var.xres) && (cur.yres == var.yres)) {
+ /* diff less than 500K */
+ cur_pixclk = PICOS2KHZ(cur.pixclock);
+ new_pixclk = PICOS2KHZ(var.pixclock);
+ if (abs(new_pixclk - cur_pixclk) < 500) {
+ var.pixclock = cur.pixclock;
+ var.refresh = cur.refresh;
+ }
+ if (abs(var.refresh - cur.refresh) <= 2) /* diff less than 2 */
+ var.refresh = cur.refresh;
+ }
+ if (memcmp(&var, &cur, sizeof(struct fb_videomode))) {
+#ifdef DEBUG
+ DPRINT("[wmtfb] set_par %d: set timing\n", info->node);
+ vpp_show_timing("cur", &cur, 0);
+ vpp_show_timing("new", &var, 0);
+#endif
+ vout_config(mask, vo_info, &var);
+ }
+ }
+}
+ vpp_set_mutex(info->node, 0);
+ return 0;
+}
+
+static int wmtfb_open
+(
+ struct fb_info *info, /*!<; // a pointer point to struct fb_info */
+ int user /*!<; // user space mode */
+)
+{
+ DBG_MSG("Enter wmtfb_open\n");
+ return 0;
+} /* End of wmtfb_open */
+
+static int wmtfb_release
+(
+ struct fb_info *info, /*!<; // a pointer point to struct fb_info */
+ int user /*!<; // user space mode */
+)
+{
+ DBG_MSG("Enter wmtfb_release\n");
+ return 0;
+} /* End of wmtfb_release */
+
+int wmtfb_check_var(struct fb_var_screeninfo *var, struct fb_info *info)
+{
+ int temp;
+ int force = 0;
+
+ DBG_DETAIL("Enter %d\n", info->node);
+
+ var->xres_virtual = vpp_calc_fb_width((var->bits_per_pixel == 16) ?
+ VDO_COL_FMT_RGB_565 : VDO_COL_FMT_ARGB, var->xres);
+
+ if ((info->node == 1) && g_vpp.alloc_framebuf) {
+ if( (g_vpp.mb[0] == 0) || ((var->xres != info->var.xres) ||
+ (var->yres != info->var.yres))) {
+ if (var->nonstd) {
+ g_vpp.mb_colfmt = WMT_FB_COLFMT(var->nonstd);
+ } else {
+ g_vpp.mb_colfmt = (var->bits_per_pixel == 16) ?
+ VDO_COL_FMT_RGB_565 : VDO_COL_FMT_ARGB;
+ }
+ if (g_vpp.virtual_display_mode == 1) {
+#ifdef CONFIG_VPP_DYNAMIC_ALLOC
+ vpp_free_framebuffer();
+#endif
+ } else {
+ if (g_vpp.alloc_framebuf(var->xres, var->yres))
+ return -ENOMEM;
+
+ info->fix.smem_start = g_vpp.mb[0];
+ info->fix.smem_len =
+ g_vpp.mb_fb_size * VPP_MB_ALLOC_NUM;
+ info->screen_base =
+ mb_phys_to_virt(info->fix.smem_start);
+ }
+ }
+ }
+
+ if ((var->xres == 0) || (var->yres == 0))
+ return -1;
+
+ temp = (info->fix.smem_len /
+ (var->xres_virtual * var->yres * (var->bits_per_pixel >> 3)));
+ if (temp < 2) {
+ DBG_MSG("smem_len %d,%d\n", info->fix.smem_len, temp);
+ temp = 2;
+ }
+ if (var->yres_virtual > (var->yres * temp))
+ var->yres_virtual = var->yres * temp;
+
+ /* more than 1M is khz not picos (for ut_vpp) */
+ if (var->pixclock > 1000000) {
+ temp = KHZ2PICOS(var->pixclock / 1000);
+ DBG_MSG("pixclock patch(>1000000)%d-->%d(%d)\n",
+ var->pixclock, temp, (int)PICOS2KHZ(temp) * 1000);
+ var->pixclock = temp;
+ }
+
+ /* less than 100 is fps not picos (for ut_vpp) */
+ if ((var->pixclock > 0) && (var->pixclock < 100)) {
+ unsigned int htotal, vtotal;
+
+ htotal = var->xres + var->right_margin + var->hsync_len +
+ var->left_margin;
+ vtotal = var->yres + var->lower_margin + var->vsync_len +
+ var->upper_margin;
+ temp = htotal * vtotal * var->pixclock;
+ temp = KHZ2PICOS(temp / 1000);
+ DBG_MSG("pixclock patch(<100)%d-->%d(%d)\n",
+ var->pixclock, temp, (int)PICOS2KHZ(temp) * 1000);
+ var->pixclock = temp;
+ }
+
+#ifdef DEBUG_DETAIL
+ wmtfb_show_var("cur var", &info->var);
+ wmtfb_show_var("new var", var);
+#endif
+ switch (var->bits_per_pixel) {
+ case 1:
+ case 8:
+ if (var->red.offset > 8) {
+ var->red.offset = 0;
+ var->red.length = 8;
+ var->green.offset = 0;
+ var->green.length = 8;
+ var->blue.offset = 0;
+ var->blue.length = 8;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ }
+ break;
+ case 16: /* ARGB 1555 */
+ if (var->transp.length) {
+ var->red.offset = 10;
+ var->red.length = 5;
+ var->green.offset = 5;
+ var->green.length = 5;
+ var->blue.offset = 0;
+ var->blue.length = 5;
+ var->transp.offset = 15;
+ var->transp.length = 1;
+ } else { /* RGB 565 */
+ var->red.offset = 11;
+ var->red.length = 5;
+ var->green.offset = 5;
+ var->green.length = 6;
+ var->blue.offset = 0;
+ var->blue.length = 5;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ }
+ break;
+ case 24: /* RGB 888 */
+ case 32: /* ARGB 8888 */
+ var->red.offset = 16;
+ var->red.length = 8;
+ var->green.offset = 8;
+ var->green.length = 8;
+ var->blue.offset = 0;
+ var->blue.length = 8;
+ var->transp.offset = 0;
+ var->transp.length = 0;
+ break;
+ }
+
+ if (g_vpp.fb_manual)
+ return 0;
+
+ if (g_vpp.fb_recheck & (0x1 << info->node)) {
+ force = 1;
+ g_vpp.fb_recheck &= ~(0x1 << info->node);
+ }
+
+ if ((var->xres != info->var.xres) || (var->yres != info->var.yres) ||
+ memcmp(&info->var.pixclock, &var->pixclock, 4 * 9) || force) {
+ struct fb_videomode varfbmode;
+ unsigned int yres_virtual;
+
+ DPRINT("[wmtfb_check_var] fb%d res(%d,%d)->(%d,%d),force %d\n",
+ info->node, info->var.xres, info->var.yres,
+ var->xres, var->yres, force);
+#ifdef DEBUG
+ wmtfb_show_var("cur var", &info->var);
+ wmtfb_show_var("new var", var);
+#endif
+ yres_virtual = var->yres_virtual;
+ fb_var_to_videomode(&varfbmode, var);
+#ifdef DEBUG
+ DPRINT("new fps %d\n", varfbmode.refresh);
+#endif
+ if (vout_find_match_mode(info->node, &varfbmode, 1)) {
+ DPRINT("[wmtfb] not support\n");
+ return -1;
+ }
+ fb_videomode_to_var(var, &varfbmode);
+ var->yres_virtual = yres_virtual;
+#ifdef DEBUG
+ wmtfb_show_var("[wmtfb] time change", var);
+#endif
+ }
+ return 0;
+} /* End of wmtfb_check_var */
+
+static int wmtfb_set_par
+(
+ struct fb_info *info /*!<; // a pointer point to struct fb_info */
+)
+{
+ struct fb_var_screeninfo *var = &info->var;
+
+ DBG_DETAIL("Enter fb%d(%dx%d)\n", info->node, var->xres, var->yres);
+
+ /* init your hardware here */
+ /* ... */
+ if (var->bits_per_pixel == 8)
+ info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
+ else
+ info->fix.visual = FB_VISUAL_TRUECOLOR;
+ info->node = 1;
+ vpp_set_par(info);
+ info->fix.line_length = var->xres_virtual * var->bits_per_pixel / 8;
+ return 0;
+} /* End of vt8430fb_set_par */
+
+static int wmtfb_setcolreg
+(
+ unsigned regno, /*!<; // register no */
+ unsigned red, /*!<; // red color map */
+ unsigned green, /*!<; // green color map */
+ unsigned blue, /*!<; // blue color map */
+ unsigned transp, /*!<; // transp map */
+ struct fb_info *info /*!<; // a pointer point to struct fb_info */
+)
+{
+ return 0;
+
+} /* End of wmtfb_setcolreg */
+
+static int wmtfb_pan_display
+(
+ struct fb_var_screeninfo *var, /*!<; // a pointer fb_var_screeninfo */
+ struct fb_info *info /*!<; // a pointer fb_info */
+)
+{
+ static struct timeval tv1 = {0, 0};
+
+ DBG_DETAIL("Enter wmtfb_pan_display\n");
+
+ vpp_set_mutex(1, 1);
+ if (var->activate & FB_ACTIVATE_VBL) {
+ struct timeval tv2;
+
+ do_gettimeofday(&tv2);
+ if (tv1.tv_sec) {
+ int us;
+
+ us = (tv2.tv_sec == tv1.tv_sec) ?
+ (tv2.tv_usec - tv1.tv_usec) :
+ (1000000 + tv2.tv_usec - tv1.tv_usec);
+ if (us < 16667)
+ vpp_wait_vsync(1, 1);
+ }
+ }
+ vpp_pan_display(var, info);
+ do_gettimeofday(&tv1);
+ vpp_set_mutex(1, 0);
+
+ DBG_DETAIL("Exit wmtfb_pan_display\n");
+ return 0;
+} /* End of wmtfb_pan_display */
+
+static int wmtfb_ioctl
+(
+ struct fb_info *info, /*!<; // a pointer point to struct fb_info */
+ unsigned int cmd, /*!<; // ioctl command */
+ unsigned long arg /*!<; // a argument pointer */
+)
+{
+ int retval = 0;
+
+/* printk("Enter wmtfb_ioctl %x\n",cmd); */
+ if (_IOC_TYPE(cmd) != VPPIO_MAGIC)
+ return retval;
+
+ unlock_fb_info(info);
+ retval = vpp_ioctl(cmd, arg);
+ lock_fb_info(info);
+ return retval;
+} /* End of wmtfb_ioctl */
+
+static int wmtfb_mmap
+(
+ struct fb_info *info, /*!<; // a pointer fb_info */
+ struct vm_area_struct *vma /*!<; // a pointer vm_area_struct */
+)
+{
+ unsigned long off;
+ unsigned long start;
+ u32 len;
+
+ DBGMSG("Enter wmtfb_mmap\n");
+
+ /* frame buffer memory */
+ start = info->fix.smem_start;
+ off = vma->vm_pgoff << PAGE_SHIFT;
+ len = PAGE_ALIGN((start & ~PAGE_MASK) + info->fix.smem_len);
+ if (off >= len) {
+ /* memory mapped io */
+ off -= len;
+ if (info->var.accel_flags)
+ return -EINVAL;
+ start = info->fix.mmio_start;
+ len = PAGE_ALIGN((start & ~PAGE_MASK) + info->fix.mmio_len);
+ }
+
+ start &= PAGE_MASK;
+ if ((vma->vm_end - vma->vm_start + off) > len)
+ return -EINVAL;
+ off += start;
+ vma->vm_pgoff = off >> PAGE_SHIFT;
+ vma->vm_flags |= VM_IO | VM_RESERVED;
+ vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
+ if (remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff,
+ vma->vm_end - vma->vm_start, vma->vm_page_prot))
+ return -EAGAIN;
+ return 0;
+} /* End of wmtfb_mmap */
+
+int wmtfb_hw_cursor(struct fb_info *info, struct fb_cursor *cursor)
+{
+ return 0;
+} /* End of wmtfb_hw_cursor */
+
+int wmtfb_blank(int blank, struct fb_info *info)
+{
+ DBGMSG("(%d,%d)\n", info->node, blank);
+ vpp_set_blank(info, blank);
+ return 0;
+}
+
+/***************************************************************************
+ driver file operations struct define
+****************************************************************************/
+static struct fb_ops wmtfb_ops = {
+ .owner = THIS_MODULE,
+ .fb_open = wmtfb_open,
+ .fb_release = wmtfb_release,
+#if 0
+ .fb_read = wmtfb_read,
+ .fb_write = wmtfb_write,
+#endif
+ .fb_check_var = wmtfb_check_var,
+ .fb_set_par = wmtfb_set_par,
+ .fb_setcolreg = wmtfb_setcolreg,
+ .fb_pan_display = wmtfb_pan_display,
+ .fb_fillrect = cfb_fillrect,
+ .fb_copyarea = cfb_copyarea,
+ .fb_imageblit = cfb_imageblit,
+ .fb_blank = wmtfb_blank,
+ .fb_cursor = wmtfb_hw_cursor,
+ .fb_ioctl = wmtfb_ioctl,
+ .fb_mmap = wmtfb_mmap,
+};
+
+static int __init wmtfb_probe
+(
+ struct platform_device *dev /*!<; // a pointer point to struct device */
+)
+{
+ struct fb_info *info;
+ int cmap_len;
+ u32 map_size;
+ char mode_option[20];
+
+ sprintf(mode_option, "%dx%d@%d",
+ VPP_HD_DISP_RESX, VPP_HD_DISP_RESY, VPP_HD_DISP_FPS);
+
+ /* Dynamically allocate memory for fb_info and par.*/
+ info = framebuffer_alloc(sizeof(u32) * 16, &dev->dev);
+ if (!info) {
+ release_mem_region(info->fix.smem_start, info->fix.smem_len);
+ return -ENOMEM;
+ }
+
+ /* Set default fb_info */
+ info->fbops = &wmtfb_ops;
+ info->fix = wmtfb_fix;
+
+#if 1
+ if (g_vpp.alloc_framebuf) {
+ info->fix.smem_start = g_vpp.mb[0];
+ info->fix.smem_len = g_vpp.mb_fb_size * VPP_MB_ALLOC_NUM;
+ info->screen_base = mb_phys_to_virt(info->fix.smem_start);
+ }
+#endif
+#if 0
+ /* Set video memory */
+ if (!request_mem_region(info->fix.smem_start,
+ info->fix.smem_len, "wmtfb")) {
+ printk(KERN_WARNING
+ "wmtfb: abort, cannot reserve video memory at 0x%lx\n",
+ info->fix.smem_start);
+ }
+
+ info->screen_base = ioremap(info->fix.smem_start, info->fix.smem_len);
+ if (!info->screen_base) {
+ printk(KERN_ERR
+ "wmtfb: abort, cannot ioremap video memory 0x%x @ 0x%lx\n",
+ info->fix.smem_len, info->fix.smem_start);
+ return -EIO;
+ }
+#endif
+ printk(KERN_INFO "wmtfb: framebuffer at 0x%lx, mapped to 0x%p, "
+ "using %d, total %d\n",
+ info->fix.smem_start, info->screen_base,
+ info->fix.smem_len, info->fix.smem_len);
+
+ /*
+ * Do as a normal fbdev does, but allocate a larger memory for GE.
+ */
+ map_size = PAGE_ALIGN(info->fix.smem_len);
+
+ /*
+ * The pseudopalette is an 16-member array for fbcon.
+ */
+ info->pseudo_palette = info->par;
+ info->par = NULL;
+ info->flags = FBINFO_DEFAULT; /* flag for fbcon. */
+
+ /*
+ * This has to been done !!!
+ */
+ cmap_len = 256; /* Be the same as VESA. */
+ fb_alloc_cmap(&info->cmap, cmap_len, 0);
+
+ /*
+ * The following is done in the case of
+ * having hardware with a static mode.
+ */
+ info->var = wmtfb_var;
+ vpp_get_info(1, &info->var);
+
+ /*
+ * This should give a reasonable default video mode.
+ */
+
+ /*
+ * For drivers that can...
+ */
+ wmtfb_check_var(&info->var, info);
+
+ /*
+ * It's safe to allow fbcon to do it for you.
+ * But in this case, we need it here.
+ */
+ wmtfb_set_par(info);
+
+ if (register_framebuffer(info) < 0)
+ return -EINVAL;
+ info->dev->power.async_suspend = 1;
+ printk(KERN_INFO "fb%d: %s frame buffer device\n",
+ info->node, info->fix.id);
+ dev_set_drvdata(&dev->dev, info);
+ wmtfb_fb1_probe = 1;
+ return 0;
+} /* End of wmtfb_probe */
+
+static int wmtfb_remove
+(
+ struct platform_device *dev /*!<; // a pointer point to struct device */
+)
+{
+ struct fb_info *info = dev_get_drvdata(&dev->dev);
+
+ if (info) {
+ unregister_framebuffer(info);
+ fb_dealloc_cmap(&info->cmap);
+ framebuffer_release(info);
+ }
+ return 0;
+}
+
+#ifdef CONFIG_PM
+unsigned int vpp_vout_blank_mask;
+
+static int wmtfb_suspend
+(
+ struct platform_device *pDev, /*!<; // a pointer struct device */
+ pm_message_t state /*!<; // suspend state */
+)
+{
+ vpp_mod_base_t *mod_p;
+ vout_t *vo;
+ int i;
+
+ DBGMSG("Enter wmtfb_suspend\n");
+ vpp_vout_blank_mask = 0;
+ for (i = 0; i <= VPP_VOUT_NUM; i++) {
+ vo = vout_get_entry(i);
+ if (vo && !(vo->status & VPP_VOUT_STS_BLANK))
+ vpp_vout_blank_mask |= (0x1 << i);
+ }
+ vout_set_blank(VPP_VOUT_ALL, VOUT_BLANK_POWERDOWN);
+ if (vout_check_plugin(1))
+ vpp_netlink_notify_plug(VPP_VOUT_ALL ,0);
+ else
+ wmt_set_mmfreq(0);
+
+ /* disable module */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->suspend)
+ mod_p->suspend(0);
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_suspend(0);
+#endif
+ wmt_suspend_mmfreq();
+#ifdef WMT_FTBLK_LVDS
+ lvds_suspend(0);
+#endif
+ /* disable tg */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->suspend)
+ mod_p->suspend(1);
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_suspend(1);
+#endif
+#ifdef WMT_FTBLK_LVDS
+ lvds_suspend(1);
+#endif
+ /* backup registers */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->suspend)
+ mod_p->suspend(2);
+ }
+#ifdef WMT_FTBLK_HDMI
+ hdmi_suspend(2);
+#endif
+#ifdef WMT_FTBLK_LVDS
+ lvds_suspend(2);
+#endif
+#if 0
+ if (lcd_get_lvds_id() == LCD_LVDS_1024x600) {
+ mdelay(5);
+ /* GPIO10 off 8ms -> clock -> off */
+ REG32_VAL(GPIO_BASE_ADDR + 0xC0) &= ~0x400;
+ }
+#endif
+ return 0;
+} /* End of wmtfb_suspend */
+
+static int wmtfb_resume
+(
+ struct platform_device *pDev /*!<; // a pointer struct device */
+)
+{
+ vpp_mod_base_t *mod_p;
+ int i;
+
+#if 0
+ if (lcd_get_lvds_id() == LCD_LVDS_1024x600) {
+ /* GPIO10 6ms -> clock r0.02.04 */
+ REG32_VAL(GPIO_BASE_ADDR+0x80) |= 0x400;
+ REG32_VAL(GPIO_BASE_ADDR+0xC0) |= 0x400;
+ }
+#endif
+
+ DBGMSG("Enter wmtfb_resume\n");
+
+ /* restore registers */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->resume)
+ mod_p->resume(0);
+ }
+#ifdef WMT_FTBLK_LVDS
+ lvds_resume(0);
+#endif
+#ifdef WMT_FTBLK_HDMI
+ hdmi_check_plugin(0);
+ hdmi_resume(0);
+#endif
+ /* enable tg */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->resume)
+ mod_p->resume(1);
+ }
+#ifdef WMT_FTBLK_LVDS
+ lvds_resume(1);
+#endif
+#ifdef WMT_FTBLK_HDMI
+ hdmi_resume(1);
+#endif
+ /* wait */
+ if(!(g_vpp.virtual_display || g_vpp.dual_display == 0))
+ msleep(150);
+
+ /* enable module */
+ for (i = 0; i < VPP_MOD_MAX; i++) {
+ mod_p = vpp_mod_get_base(i);
+ if (mod_p && mod_p->resume)
+ mod_p->resume(2);
+ }
+#ifdef WMT_FTBLK_LVDS
+ lvds_resume(2);
+#endif
+ if (lcd_get_lvds_id() != LCD_LVDS_1024x600)
+ vout_set_blank(vpp_vout_blank_mask, VOUT_BLANK_UNBLANK);
+ wmt_resume_mmfreq();
+ if (vout_check_plugin(0))
+ vpp_netlink_notify_plug(VPP_VOUT_ALL, 1);
+ else {
+ wmt_set_mmfreq(0);
+ }
+
+#ifdef WMT_FTBLK_HDMI
+ hdmi_resume(2);
+#endif
+ return 0;
+} /* End of wmtfb_resume */
+
+static void wmtfb_shutdown
+(
+ struct platform_device *pDev /*!<; // a pointer struct device */
+)
+{
+ DPRINT("wmtfb_shutdown\n");
+ hdmi_set_power_down(1);
+ lvds_set_power_down(1);
+}
+
+#else
+#define wmtfb_suspend NULL
+#define wmtfb_resume NULL
+#define wmtfb_shutdown NULL
+#endif
+
+/***************************************************************************
+ device driver struct define
+****************************************************************************/
+static struct platform_driver wmtfb_driver = {
+ .driver.name = "wmtfb",
+ .driver.bus = &platform_bus_type,
+ .probe = wmtfb_probe,
+ .remove = wmtfb_remove,
+ .suspend = wmtfb_suspend,
+ .resume = wmtfb_resume,
+ .shutdown = wmtfb_shutdown,
+};
+
+/***************************************************************************
+ platform device struct define
+****************************************************************************/
+static u64 wmtfb_dma_mask = 0xffffffffUL;
+static struct platform_device wmtfb_device = {
+ .name = "wmtfb",
+ .dev = {
+ .dma_mask = &wmtfb_dma_mask,
+ .coherent_dma_mask = ~0,
+ .power.async_suspend = 1,
+ },
+
+#if 0
+ .id = 0,
+ .dev = {
+ .release = wmtfb_platform_release,
+ },
+ .num_resources = 0, /* ARRAY_SIZE(wmtfb_resources), */
+ .resource = NULL, /* wmtfb_resources, */
+#endif
+};
+
+static int __init wmtfb_init(void)
+{
+ int ret;
+
+ /*
+ * For kernel boot options (in 'video=wmtfb:<options>' format)
+ */
+ ret = platform_driver_register(&wmtfb_driver);
+ if (!ret) {
+ ret = platform_device_register(&wmtfb_device);
+ if (ret)
+ platform_driver_unregister(&wmtfb_driver);
+ }
+ return ret;
+
+} /* End of wmtfb_init */
+module_init(wmtfb_init);
+
+static void __exit wmtfb_exit(void)
+{
+ printk(KERN_ALERT "Enter wmtfb_exit\n");
+
+ platform_driver_unregister(&wmtfb_driver);
+ platform_device_unregister(&wmtfb_device);
+ return;
+} /* End of wmtfb_exit */
+module_exit(wmtfb_exit);
+
+MODULE_AUTHOR("WonderMedia SW Team");
+MODULE_DESCRIPTION("wmtfb device driver");
+MODULE_LICENSE("GPL");
+/*--------------------End of Function Body -----------------------------------*/
+#undef WMTFB_C