forked from ashchu/ashchu.github.io
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcs184project1.html
260 lines (218 loc) · 22.7 KB
/
cs184project1.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<link rel="preconnect" href="https://fonts.googleapis.com">
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
<link href="https://fonts.googleapis.com/css2?family=Ubuntu:wght@300&display=swap" rel="stylesheet">
<link rel="icon" href="images/sphere_render.png">
<style>
body {
padding: 100px;
width: 1000px;
margin: auto;
text-align: left;
font-weight: 300;
font-family: 'Ubuntu', sans-serif;
color: #121212;
}
h1, h2, h3, h4 {
font-family: 'Source Sans Pro', sans-serif;
}
</style>
<title>CS 184 Rasterizer</title>
<meta http-equiv="content-type" content="text/html; charset=utf-8" />
<link href="https://fonts.googleapis.com/css?family=Open+Sans|Source+Sans+Pro" rel="stylesheet">
</head>
<body>
<img src="images/image1.png" style="margin-left:auto; margin-right:auto; display:block; width: 100px;">
<h1 align="middle">CS 184: Computer Graphics and Imaging, Spring 2022</h1>
<h1 align="middle">Project 1: Rasterizer</h1>
<h2 align="middle">Ashley Chu (SID: 303485876) & Manaal Siddiqui (SID: 3034654585)</h2>
<br><br>
<img src="images/lion.png" align="middle" style="margin-left:auto; margin-right:auto; display:block; width: 400px;" alt="lion.svg">
<figcaption align="middle" style="font-style:italic">Screenshot of Rasterized Lion</figcaption>
<h2 align="middle">Overview</h2>
<!--<p>Give a high-level overview of what you implemented in this project. Think about what you've built as a whole. Share your thoughts on what interesting things you've learned from completing the project.</p>-->
<p>Throughout the course of this project, we learned the basics to rasterization and implemented the material mentioned in lectures 2-5 to rasterize and draw triangles.
Starting from the basics with drawing triangles, we advanced into the process of refining our triangles through supersampling, transforms, and texture mapping using antialiasing.</p>
<h2 align="middle">Section I: Rasterization</h2>
<p style="text-align:center">Drawing, smoothing, and transforming basic single-color triangles. </p>
<h3 align="middle">Part 1: Rasterizing single-color triangles</h3>
<p>To draw a single-color triangle, we implemented basic rasterization sampling at rate = 1. Implementing the `rasterize_triangle()` function, we filled in each pixel in our resolution with a specific given color.
The function iterated over the bounded width and height of the triangles using max_x and min_x of the three points to bound our x-axis and respectively, max_y and min_y of the three points to bound our y-axis. Using two loops and each of the respective bounds, we accessed each relevant, possible point within the frame that could exist in the triangle. Then, we utilized the Point in Triangle test to validate whether each individual pixel should be rasterized.
<p style="font-size: 10px">Note: each pixel is accessed at (x+0.5, y+0.5) to center the pixel. </p>
<br>
The Point-in-Triangle test works by calculating the lines AB, AC, and BC between points A, B, and C and checking if the given pixel (x+.5,y+.5) exists inside, outside, or on the given lines.
Using the equation, L(x,y) = -(x - X_1) * (Y_1 - Y_0) + (y - Y_1) * (X_1 - X_0):
<ul>
<li>if L(x+.5,y+.5) = 0 -> (x+.5,y+.5) lies on one of the lines of the triangle</li>
<li>if L(x+.5,y+.5) < 0 -> (x+.5,y+.5) lies inside the line </li>
<li>if L(x+.5,y+.5) > 0 -> (x+.5,y+.5) lies above the line</li>
</ul>
<br>
Generally, the purpose of this test is to check if a point lies on a line. In our case, if (x+.5,y+.5) lies on the line or within the triangle, we want to rasterize that point. So, all points (x+.5,y+.5) that are <= AB, BC, and AC, should be rasterized. However, this assumes we're evaluating the point counterclockwise. To account for a clockwise evaluation of the points, we also check whether a point (x+.5,y+.5) exists >= 0.
When a given location (x+.5, y+.5) exists within the triangle (as determined by the Point-in-Triangle test), the samplebuffer is filled at point (x,y) with the given input color using the `fill_pixel()` helper method.
<br>
<br>
By using the min_x, max_x, min_y, and max_y, our raster_triangle() function is an improvement from simply iterating over the bounding box of the triangle. Instead of iterating over every possible point in the bounding box, our start and end points are relevant to the height and width of the triangle. By this, in the worst case, our algoritm would be no worse than checking every pixel in the bounding box as we iterate throuh all x's and y's respective to the width and height.
</p>
<!-- In-depth process here: Loops, using min_x, Max_x and min_y, max_y -->
<div align="middle">
<table style="width=100%">
<tr>
<td>
<img src="images/basic_test4_0.png" align="middle" style="width:400px;" alt="basic/test4.svg">
<figcaption align="middle" style="font-style:italic">test4.svg is fully colored, but shows jaggies</figcaption>
</td>
<td>
<img src="images/basic_test4_1.png" align="middle" style="width:400px;" alt="basic/test4.svg">
<figcaption align="middle" style="font-style:italic">Red jaggies in test4.svg</figcaption>
</td>
</tr>
<br>
<tr>
<td>
<img src="images/basic_test_4_2.png" align="middle" style="width:400px;" alt="basic/test4.svg">
<figcaption align="middle" style="font-style:italic">Green jaggies in test4.svg</figcaption>
</td>
</tr>
</table>
</div>
<h3 align="middle">Part 2: Antialiasing triangles</h3>
<p>Yay, we successfully rasterized basic triangles! However, there seems to be an issue with jaggies in our rasterized images. To fix this, we used a process called supersampling, an antialiasing method, in which we use subpixels to help sample our image and refine our color output. The possible sampling rates are 1, 4, 9, and 16 supersamples/pixel.</p>
<h4 style="font-style:italic">The Algorithm</h4>
<p> To begin the supersampling process, we began by implementing small changes to help our buffer memory and helper functions adjust to the respective sampling rate inputted by `sample_rate` parameter from DrawRend. Then, to implement the supersampling functionality, we needed to adjust our rasterize_triangle(), fill_pixel(), and resolve_to_frame_buffer() functions.</p>
<p>A large part of this process relied on our implementation of rasterize_triangle() and resolve_to_framebuffer() in which we used our sample_buffer as a helper buffer to connect to our final framebuffer_target output.
To start, we largely build on our implementation from Task 1 in which we use 2 for loops over the width and height of the triangle we want to rasterize in rasterize_triangle().
</p>
<p>Instead of simply iterating though, we created 2 additional for loops to subsample and access the subpixel information of each pixel accessed in the outer for loops. One way to think about it is: In our first outer for-loops, we're accessing the current image space by indexing at each relevant pixel, and the purpose of our inner two for-loops is to help access the space within the pixel, allowing us to develop more detailed information about the color space and how the pixel should be represented.</p>
<p>In Task 1, we simply supersampled at a rate of 1, the pixel itself. To factor in sampling at a rate > 1, we created two inner loops with a lower bound of 0 and an upperbound constraint of sqrt(sample_rate). By this, we can sample NxN time "inside" each pixel, with N = sqrt(sample_rate). Inside each pixel, supersampling allows us to step horizontally along the x-axis and vertically along the y-axis by sqrt(sample_rate). For example, if we supersampled at 4 per pixel, we'd step horizontally by 2 and vertically by 2 to access the 4 quadrants of the pixel. By implementing NxN sampling rather than being constrained to 1x1 sampling per pixel, we also need to shift the center point for each subpixel accessed. Our calculations estimated the shift to be current(x,y) + float((0.5 + (current_step(x,y)) / sqrt(sample_rate)) to account for the substep in x, y, and sampling rate. Additionally, our indexing to the sample_buffer and frame_buffer changed to `(y * sqrt(sampling_rate) + ystep) * width * lerate + x * lerate + xstep` to account for the offset and sampling rate for each subpixel.
</p>
<p>Woohoo! Now that we can access each subpixel, we can give a more accurate representation on how to input color into which respective subpixel to represent the pixel better. Our resolution is still the same though, so now we need to take our subpixels and average them for our outputted pixel color value which is implemented in resolve_to_frame_buffer(). </p>
<h4 style="font-style:italic">Why Supersample?</h4>
<p>Supersampling is helpful because it helps lessen the impact and possibility of jaggies creating a jarring image as seen in the images from task 1. This antialiasing method gives us the additional information needed to measure out how our pixels should be outputted. Instead of creating jagggies, supersampling allows us to measure whether color should remain the same or be lessened to smooth the image out. By using supersampling, we can average the subpixel values measure to create a gradient of colors rather than have stark color contrasts and as a result, jaggies. In other words, we're removing the higher frequencies color changes across all the pixels to blend the image together better.</p>
<p>However, at the same time, supersampling can be expensive as it creates additional calculations and can take large computations or be very costly for large images or resolutions.</p>
<p>Below, you can see the result of supersampling over various supersampling rates. At rate 1, the jaggies and high frequency colors are sharp. At rate 4, the edges begin slowly smoothing out a bit. And when we reach rate 16, the triangle edges seem to smoothen out completely.</p>
<div align="middle">
<table style="width:100%">
<tr>
<td>
<img src="images/basic_1.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 1">
<figcaption align="middle" style="font-style:italic">Supersampling with rate 1/pixel</figcaption>
</td>
<td>
<img src="images/basic_4.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 4">
<figcaption align="middle" style="font-style:italic">Supersampling with rate 4/pixel</figcaption>
</td>
</tr>
<br>
<tr>
<td>
<img src="images/basic_9.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 9">
<figcaption align="middle" style="font-style:italic">Supersampling with rate 9/pixel</figcaption>
</td>
<td>
<img src="images/basic_16.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 16">
<figcaption align="middle" style="font-style:italic">Supersampling with rate 16/pixel</figcaption>
</td>
</tr>
</table>
</div>
<h3 align="middle">Part 3: Transforms</h3>
<h4 style="font-weight: 100">IMPORTANT: Apologies – instead of calling our robot, "my_robot.svg", our robot is named "robot_transf.svg"!</h4>
<p>To implement transforms, we created functions for basic matrix transformations that calculated translation, scaling, and rotation.</p>
<img src="images/robot_8.png" align="middle" style="margin-left:auto; margin-right:auto; display:block; width: 400px;" alt="robot_8.svg">
<figcaption align="middle" style="font-style:italic">Robot making the number 8!</figcaption>
<p>8-Bot was formed by implementing a series of rotations and editing a few translations. We wanted the robot to do a fun dance, so we first played with their legs by rotating the top half of their legs outwards by about 45 degrees and the bottom half of their legs inwards by about 45 degrees respectively. Similarly, we followed the same process as the legs for the robot's arms, except reversed. The bottom half of the arms are rotated slightly outwards by 45 degrees while the top half is rotated slightly inwards by 45 degrees. To account for the spacing changes caused by rotation, we translated the bottom and top half of the arms and the legs slight by plus or minus 20 units.</p>
<h2 align="middle">Section II: Sampling</h2>
<h3 align="middle">Part 4: Barycentric coordinates</h3>
<p>Barycentric coordinates are extremely useful as another way to manipulate and express the position of any point located on a triangle with three scalar values. In other words, barycentric coordinates are an areal coordinate system in which we can interpolate across the points in a triangle. The ability to interpolate across the triangle range from properties such as color, position, and texture coordinates, and are essentially another way to represent a triangle.</p>
<p>To calculate barycentric coordinates, the coordinate is fairly simple using our three triangle verticies A (x0,y0), B (x1,y1), and C (x2,y2). Using these values, we can calculate the corresponding weights: alpha, beta, and gamma. The relationship of alpha, beta, and gamma is shown by the equation alpha + beta + gamma = 1. To calculate alpha, beta, and gamma, we can find alpha and beta using a line interpolation and specifically the equation derived in lecture 2:
<br>
<br>
alpha = Lbc(x,y)/Lbc(x_A, y_A) <br> beta = Lca(x,y)/Lca(x_B,y_B) <br> gamma = 1 - alpha - beta <br> <br> From these steps, we can calculate the barycentric coordinates through <br> (x,y) = alpha*(x0, y0) + beta*(x1, y1) + gamma*(x2, y2)</p>
<p>With these scalar values, we can represent the color values interpolated by the triangle.
<br>
To visualize this, view the triangle below. Each of the three verticies forming the triangle are a different RGB value which we can represent with P0, P1, P2. P0 (left) is red, P1 (right) is blue, and P2 (top) is green. Using the barycentric coordinates, we interpolated the triangle, and thus, you can see the resulting output example shown.</p>
<img src="images/colorful_triangle.png" align="middle" style="margin-left:auto; margin-right:auto; display:block; width: 600px;" alt="colorful triangle">
<figcaption align="middle" style="font-style:italic">Triangle interpolated using barycentric coordinates.</figcaption>
<p>An additional example of test7 is shown below to show another version of interpolation through barycentric coordinates.</p>
<img src="images/test7.png" align="middle" style="margin-left:auto; margin-right:auto; display:block; width: 600px;" alt="test7.svg">
<figcaption align="middle" style="font-style:italic">Using barycentric coordinates to interpolate color in test7 (at sample rate 1)!</figcaption>
<h3 align="middle">Part 5: "Pixel sampling" for texture mapping</h3>
<p>Generally, pixel sampling refers to sampling specific sample points from an image at a specified rate, and the more you sample (the higher the rate), the smoother the output since it averages colors more frequently for a better color representation. </p>
<p>In regards to texture mapping, pixel sampling refers to evaluating a sample points in order to get the corresponding (u,v) coordinate in the texel space. If there's a texel point for the (x,y) point, we calculate the subpixel's barycentric coordinates to get its position within the triangle and then we use the texture triangle's vertices to get the subpixels (u,v) location. </p>
<p>Depending on the user input, we sample nearest or bilinearly with a default mipmap level of 0. In nearest pixel sampling, we set the value of the subpixel to the color of the closest texel by the (u,v) point in texel space. We have to index into the texel array, but be mindful that its 1-dimensional, so we have to scale by the width and the height and multiply by 3 for both x and y to account for the RGB representation. After getting that color, we set the subpixel’s color as such. Then, in bilinear sampling, we sample the colors of the four closest texels to our current (u,v). We get the colors the same way we described earlier and linearly interpolate and get the weighted average of the four points. </p>
<div align="middle">
<table style="width=100%">
<tr>
<td>
<img src="images/nearest pixel 1.png" align="middle" style="width:400px;" alt="Bilinear, Supersample rate = 1">
<figcaption align="middle" style="font-style:italic">Bilinear sampling with a rate 1</figcaption>
</td>
<td>
<img src="images/nearest 16.png" align="middle" style="width:400px;" alt="Bilinear, Supersample rate = 16">
<figcaption align="middle" style="font-style:italic">Bilinear sampling with a rate 16</figcaption>
</td>
</tr>
<br>
<tr>
<td>
<img src="images/bilinear 1.png" align="middle" style="width:400px;" alt="Nearest, Supersample rate = 1">
<figcaption align="middle" style="font-style:italic">Nearest Sampling with a rate 1</figcaption>
</td>
<td>
<img src="images/bilinear 16.png" align="middle" style="width:400px;" alt="Nearest, Supersample rate = 16">
<figcaption align="middle" style="font-style:italic">Nearest Sampling with a rate 16</figcaption>
</td>
</tr>
</table>
</div>
<p>As we see, increasing the sampling rate generally increases the accuracy of the color. However, nearest sampling hardly gets the white line if at all, because it’s a thin and high-frequency line of pixels. Bilinear sampling helps with aliasing and frequency effects, and is able to catch the white line since it samples the four nearest pixels and averages them out. That said, it also seems to generally blur more as well. Bilinear interpolation will perform better in terms of aliasing artifacts and continuous data, since output cells are calculated based on the relative position of the four nearest values from the input grid. Nearest neighbor sampling doesn't fare as well with high-frequency color points but is faster and preserves original color values.</p>
<h3 align="middle">Part 6: "Level sampling" with mipmaps for texture mapping</h3>
<p>Level sampling is using specific samples from a series of "levels" of an image, each subsequent level having lower and lower resolutions with a factor of 2. Level sampling means sampling from between different mipmaps levels to provide different levels of detail depending on distance between the object and the viewpoint. For example, in portrait mode on a phone, you are using lower resolution for the background and higher resolution for your face.
</p>
<p>In order to implement this, we built on our code from task 5. We created a params struct to pass into the tex.sample() method to streamline which sampling methods are used. We find the uv coordinates of (x,y+1), (x+1, y) and then use barycentric coordinates to obtain the differentials, which allow us to determine the distance away from the viewpoint. These differential/distance vectors (p_dx_uv and p_dy_uv are scaled to the appropriate texture dimensions for width and height, and then we use the equation from class for level = max(sqrt((du_dx * du_dx)+(dv_dx * dv_dx)),sqrt((du_dy * du_dy)+(dv_dy * dv_dy))). The effect of this is to produce a high level if the distance of the point in texture space is large. If it's small, then the mipmap level is also small. However, these calculations return a float. Depending on the user input, we either use trilinear sampling or nearest level sampling. If the user specified the nearest level, we just index into that level using the same procedure from task 5 of indexing into the texel array. If the user specifies bilinear level sampling, we interpolate and take weighted averages between the lower and upper level, and we can do so with either nearest pixel or bilinear pixel sampling, with the logic from the previous task at the corresponding level(s).
</p>
<p> Level 0 or nearest level with nearest sampling is the fastest. Nearest pixel sampling is quick, simple, and preserves color values as they originally appeared. However, nearest pixel sampling and nearest level sampling also produce more antialiasing than the others. Bilinear pixel sampling is better, albeit slower. It’s better because it takes weighted averages of nearby pixels and in doing so makes the quality smoother and accurate. It’s especially useful for continuous data. Still, it’s slower because of increased rates of calculation and can cause blurring and displacement of colors. Supersampling makes a large difference, as, one would think, it samples more points and therefore has more data, but it also adds another performance barrier in regards to both memory and speed since you loop through sample_rate number of times. Compared to nearest level, bilinear level sampling fares much better – although both scale blurriness with distance from the viewpoint, bilinear/trilinear level sampling is more accurate and deals better with antialiasing but it is slowest because you have to sample across layers and make a lot of calculations over nested loops. Overall, the best quality and worst performance is produced by trilinear level sampling with bilinear pixel sampling. The best performance and worst quality is produced by nearest pixel sampling at level 0.
</p>
<div align="middle">
<table style="width:100%">
<tr>
<td>
<img src="images/et1.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 1">
<figcaption align="middle" style="font-style:italic">L0, PNearest</figcaption>
</td>
<td>
<img src="images/et2.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 4">
<figcaption align="middle" style="font-style:italic">L0, PLinear</figcaption>
</td>
</tr>
<br>
<tr>
<td>
<img src="images/et3.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 9">
<figcaption align="middle" style="font-style:italic">LNearest, PLinear</figcaption>
</td>
<td>
<img src="images/et4.png" align="middle" style="width:400px;" alt="Test 4, Supersample rate = 16">
<figcaption align="middle" style="font-style:italic">LNearest, PNearest</figcaption>
</td>
</tr>
</table>
</div>
<h3 align="middle">Website Link</h3>
<p>You can view our website at: <a href="https://ashchu.github.io/cs184project1">https://ashchu.github.io/cs184project1</a></p>
<h2 align="middle">Section III: Art Competition</h2>
<p>If you are not participating in the optional art competition, don't worry about this section!</p>
<h3 align="middle">Part 7: Draw something interesting!</h3>
<p>Ah! we didn't make it to this part :(</p>
<h2>Concluding Notes + Reflection + Snapshots!</h2>
<p style="font-style: italic">General Key Learnings</p>
<ul>
<li>Check use of ints and floats</li>
<li>Check 0-indexed!</li>
<li>Make sure that the way you index into the sample buffer is consistent with the rest of the code</li>
</ul>
</body>
</html>