1*** General notes about rounding 2 3Suppose a function is sampled at positions [k + o] where k is an 4integer and o is a fractional offset 0 <= o < 1. 5 6To round a value to the nearest sample, breaking ties by rounding up, 7we can do this: 8 9 round(x) = floor(x - o + 0.5) + o 10 11That is, first subtract o to let us pretend that the samples are at 12integer coordinates, then add 0.5 and floor to round to nearest 13integer, then add the offset back in. 14 15To break ties by rounding down: 16 17 round(x) = ceil(x - o - 0.5) + o 18 19or if we have an epsilon value: 20 21 round(x) = floor(x - o + 0.5 - e) + o 22 23To always round *up* to the next sample: 24 25 round_up(x) = ceil(x - o) + o 26 27To always round *down* to the previous sample: 28 29 round_down(x) = floor(x - o) + o 30 31If a set of samples is stored in an array, you get from the sample 32position to an index by subtracting the position of the first sample 33in the array: 34 35 index(s) = s - first_sample 36 37 38*** Application to pixman 39 40In pixman, images are sampled with o = 0.5, that is, pixels are 41located midways between integers. We usually break ties by rounding 42down (i.e., "round towards north-west"). 43 44 45-- NEAREST filtering: 46 47The NEAREST filter simply picks the closest pixel to the given 48position: 49 50 round(x) = floor(x - 0.5 + 0.5 - e) + 0.5 = floor (x - e) + 0.5 51 52The first sample of a pixman image has position 0.5, so to find the 53index in the pixel array, we have to subtract 0.5: 54 55 floor (x - e) + 0.5 - 0.5 = floor (x - e). 56 57Therefore a 16.16 fixed-point image location is turned into a pixel 58value with NEAREST filtering by doing this: 59 60 pixels[((y - e) >> 16) * stride + ((x - e) >> 16)] 61 62where stride is the number of pixels allocated per scanline and e = 630x0001. 64 65 66-- CONVOLUTION filtering: 67 68A convolution matrix is considered a sampling of a function f at 69values surrounding 0. For example, this convolution matrix: 70 71 [a, b, c, d] 72 73is interpreted as the values of a function f: 74 75 a = f(-1.5) 76 b = f(-0.5) 77 c = f(0.5) 78 d = f(1.5) 79 80The sample offset in this case is o = 0.5 and the first sample has 81position s0 = -1.5. If the matrix is: 82 83 [a, b, c, d, e] 84 85the sample offset is o = 0 and the first sample has position s0 = 86-2.0. In general we have 87 88 s0 = (- width / 2.0 + 0.5). 89 90and 91 92 o = frac (s0) 93 94To evaluate f at a position between the samples, we round to the 95closest sample, and then we subtract the position of the first sample 96to get the index in the matrix: 97 98 f(t) = matrix[floor(t - o + 0.5) + o - s0] 99 100Note that in this case we break ties by rounding up. 101 102If we write s0 = m + o, where m is an integer, this is equivalent to 103 104 f(t) = matrix[floor(t - o + 0.5) + o - (m + o)] 105 = matrix[floor(t - o + 0.5 - m) + o - o] 106 = matrix[floor(t - s0 + 0.5)] 107 108The convolution filter in pixman positions f such that 0 aligns with 109the given position x. For a given pixel x0 in the image, the closest 110sample of f is then computed by taking (x - x0) and rounding that to 111the closest index: 112 113 i = floor ((x0 - x) - s0 + 0.5) 114 115To perform the convolution, we have to find the first pixel x0 whose 116corresponding sample has index 0. We can write x0 = k + 0.5, where k 117is an integer: 118 119 0 = floor(k + 0.5 - x - s0 + 0.5) 120 121 = k + floor(1 - x - s0) 122 123 = k - ceil(x + s0 - 1) 124 125 = k - floor(x + s0 - e) 126 127 = k - floor(x - (width - 1) / 2.0 - e) 128 129And so the final formula for the index k of x0 in the image is: 130 131 k = floor(x - (width - 1) / 2.0 - e) 132 133Computing the result is then simply a matter of convolving all the 134pixels starting at k with all the samples in the matrix. 135 136 137--- SEPARABLE_CONVOLUTION 138 139For this filter, x is first rounded to one of n regularly spaced 140subpixel positions. This subpixel position determines which of n 141convolution matrices is being used. 142 143Then, as in a regular convolution filter, the first pixel to be used 144is determined: 145 146 k = floor (x - (width - 1) / 2.0 - e) 147 148and then the image pixels starting there are convolved with the chosen 149matrix. If we write x = xi + frac, where xi is an integer, we get 150 151 k = xi + floor (frac - (width - 1) / 2.0 - e) 152 153so the location of k relative to x is given by: 154 155 (k + 0.5 - x) = xi + floor (frac - (width - 1) / 2.0 - e) + 0.5 - x 156 157 = floor (frac - (width - 1) / 2.0 - e) + 0.5 - frac 158 159which means the contents of the matrix corresponding to (frac) should 160contain width samplings of the function, with the first sample at: 161 162 floor (frac - (width - 1) / 2.0 - e) + 0.5 - frac 163 164This filter is called separable because each of the k x k convolution 165matrices is specified with two k-wide vectors, one for each dimension, 166where each entry in the matrix is computed as the product of the 167corresponding entries in the vectors. 168