1049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/***************************************************************************/
2049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*                                                                         */
3049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  ftlcdfil.c                                                             */
4049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*                                                                         */
5049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*    FreeType API for color filtering of subpixel bitmap glyphs (body).   */
6049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*                                                                         */
7a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang/*  Copyright 2006, 2008-2010, 2013 by                                     */
8049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  David Turner, Robert Wilhelm, and Werner Lemberg.                      */
9049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*                                                                         */
10049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  This file is part of the FreeType project, and may only be used,       */
11049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  modified, and distributed under the terms of the FreeType project      */
12049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  license, LICENSE.TXT.  By continuing to use, modify, or distribute     */
13049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  this file you indicate that you have read the license and              */
14049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*  understand and accept it fully.                                        */
15049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/*                                                                         */
16049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/***************************************************************************/
17049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
18049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
19049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#include <ft2build.h>
20a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang#include FT_INTERNAL_DEBUG_H
21a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang
22049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#include FT_LCD_FILTER_H
23049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#include FT_IMAGE_H
24049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#include FT_INTERNAL_OBJECTS_H
25049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
26049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
27049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#ifdef FT_CONFIG_OPTION_SUBPIXEL_RENDERING
28049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
29049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/* define USE_LEGACY to implement the legacy filter */
30049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#define  USE_LEGACY
31049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
32049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  /* FIR filter used by the default and light filters */
33049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  static void
34049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  _ft_lcd_filter_fir( FT_Bitmap*      bitmap,
35049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project                      FT_Render_Mode  mode,
36049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project                      FT_Library      library )
37049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  {
38049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_Byte*  weights = library->lcd_weights;
39049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UInt   width   = (FT_UInt)bitmap->width;
40049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UInt   height  = (FT_UInt)bitmap->rows;
41049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
42049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
43049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    /* horizontal in-place FIR filter */
44049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    if ( mode == FT_RENDER_MODE_LCD && width >= 4 )
45049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    {
46049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      FT_Byte*  line = bitmap->buffer;
47049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
48049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
49049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      for ( ; height > 0; height--, line += bitmap->pitch )
50049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      {
51049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_UInt  fir[5];
52049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_UInt  val1, xx;
53049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
54049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
55049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        val1   = line[0];
56049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[0] = weights[2] * val1;
57049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[1] = weights[3] * val1;
58049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[2] = weights[4] * val1;
59049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[3] = 0;
60049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[4] = 0;
61049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
62049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        val1    = line[1];
63049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[0] += weights[1] * val1;
64049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[1] += weights[2] * val1;
65049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[2] += weights[3] * val1;
66049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[3] += weights[4] * val1;
67049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
68049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        for ( xx = 2; xx < width; xx++ )
69049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        {
70049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  val, pix;
71049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
72049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
73049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          val    = line[xx];
74049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix    = fir[0] + weights[0] * val;
75049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[0] = fir[1] + weights[1] * val;
76049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[1] = fir[2] + weights[2] * val;
77049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[2] = fir[3] + weights[3] * val;
78049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[3] =          weights[4] * val;
79049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
80049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix        >>= 8;
81049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix         |= -( pix >> 8 );
82049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          line[xx - 2] = (FT_Byte)pix;
83049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        }
84049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
85049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        {
86049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  pix;
87049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
88049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
89049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix          = fir[0] >> 8;
90049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix         |= -( pix >> 8 );
91049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          line[xx - 2] = (FT_Byte)pix;
92049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
93049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix          = fir[1] >> 8;
94049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix         |= -( pix >> 8 );
95049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          line[xx - 1] = (FT_Byte)pix;
96049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        }
97049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      }
98049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    }
99049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
100049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    /* vertical in-place FIR filter */
101049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    else if ( mode == FT_RENDER_MODE_LCD_V && height >= 4 )
102049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    {
103049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      FT_Byte*  column = bitmap->buffer;
104049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      FT_Int    pitch  = bitmap->pitch;
105049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
106049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
107049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      for ( ; width > 0; width--, column++ )
108049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      {
109049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_Byte*  col = column;
110049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_UInt   fir[5];
111049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_UInt   val1, yy;
112049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
113049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
114049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        val1   = col[0];
115049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[0] = weights[2] * val1;
116049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[1] = weights[3] * val1;
117049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[2] = weights[4] * val1;
118049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[3] = 0;
119049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[4] = 0;
120049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        col   += pitch;
121049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
122049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        val1    = col[0];
123049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[0] += weights[1] * val1;
124049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[1] += weights[2] * val1;
125049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[2] += weights[3] * val1;
126049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        fir[3] += weights[4] * val1;
127049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        col    += pitch;
128049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
129049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        for ( yy = 2; yy < height; yy++ )
130049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        {
131049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  val, pix;
132049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
133049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
134049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          val    = col[0];
135049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix    = fir[0] + weights[0] * val;
136049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[0] = fir[1] + weights[1] * val;
137049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[1] = fir[2] + weights[2] * val;
138049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[2] = fir[3] + weights[3] * val;
139049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          fir[3] =          weights[4] * val;
140049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
141049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix           >>= 8;
142049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix            |= -( pix >> 8 );
143049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col[-2 * pitch] = (FT_Byte)pix;
144049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col            += pitch;
145049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        }
146049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
147049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        {
148049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  pix;
149049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
150049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
151049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix             = fir[0] >> 8;
152049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix            |= -( pix >> 8 );
153049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col[-2 * pitch] = (FT_Byte)pix;
154049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
155049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix         = fir[1] >> 8;
156049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          pix        |= -( pix >> 8 );
157049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col[-pitch] = (FT_Byte)pix;
158049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        }
159049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      }
160049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    }
161049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  }
162049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
163049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
164049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#ifdef USE_LEGACY
165049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
166049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  /* intra-pixel filter used by the legacy filter */
167049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  static void
168049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  _ft_lcd_filter_legacy( FT_Bitmap*      bitmap,
169049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project                         FT_Render_Mode  mode,
170049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project                         FT_Library      library )
171049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  {
172049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UInt  width  = (FT_UInt)bitmap->width;
173049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UInt  height = (FT_UInt)bitmap->rows;
174049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_Int   pitch  = bitmap->pitch;
175049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
176049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    static const int  filters[3][3] =
177049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    {
178049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      { 65538 * 9/13, 65538 * 1/6, 65538 * 1/13 },
179049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      { 65538 * 3/13, 65538 * 4/6, 65538 * 3/13 },
180049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      { 65538 * 1/13, 65538 * 1/6, 65538 * 9/13 }
181049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    };
182049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
183049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UNUSED( library );
184049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
185049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
186049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    /* horizontal in-place intra-pixel filter */
187049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    if ( mode == FT_RENDER_MODE_LCD && width >= 3 )
188049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    {
189049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      FT_Byte*  line = bitmap->buffer;
190049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
191049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
192049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      for ( ; height > 0; height--, line += pitch )
193049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      {
194049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_UInt  xx;
195049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
196049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
197049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        for ( xx = 0; xx < width; xx += 3 )
198049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        {
199049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  r = 0;
200049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  g = 0;
201049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  b = 0;
202049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  p;
203049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
204049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
205049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          p  = line[xx];
206049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          r += filters[0][0] * p;
207049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          g += filters[0][1] * p;
208049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          b += filters[0][2] * p;
209049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
210049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          p  = line[xx + 1];
211049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          r += filters[1][0] * p;
212049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          g += filters[1][1] * p;
213049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          b += filters[1][2] * p;
214049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
215049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          p  = line[xx + 2];
216049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          r += filters[2][0] * p;
217049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          g += filters[2][1] * p;
218049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          b += filters[2][2] * p;
219049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
220049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          line[xx]     = (FT_Byte)( r / 65536 );
221049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          line[xx + 1] = (FT_Byte)( g / 65536 );
222049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          line[xx + 2] = (FT_Byte)( b / 65536 );
223049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        }
224049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      }
225049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    }
226049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    else if ( mode == FT_RENDER_MODE_LCD_V && height >= 3 )
227049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    {
228049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      FT_Byte*  column = bitmap->buffer;
229049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
230049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
231049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      for ( ; width > 0; width--, column++ )
232049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      {
233049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_Byte*  col     = column;
234049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        FT_Byte*  col_end = col + height * pitch;
235049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
236049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
237049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        for ( ; col < col_end; col += 3 * pitch )
238049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        {
239049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  r = 0;
240049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  g = 0;
241049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  b = 0;
242049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          FT_UInt  p;
243049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
244049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
245049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          p  = col[0];
246049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          r += filters[0][0] * p;
247049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          g += filters[0][1] * p;
248049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          b += filters[0][2] * p;
249049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
250049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          p  = col[pitch];
251049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          r += filters[1][0] * p;
252049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          g += filters[1][1] * p;
253049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          b += filters[1][2] * p;
254049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
255049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          p  = col[pitch * 2];
256049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          r += filters[2][0] * p;
257049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          g += filters[2][1] * p;
258049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          b += filters[2][2] * p;
259049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
260049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col[0]         = (FT_Byte)( r / 65536 );
261049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col[pitch]     = (FT_Byte)( g / 65536 );
262049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project          col[2 * pitch] = (FT_Byte)( b / 65536 );
263049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project        }
264049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      }
265049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    }
266049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  }
267049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
268049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#endif /* USE_LEGACY */
269049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
270049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
2710a9d06e2b5cf75c3d6ba958026bfdf4745f576d6The Android Open Source Project  FT_EXPORT_DEF( FT_Error )
272aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  FT_Library_SetLcdFilterWeights( FT_Library      library,
273aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich                                  unsigned char  *weights )
274aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  {
275aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    if ( !library || !weights )
276a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang      return FT_THROW( Invalid_Argument );
277aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
278aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    ft_memcpy( library->lcd_weights, weights, 5 );
279aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
280aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    return FT_Err_Ok;
281aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  }
282aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
283aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
284aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  FT_EXPORT_DEF( FT_Error )
285aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  FT_Library_SetLcdFilter( FT_Library    library,
286aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich                           FT_LcdFilter  filter )
287049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  {
288049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    static const FT_Byte  light_filter[5] =
289aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich                            { 0x00, 0x55, 0x56, 0x55, 0x00 };
290049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    /* the values here sum up to a value larger than 256, */
291049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    /* providing a cheap gamma correction                 */
292049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    static const FT_Byte  default_filter[5] =
293049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project                            { 0x10, 0x40, 0x70, 0x40, 0x10 };
294049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
295049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
296aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    if ( !library )
297a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang      return FT_THROW( Invalid_Argument );
298049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
299049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    switch ( filter )
300049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    {
301049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    case FT_LCD_FILTER_NONE:
302049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_filter_func = NULL;
303049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_extra       = 0;
304049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      break;
305049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
306049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    case FT_LCD_FILTER_DEFAULT:
307049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#if defined( FT_FORCE_LEGACY_LCD_FILTER )
308049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
309049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_filter_func = _ft_lcd_filter_legacy;
310049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_extra       = 0;
311049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
312049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#elif defined( FT_FORCE_LIGHT_LCD_FILTER )
313049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
3140a9d06e2b5cf75c3d6ba958026bfdf4745f576d6The Android Open Source Project      ft_memcpy( library->lcd_weights, light_filter, 5 );
315049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_filter_func = _ft_lcd_filter_fir;
316049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_extra       = 2;
317049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
318049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#else
319049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
3200a9d06e2b5cf75c3d6ba958026bfdf4745f576d6The Android Open Source Project      ft_memcpy( library->lcd_weights, default_filter, 5 );
321049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_filter_func = _ft_lcd_filter_fir;
322049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_extra       = 2;
323049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
324049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#endif
325049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
326049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      break;
327049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
328049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    case FT_LCD_FILTER_LIGHT:
3290a9d06e2b5cf75c3d6ba958026bfdf4745f576d6The Android Open Source Project      ft_memcpy( library->lcd_weights, light_filter, 5 );
330049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_filter_func = _ft_lcd_filter_fir;
331049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_extra       = 2;
332049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      break;
333049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
334049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#ifdef USE_LEGACY
335049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
336049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    case FT_LCD_FILTER_LEGACY:
337049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_filter_func = _ft_lcd_filter_legacy;
338049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      library->lcd_extra       = 0;
339049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project      break;
340049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
341049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#endif
342049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
343049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    default:
344a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang      return FT_THROW( Invalid_Argument );
345049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    }
346049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
347049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    library->lcd_filter = filter;
348aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
349aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    return FT_Err_Ok;
350049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  }
351049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
352049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#else /* !FT_CONFIG_OPTION_SUBPIXEL_RENDERING */
353049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
3540a9d06e2b5cf75c3d6ba958026bfdf4745f576d6The Android Open Source Project  FT_EXPORT_DEF( FT_Error )
355aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  FT_Library_SetLcdFilterWeights( FT_Library      library,
356aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich                                  unsigned char  *weights )
357aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  {
358aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    FT_UNUSED( library );
359aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich    FT_UNUSED( weights );
360aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
361a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang    return FT_THROW( Unimplemented_Feature );
362aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  }
363aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
364aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich
365aacb8e1368a883fcbc9fe64fd0e460cef9c9b20cNick Kralevich  FT_EXPORT_DEF( FT_Error )
366049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  FT_Library_SetLcdFilter( FT_Library    library,
367049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project                           FT_LcdFilter  filter )
368049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  {
369049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UNUSED( library );
370049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project    FT_UNUSED( filter );
371049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
372a2b9955b49034a51dfbc8bf9f4e9d312149cecacXianzhu Wang    return FT_THROW( Unimplemented_Feature );
373049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project  }
374049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
375049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project#endif /* !FT_CONFIG_OPTION_SUBPIXEL_RENDERING */
376049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
377049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project
378049d6fea481044fcc000e7782e5bc7046fc70844The Android Open Source Project/* END */
379