1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS 4 * FILE: dll/opengl/opengl32/swimpl.c 5 * PURPOSE: OpenGL32 DLL, opengl software implementation 6 */ 7 8 #include "opengl32.h" 9 10 /* MESA includes */ 11 #include <context.h> 12 #include <matrix.h> 13 14 WINE_DEFAULT_DEBUG_CHANNEL(opengl32); 15 16 #define WIDTH_BYTES_ALIGN32(cx, bpp) ((((cx) * (bpp) + 31) & ~31) >> 3) 17 #define WIDTH_BYTES_ALIGN16(cx, bpp) ((((cx) * (bpp) + 15) & ~15) >> 3) 18 19 /* Flags for our pixel formats */ 20 #define SB_FLAGS (PFD_DRAW_TO_BITMAP | PFD_SUPPORT_GDI | PFD_SUPPORT_OPENGL | PFD_GENERIC_FORMAT) 21 #define SB_FLAGS_WINDOW (PFD_DRAW_TO_BITMAP | PFD_SUPPORT_GDI | PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_GENERIC_FORMAT) 22 #define SB_FLAGS_PALETTE (PFD_DRAW_TO_BITMAP | PFD_SUPPORT_GDI | PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_GENERIC_FORMAT | PFD_NEED_PALETTE) 23 #define DB_FLAGS (PFD_DOUBLEBUFFER | PFD_SWAP_COPY | PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_GENERIC_FORMAT) 24 #define DB_FLAGS_PALETTE (PFD_DOUBLEBUFFER | PFD_SWAP_COPY | PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_GENERIC_FORMAT | PFD_NEED_PALETTE) 25 26 27 struct pixel_format 28 { 29 DWORD dwFlags; 30 BYTE iPixelType; 31 BYTE cColorBits; 32 BYTE cRedBits; BYTE cRedShift; 33 BYTE cGreenBits; BYTE cGreenShift; 34 BYTE cBlueBits; BYTE cBlueShift; 35 BYTE cAlphaBits; BYTE cAlphaShift; 36 BYTE cAccumBits; 37 BYTE cAccumRedBits; BYTE cAccumGreenBits; BYTE cAccumBlueBits; BYTE cAccumAlphaBits; 38 BYTE cDepthBits; 39 }; 40 41 static const struct pixel_format pixel_formats_32[] = 42 { 43 /* 32bpp */ 44 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 45 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 46 {DB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 47 {DB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 48 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 49 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 50 {DB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 51 {DB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 52 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 53 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 54 {DB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 55 {DB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 56 /* 24bpp */ 57 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 58 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 59 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 60 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 61 {SB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 62 {SB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 63 /* 16 bpp */ 64 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 32}, 65 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 16}, 66 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 32}, 67 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 16}, 68 {SB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 10, 5, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 69 {SB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 10, 5, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 70 /* 8bpp */ 71 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 32}, 72 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 16}, 73 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 32}, 74 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 16}, 75 {SB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 32}, 76 {SB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 16}, 77 /* 4bpp */ 78 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 32}, 79 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 16}, 80 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 32}, 81 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 16}, 82 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 32}, 83 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 16}, 84 }; 85 86 static const struct pixel_format pixel_formats_24[] = 87 { 88 /* 24bpp */ 89 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 90 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 91 {DB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 92 {DB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 93 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 94 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 95 {DB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 96 {DB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 97 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 98 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 99 {DB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 100 {DB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 101 /* 32bpp */ 102 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 103 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 104 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 105 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 106 {SB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 107 {SB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 108 /* 16 bpp */ 109 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 32}, 110 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 16}, 111 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 32}, 112 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 16}, 113 {SB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 10, 5, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 114 {SB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 10, 5, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 115 /* 8bpp */ 116 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 32}, 117 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 16}, 118 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 32}, 119 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 16}, 120 {SB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 32}, 121 {SB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 16}, 122 /* 4bpp */ 123 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 32}, 124 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 16}, 125 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 32}, 126 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 16}, 127 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 32}, 128 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 16}, 129 }; 130 131 static const struct pixel_format pixel_formats_16[] = 132 { 133 /* 16 bpp - 565 */ 134 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 32}, 135 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 16}, 136 {DB_FLAGS, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 32}, 137 {DB_FLAGS, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 16}, 138 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 32}, 139 {SB_FLAGS_WINDOW, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 16}, 140 {DB_FLAGS, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 32}, 141 {DB_FLAGS, PFD_TYPE_RGBA, 16, 5, 11, 6, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 16}, 142 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 16, 5, 11, 6, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 143 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 16, 5, 11, 6, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 144 {DB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 11, 6, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 145 {DB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 11, 6, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 146 /* 24bpp */ 147 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 148 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 149 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 150 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 151 {SB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 152 {SB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 153 /* 32bpp */ 154 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 155 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 156 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 157 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 158 {SB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 159 {SB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 160 /* 8bpp */ 161 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 32}, 162 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 16}, 163 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 32}, 164 {SB_FLAGS, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 16}, 165 {SB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 32}, 166 {SB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 16}, 167 /* 4bpp */ 168 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 32}, 169 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 16}, 170 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 32}, 171 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 16}, 172 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 32}, 173 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 16}, 174 }; 175 176 static const struct pixel_format pixel_formats_8[] = 177 { 178 /* 8bpp */ 179 {SB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 32}, 180 {SB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 16}, 181 {DB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 32}, 182 {DB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 0, 0, 32, 11, 11, 10, 0, 16}, 183 {SB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 32}, 184 {SB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 16}, 185 {DB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 32}, 186 {DB_FLAGS_PALETTE, PFD_TYPE_RGBA, 8, 3, 0, 3, 3, 2, 6, 8, 0, 32, 8, 8, 8, 8, 16}, 187 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 32}, 188 {SB_FLAGS_WINDOW, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 16}, 189 {DB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 32}, 190 {DB_FLAGS, PFD_TYPE_COLORINDEX, 8, 3, 0, 3, 3, 2, 6, 0, 0, 0, 0, 0, 0, 0, 16}, 191 /* 24bpp */ 192 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 193 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 194 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 195 {SB_FLAGS, PFD_TYPE_RGBA, 24, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 196 {SB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 197 {SB_FLAGS, PFD_TYPE_COLORINDEX, 24, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 198 /* 32bpp */ 199 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 32}, 200 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 0, 0, 64, 16, 16, 16, 0, 16}, 201 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 32}, 202 {SB_FLAGS, PFD_TYPE_RGBA, 32, 8, 16, 8, 8, 8, 0, 8, 0, 64, 16, 16, 16, 16, 16}, 203 {SB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 204 {SB_FLAGS, PFD_TYPE_COLORINDEX, 32, 8, 16, 8, 8, 8, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 205 /* 16 bpp */ 206 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 32}, 207 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 0, 0, 32, 11, 11, 10, 0, 16}, 208 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 32}, 209 {SB_FLAGS, PFD_TYPE_RGBA, 16, 5, 10, 5, 5, 5, 0, 8, 0, 32, 8, 8, 8, 8, 16}, 210 {SB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 10, 5, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 32}, 211 {SB_FLAGS, PFD_TYPE_COLORINDEX, 16, 5, 10, 5, 5, 5, 0, 0, 0, 0, 0, 0, 0, 0, 16}, 212 /* 4bpp */ 213 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 32}, 214 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 0, 0, 16, 5, 6, 5, 0, 16}, 215 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 32}, 216 {SB_FLAGS, PFD_TYPE_RGBA, 4, 1, 0, 1, 1, 1, 2, 8, 0, 16, 4, 4, 4, 4, 16}, 217 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 32}, 218 {SB_FLAGS, PFD_TYPE_COLORINDEX, 4, 1, 0, 1, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 16}, 219 }; 220 221 struct sw_framebuffer 222 { 223 GLvisual *gl_visual; /* Describes the buffers */ 224 GLframebuffer *gl_buffer; /* Depth, stencil, accum, etc buffers */ 225 226 const struct pixel_format* pixel_format; 227 HDC Hdc; 228 229 /* Current width/height */ 230 GLuint width; GLuint height; 231 232 /* BackBuffer, if any */ 233 BYTE* BackBuffer; 234 }; 235 236 struct sw_context 237 { 238 GLcontext *gl_ctx; /* The core GL/Mesa context */ 239 240 /* This is to keep track of the size of the front buffer */ 241 HHOOK hook; 242 243 /* Our frame buffer*/ 244 struct sw_framebuffer* fb; 245 246 /* State variables */ 247 union 248 { 249 struct 250 { 251 BYTE ClearColor; 252 BYTE CurrentColor; 253 } u8; 254 struct 255 { 256 USHORT ClearColor; 257 USHORT CurrentColor; 258 } u16; 259 struct 260 { 261 ULONG ClearColor; 262 ULONG CurrentColor; 263 } u24; 264 struct 265 { 266 ULONG ClearColor; 267 ULONG CurrentColor; 268 } u32; 269 }; 270 GLenum Mode; 271 }; 272 273 /* WGL <-> mesa glue */ 274 static const struct pixel_format* get_format(INT pf_index, INT* pf_count) 275 { 276 HDC hdc; 277 INT bpp, nb_format; 278 const struct pixel_format* ret; 279 280 hdc = GetDC(NULL); 281 bpp = GetDeviceCaps(hdc, BITSPIXEL); 282 ReleaseDC(NULL, hdc); 283 284 switch (bpp) 285 { 286 #define HANDLE_BPP(__x__) \ 287 case __x__: \ 288 nb_format = ARRAYSIZE(pixel_formats_##__x__); \ 289 if ((pf_index > nb_format) || (pf_index <= 0)) \ 290 ret = NULL; \ 291 else \ 292 ret = &pixel_formats_##__x__[pf_index - 1]; \ 293 break 294 295 HANDLE_BPP(32); 296 HANDLE_BPP(24); 297 HANDLE_BPP(16); 298 HANDLE_BPP(8); 299 #undef HANDLE_BPP 300 default: 301 FIXME("Unhandled bit depth %u, defaulting to 32bpp\n", bpp); 302 nb_format = ARRAYSIZE(pixel_formats_32); 303 if ((pf_index > nb_format) || (pf_index == 0)) 304 ret = NULL; 305 else 306 ret = &pixel_formats_32[pf_index - 1]; 307 } 308 309 if (pf_count) 310 *pf_count = nb_format; 311 312 return ret; 313 } 314 315 INT sw_DescribePixelFormat(HDC hdc, INT format, UINT size, PIXELFORMATDESCRIPTOR* descr) 316 { 317 INT ret; 318 const struct pixel_format *pixel_format; 319 320 TRACE("Describing format %i.\n", format); 321 322 pixel_format = get_format(format, &ret); 323 if(!descr) 324 return ret; 325 if((format > ret) || (size != sizeof(*descr))) 326 return 0; 327 328 /* Fill the structure */ 329 descr->nSize = sizeof(*descr); 330 descr->nVersion = 1; 331 descr->dwFlags = pixel_format->dwFlags; 332 descr->iPixelType = pixel_format->iPixelType; 333 descr->cColorBits = pixel_format->cColorBits; 334 descr->cRedBits = pixel_format->cRedBits; 335 descr->cRedShift = pixel_format->cRedShift; 336 descr->cGreenBits = pixel_format->cGreenBits; 337 descr->cGreenShift = pixel_format->cGreenShift; 338 descr->cBlueBits = pixel_format->cBlueBits; 339 descr->cBlueShift = pixel_format->cBlueShift; 340 descr->cAlphaBits = pixel_format->cAlphaBits; 341 descr->cAlphaShift = pixel_format->cAlphaShift; 342 descr->cAccumBits = pixel_format->cAccumBits; 343 descr->cAccumRedBits = pixel_format->cAccumRedBits; 344 descr->cAccumGreenBits = pixel_format->cAccumGreenBits; 345 descr->cAccumBlueBits = pixel_format->cAccumBlueBits; 346 descr->cAccumAlphaBits = pixel_format->cAccumAlphaBits; 347 descr->cDepthBits = pixel_format->cDepthBits; 348 descr->cStencilBits = STENCIL_BITS; 349 descr->cAuxBuffers = 0; 350 descr->iLayerType = PFD_MAIN_PLANE; 351 descr->bReserved = 0; 352 descr->dwLayerMask = 0; 353 descr->dwVisibleMask = 0; 354 descr->dwDamageMask = 0; 355 356 return ret; 357 } 358 359 BOOL sw_SetPixelFormat(HDC hdc, struct wgl_dc_data* dc_data, INT format) 360 { 361 struct sw_framebuffer* fb; 362 const struct pixel_format *pixel_format; 363 364 /* So, someone is crazy enough to ask for sw implementation. Announce it. */ 365 TRACE("OpenGL software implementation START for hdc %p, format %i!\n", hdc, format); 366 367 pixel_format = get_format(format, NULL); 368 if (!pixel_format) 369 return FALSE; 370 371 /* allocate our structure */ 372 fb = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*fb)); 373 if(!fb) 374 { 375 ERR("HeapAlloc FAILED!\n"); 376 return FALSE; 377 } 378 /* Set the format */ 379 fb->pixel_format = pixel_format; 380 381 fb->gl_visual = gl_create_visual( 382 pixel_format->iPixelType == PFD_TYPE_RGBA, 383 pixel_format->cAlphaBits != 0, 384 (pixel_format->dwFlags & PFD_DOUBLEBUFFER) != 0, 385 pixel_format->cDepthBits, 386 STENCIL_BITS, 387 max(max(max(pixel_format->cAccumRedBits, pixel_format->cAccumGreenBits), pixel_format->cAccumBlueBits), pixel_format->cAccumAlphaBits), 388 pixel_format->iPixelType == PFD_TYPE_COLORINDEX ? pixel_format->cColorBits : 0, 389 ((1ul << pixel_format->cRedBits) - 1), 390 ((1ul << pixel_format->cGreenBits) - 1), 391 ((1ul << pixel_format->cBlueBits) - 1), 392 pixel_format->cAlphaBits != 0 ? ((1ul << pixel_format->cAlphaBits) - 1) : 255.0f, 393 pixel_format->cRedBits, 394 pixel_format->cGreenBits, 395 pixel_format->cBlueBits, 396 pixel_format->cAlphaBits); 397 398 if(!fb->gl_visual) 399 { 400 ERR("Failed to allocate a GL visual.\n"); 401 HeapFree(GetProcessHeap(), 0, fb); 402 return FALSE; 403 } 404 405 /* Allocate the framebuffer structure */ 406 fb->gl_buffer = gl_create_framebuffer(fb->gl_visual); 407 if (!fb->gl_buffer) { 408 ERR("Failed to allocate the mesa framebuffer structure.\n"); 409 gl_destroy_visual( fb->gl_visual ); 410 HeapFree(GetProcessHeap(), 0, fb); 411 return FALSE; 412 } 413 414 /* Save our DC */ 415 fb->Hdc = hdc; 416 417 /* Everything went fine */ 418 dc_data->sw_data = fb; 419 return TRUE; 420 } 421 422 DHGLRC sw_CreateContext(struct wgl_dc_data* dc_data) 423 { 424 struct sw_context* sw_ctx; 425 struct sw_framebuffer* fb = dc_data->sw_data; 426 427 sw_ctx = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*sw_ctx)); 428 if(!sw_ctx) 429 return NULL; 430 431 /* Initialize the context */ 432 sw_ctx->gl_ctx = gl_create_context(fb->gl_visual, NULL, sw_ctx); 433 if(!sw_ctx->gl_ctx) 434 { 435 ERR("Failed to initialize the mesa context.\n"); 436 HeapFree(GetProcessHeap(), 0, sw_ctx); 437 return NULL; 438 } 439 440 sw_ctx->fb = fb; 441 442 /* Choose relevant default */ 443 sw_ctx->Mode = fb->gl_visual->DBflag ? GL_BACK : GL_FRONT; 444 445 return (DHGLRC)sw_ctx; 446 } 447 448 BOOL sw_DeleteContext(DHGLRC dhglrc) 449 { 450 struct sw_context* sw_ctx = (struct sw_context*)dhglrc; 451 /* Those get clobbered by _mesa_free_context_data via _glapi_set{context,dispath_table} */ 452 void* icd_save = IntGetCurrentICDPrivate(); 453 const GLDISPATCHTABLE* table_save = IntGetCurrentDispatchTable(); 454 455 /* Destroy everything */ 456 gl_destroy_context(sw_ctx->gl_ctx); 457 458 HeapFree(GetProcessHeap(), 0, sw_ctx); 459 460 /* Restore this */ 461 IntSetCurrentDispatchTable(table_save); 462 IntSetCurrentICDPrivate(icd_save); 463 464 return TRUE; 465 } 466 467 extern void APIENTRY _mesa_ColorTableEXT(GLenum, GLenum, GLsizei, GLenum, GLenum, const void*); 468 extern void APIENTRY _mesa_ColorSubTableEXT(GLenum, GLsizei, GLsizei, GLenum, GLenum, const void*); 469 extern void APIENTRY _mesa_GetColorTableEXT(GLenum, GLenum, GLenum, void*); 470 extern void APIENTRY _mesa_GetColorTableParameterivEXT(GLenum, GLenum, GLfloat*); 471 extern void APIENTRY _mesa_GetColorTableParameterfvEXT(GLenum, GLenum, GLint*); 472 473 static void APIENTRY _swimpl_AddSwapHintRectWIN(GLint x, GLint y, GLsizei width, GLsizei height) 474 { 475 UNIMPLEMENTED; 476 } 477 478 PROC sw_GetProcAddress(LPCSTR name) 479 { 480 /* GL_EXT_paletted_texture */ 481 if (strcmp(name, "glColorTableEXT") == 0) 482 return (PROC)_mesa_ColorTableEXT; 483 if (strcmp(name, "glColorSubTableEXT") == 0) 484 return (PROC)_mesa_ColorSubTableEXT; 485 if (strcmp(name, "glColorGetTableEXT") == 0) 486 return (PROC)_mesa_GetColorTableEXT; 487 if (strcmp(name, "glGetColorTableParameterivEXT") == 0) 488 return (PROC)_mesa_GetColorTableParameterivEXT; 489 if (strcmp(name, "glGetColorTableParameterfvEXT") == 0) 490 return (PROC)_mesa_GetColorTableParameterfvEXT; 491 if (strcmp(name, "glAddSwapHintRectWIN") == 0) 492 return (PROC)_swimpl_AddSwapHintRectWIN; 493 494 WARN("Asking for proc address %s, returning NULL.\n", name); 495 return NULL; 496 } 497 498 BOOL sw_CopyContext(DHGLRC dhglrcSrc, DHGLRC dhglrcDst, UINT mask) 499 { 500 FIXME("Software wglCopyContext is UNIMPLEMENTED, mask %lx.\n", mask); 501 return FALSE; 502 } 503 504 BOOL sw_ShareLists(DHGLRC dhglrcSrc, DHGLRC dhglrcDst) 505 { 506 #if 0 507 struct sw_context* sw_ctx_src = (struct sw_context*)dhglrcSrc; 508 struct sw_context* sw_ctx_dst = (struct sw_context*)dhglrcDst; 509 510 /* See if it was already shared */ 511 if(sw_ctx_dst->gl_ctx->Shared->RefCount > 1) 512 return FALSE; 513 514 /* Unreference the old, share the new */ 515 gl_reference_shared_state(sw_ctx_dst->gl_ctx, 516 &sw_ctx_dst->gl_ctx->Shared, 517 sw_ctx_src->gl_ctx->Shared); 518 #endif 519 FIXME("Unimplemented!\n"); 520 return TRUE; 521 } 522 523 static 524 LRESULT CALLBACK 525 sw_call_window_proc( 526 int nCode, 527 WPARAM wParam, 528 LPARAM lParam ) 529 { 530 struct wgl_dc_data* dc_data = IntGetCurrentDcData(); 531 struct sw_context* ctx = (struct sw_context*)IntGetCurrentDHGLRC(); 532 PCWPSTRUCT pParams = (PCWPSTRUCT)lParam; 533 534 if((!dc_data) || (!ctx)) 535 return 0; 536 537 if(!(dc_data->flags & WGL_DC_OBJ_DC)) 538 return 0; 539 540 if((nCode < 0) || (dc_data->owner.hwnd != pParams->hwnd) || (dc_data->sw_data == NULL)) 541 return CallNextHookEx(ctx->hook, nCode, wParam, lParam); 542 543 if (pParams->message == WM_WINDOWPOSCHANGED) 544 { 545 /* We handle WM_WINDOWPOSCHANGED instead of WM_SIZE because according to 546 * http://blogs.msdn.com/oldnewthing/archive/2008/01/15/7113860.aspx 547 * WM_SIZE is generated from WM_WINDOWPOSCHANGED by DefWindowProc so it 548 * can be masked out by the application. */ 549 LPWINDOWPOS lpWindowPos = (LPWINDOWPOS)pParams->lParam; 550 if((lpWindowPos->flags & SWP_SHOWWINDOW) || 551 !(lpWindowPos->flags & SWP_NOMOVE) || 552 !(lpWindowPos->flags & SWP_NOSIZE)) 553 { 554 /* Size in WINDOWPOS includes the window frame, so get the size 555 * of the client area via GetClientRect. */ 556 RECT client_rect; 557 UINT width, height; 558 559 TRACE("Got WM_WINDOWPOSCHANGED\n"); 560 561 GetClientRect(pParams->hwnd, &client_rect); 562 width = client_rect.right - client_rect.left; 563 height = client_rect.bottom - client_rect.top; 564 /* Do not reallocate for minimized windows */ 565 if(width <= 0 || height <= 0) 566 goto end; 567 /* Propagate to mesa */ 568 gl_ResizeBuffersMESA(ctx->gl_ctx); 569 } 570 } 571 572 end: 573 return CallNextHookEx(ctx->hook, nCode, wParam, lParam); 574 } 575 576 static const char* renderer_string(void) 577 { 578 return "ReactOS SW Implementation"; 579 } 580 581 static inline void PUT_PIXEL_8(BYTE* Buffer, BYTE Value) 582 { 583 *Buffer = Value; 584 } 585 static inline void PUT_PIXEL_16(USHORT* Buffer, USHORT Value) 586 { 587 *Buffer = Value; 588 } 589 static inline void PUT_PIXEL_24(ULONG* Buffer, ULONG Value) 590 { 591 *Buffer &= 0xFF000000ul; 592 *Buffer |= Value & 0x00FFFFFF; 593 } 594 static inline void PUT_PIXEL_32(ULONG* Buffer, ULONG Value) 595 { 596 *Buffer = Value; 597 } 598 599 static inline BYTE GET_PIXEL_8(BYTE* Buffer) 600 { 601 return *Buffer; 602 } 603 604 static inline USHORT GET_PIXEL_16(USHORT* Buffer) 605 { 606 return *Buffer; 607 } 608 609 static inline ULONG GET_PIXEL_24(ULONG* Buffer) 610 { 611 return *Buffer & 0x00FFFFFF; 612 } 613 614 static inline ULONG GET_PIXEL_32(ULONG* Buffer) 615 { 616 return *Buffer; 617 } 618 619 static inline BYTE PACK_COLOR_8(GLubyte r, GLubyte g, GLubyte b) 620 { 621 return (r & 0x7) | ((g & 0x7) << 3) | ((b & 0x3) << 6); 622 } 623 624 static inline USHORT PACK_COLOR_16(GLubyte r, GLubyte g, GLubyte b) 625 { 626 return ((r & 0x1F) << 11) | ((g & 0x3F) << 5) | (b & 0x1F); 627 } 628 629 static inline ULONG PACK_COLOR_24(GLubyte r, GLubyte g, GLubyte b) 630 { 631 return (r << 16) | (g << 8) | (b); 632 } 633 634 static inline ULONG PACK_COLOR_32(GLubyte r, GLubyte g, GLubyte b) 635 { 636 return (r << 16) | (g << 8) | (b); 637 } 638 639 static inline COLORREF PACK_COLORREF_8(GLubyte r, GLubyte g, GLubyte b) 640 { 641 return RGB(r << 5, g << 5, b << 6); 642 } 643 644 static inline COLORREF PACK_COLORREF_16(GLubyte r, GLubyte g, GLubyte b) 645 { 646 return RGB(r << 3, g << 2, b << 3); 647 } 648 649 static inline COLORREF PACK_COLORREF_24(GLubyte r, GLubyte g, GLubyte b) 650 { 651 return RGB(r, g, b); 652 } 653 654 static inline COLORREF PACK_COLORREF_32(GLubyte r, GLubyte g, GLubyte b) 655 { 656 return RGB(r, g, b); 657 } 658 659 static inline void UNPACK_COLOR_8(BYTE Color, GLubyte* r, GLubyte* g, GLubyte* b) 660 { 661 *r = Color & 0x7; 662 *g = (Color >> 3) & 0x7; 663 *b = (Color >> 6) & 0x3; 664 } 665 666 static inline void UNPACK_COLOR_16(USHORT Color, GLubyte* r, GLubyte* g, GLubyte* b) 667 { 668 *r = (Color >> 11) & 0x1F; 669 *g = (Color >> 5) & 0x3F; 670 *b = Color & 0x1F; 671 } 672 673 static inline void UNPACK_COLOR_24(ULONG Color, GLubyte* r, GLubyte* g, GLubyte* b) 674 { 675 *r = (Color >> 16) & 0xFF; 676 *g = (Color >> 8) & 0xFF; 677 *b = Color & 0xFF; 678 } 679 680 static inline void UNPACK_COLOR_32(ULONG Color, GLubyte* r, GLubyte* g, GLubyte* b) 681 { 682 *r = (Color >> 16) & 0xFF; 683 *g = (Color >> 8) & 0xFF; 684 *b = Color & 0xFF; 685 } 686 687 static inline void UNPACK_COLORREF_8(COLORREF Color, GLubyte* r, GLubyte* g, GLubyte* b) 688 { 689 *r = GetRValue(Color) >> 5; 690 *g = GetGValue(Color) >> 5; 691 *b = GetBValue(Color) >> 6; 692 } 693 694 static inline void UNPACK_COLORREF_16(COLORREF Color, GLubyte* r, GLubyte* g, GLubyte* b) 695 { 696 *r = GetRValue(Color) >> 3; 697 *g = GetGValue(Color) >> 2; 698 *b = GetBValue(Color) >> 3; 699 } 700 701 static inline void UNPACK_COLORREF_24(COLORREF Color, GLubyte* r, GLubyte* g, GLubyte* b) 702 { 703 *r = GetRValue(Color); 704 *g = GetGValue(Color); 705 *b = GetBValue(Color); 706 } 707 708 static inline void UNPACK_COLORREF_32(COLORREF Color, GLubyte* r, GLubyte* g, GLubyte* b) 709 { 710 *r = GetRValue(Color); 711 *g = GetGValue(Color); 712 *b = GetBValue(Color); 713 } 714 715 #define MAKE_COLORREF(__bpp, __type) \ 716 static inline COLORREF MAKE_COLORREF_##__bpp(const struct pixel_format *format, __type Color) \ 717 { \ 718 GLubyte r,g,b; \ 719 \ 720 if (format->iPixelType == PFD_TYPE_COLORINDEX) \ 721 return PALETTEINDEX(Color); \ 722 \ 723 UNPACK_COLOR_##__bpp(Color, &r, &g, &b); \ 724 \ 725 return PACK_COLORREF_##__bpp(r, g, b); \ 726 } 727 MAKE_COLORREF(8, BYTE) 728 MAKE_COLORREF(16, USHORT) 729 MAKE_COLORREF(24, ULONG) 730 MAKE_COLORREF(32, ULONG) 731 #undef MAKE_COLORREF 732 733 /* 734 * Set the color index used to clear the color buffer. 735 */ 736 #define CLEAR_INDEX(__bpp, __type) \ 737 static void clear_index_##__bpp(GLcontext* ctx, GLuint index) \ 738 { \ 739 struct sw_context* sw_ctx = ctx->DriverCtx; \ 740 \ 741 sw_ctx->u##__bpp.ClearColor = (__type)index; \ 742 } 743 CLEAR_INDEX(8, BYTE) 744 CLEAR_INDEX(16, USHORT) 745 CLEAR_INDEX(24, ULONG) 746 CLEAR_INDEX(32, ULONG) 747 #undef CLEAR_INDEX 748 749 /* 750 * Set the color used to clear the color buffer. 751 */ 752 #define CLEAR_COLOR(__bpp) \ 753 static void clear_color_##__bpp( GLcontext* ctx, GLubyte r, GLubyte g, GLubyte b, GLubyte a ) \ 754 { \ 755 struct sw_context* sw_ctx = ctx->DriverCtx; \ 756 \ 757 sw_ctx->u##__bpp.ClearColor = PACK_COLOR_##__bpp(r, g, b); \ 758 \ 759 TRACE("Set Clear color %u, %u, %u.\n", r, g, b); \ 760 } 761 CLEAR_COLOR(8) 762 CLEAR_COLOR(16) 763 CLEAR_COLOR(24) 764 CLEAR_COLOR(32) 765 #undef CLEAR_COLOR 766 767 /* 768 * Clear the specified region of the color buffer using the clear color 769 * or index as specified by one of the two functions above. 770 */ 771 static void clear_frontbuffer( 772 struct sw_context* sw_ctx, 773 struct sw_framebuffer* fb, 774 GLint x, 775 GLint y, 776 GLint width, 777 GLint height, 778 COLORREF ClearColor) 779 { 780 HBRUSH Brush; 781 BOOL ret; 782 783 TRACE("Clearing front buffer (%u, %u, %u, %u), color 0x%08x.\n", x, y, width, height, ClearColor); 784 785 Brush = CreateSolidBrush(ClearColor); 786 Brush = SelectObject(fb->Hdc, Brush); 787 788 ret = PatBlt(fb->Hdc, x, fb->height - (y + height), width, height, PATCOPY); 789 if (!ret) 790 { 791 ERR("PatBlt failed. last Error %d.\n", GetLastError()); 792 } 793 794 Brush = SelectObject(fb->Hdc, Brush); 795 DeleteObject(Brush); 796 } 797 798 #define CLEAR(__bpp, __type, __pixel_size) \ 799 static void clear_##__bpp(GLcontext* ctx, GLboolean all,GLint x, GLint y, GLint width, GLint height)\ 800 { \ 801 struct sw_context* sw_ctx = ctx->DriverCtx; \ 802 struct sw_framebuffer* fb = sw_ctx->fb; \ 803 BYTE* ScanLine; \ 804 \ 805 if (all) \ 806 { \ 807 x = y = 0; \ 808 width = fb->width; \ 809 height = fb->height; \ 810 } \ 811 \ 812 if (sw_ctx->Mode == GL_FRONT) \ 813 { \ 814 clear_frontbuffer(sw_ctx, fb, x, y, width, height, \ 815 MAKE_COLORREF_##__bpp(fb->pixel_format, sw_ctx->u##__bpp.ClearColor)); \ 816 return; \ 817 } \ 818 \ 819 ScanLine = fb->BackBuffer + y * WIDTH_BYTES_ALIGN32(fb->width, __bpp); \ 820 while (height--) \ 821 { \ 822 BYTE* Buffer = ScanLine + x * __pixel_size; \ 823 UINT n = width; \ 824 \ 825 while (n--) \ 826 { \ 827 PUT_PIXEL_##__bpp((__type*)Buffer, sw_ctx->u##__bpp.ClearColor); \ 828 Buffer += __pixel_size; \ 829 } \ 830 \ 831 ScanLine += WIDTH_BYTES_ALIGN32(fb->width, __bpp); \ 832 } \ 833 } 834 CLEAR(8, BYTE, 1) 835 CLEAR(16, USHORT, 2) 836 CLEAR(24, ULONG, 3) 837 CLEAR(32, ULONG, 4) 838 #undef CLEAR 839 840 /* Set the current color index. */ 841 #define SET_INDEX(__bpp) \ 842 static void set_index_##__bpp(GLcontext* ctx, GLuint index) \ 843 { \ 844 struct sw_context* sw_ctx = ctx->DriverCtx; \ 845 \ 846 sw_ctx->u##__bpp.CurrentColor = index; \ 847 } 848 SET_INDEX(8) 849 SET_INDEX(16) 850 SET_INDEX(24) 851 SET_INDEX(32) 852 #undef SET_INDEX 853 854 /* Set the current RGBA color. */ 855 #define SET_COLOR(__bpp) \ 856 static void set_color_##__bpp( GLcontext* ctx, GLubyte r, GLubyte g, GLubyte b, GLubyte a ) \ 857 { \ 858 struct sw_context* sw_ctx = ctx->DriverCtx; \ 859 \ 860 sw_ctx->u##__bpp.CurrentColor = PACK_COLOR_##__bpp(r, g, b); \ 861 } 862 SET_COLOR(8) 863 SET_COLOR(16) 864 SET_COLOR(24) 865 SET_COLOR(32) 866 #undef SET_COLOR 867 868 /* 869 * Selects either the front or back color buffer for reading and writing. 870 * mode is either GL_FRONT or GL_BACK. 871 */ 872 static GLboolean set_buffer( GLcontext* ctx, GLenum mode ) 873 { 874 struct sw_context* sw_ctx = ctx->DriverCtx; 875 struct sw_framebuffer* fb = sw_ctx->fb; 876 877 if (!fb->gl_visual->DBflag) 878 return GL_FALSE; 879 880 if ((mode != GL_FRONT) && (mode != GL_BACK)) 881 return GL_FALSE; 882 883 sw_ctx->Mode = mode; 884 return GL_TRUE; 885 } 886 887 /* Return characteristics of the output buffer. */ 888 static void buffer_size(GLcontext* ctx, GLuint *width, GLuint *height) 889 { 890 struct sw_context* sw_ctx = ctx->DriverCtx; 891 struct sw_framebuffer* fb = sw_ctx->fb; 892 HWND Window = WindowFromDC(fb->Hdc); 893 894 if (Window) 895 { 896 RECT client_rect; 897 GetClientRect(Window, &client_rect); 898 *width = client_rect.right - client_rect.left; 899 *height = client_rect.bottom - client_rect.top; 900 } 901 else 902 { 903 /* We are drawing to a bitmap */ 904 BITMAP bm; 905 HBITMAP Hbm; 906 907 Hbm = GetCurrentObject(fb->Hdc, OBJ_BITMAP); 908 909 if (!GetObjectW(Hbm, sizeof(bm), &bm)) 910 return; 911 912 TRACE("Framebuffer size : %i, %i\n", bm.bmWidth, bm.bmHeight); 913 914 *width = bm.bmWidth; 915 *height = bm.bmHeight; 916 } 917 918 if ((*width != fb->width) || (*height != fb->height)) 919 { 920 const struct pixel_format* pixel_format = fb->pixel_format; 921 922 if (pixel_format->dwFlags & PFD_DOUBLEBUFFER) 923 { 924 /* Allocate a new backbuffer */ 925 size_t BufferSize = *height * WIDTH_BYTES_ALIGN32(*width, pixel_format->cColorBits); 926 if (!fb->BackBuffer) 927 { 928 fb->BackBuffer = HeapAlloc(GetProcessHeap(), 0, BufferSize); 929 } 930 else 931 { 932 fb->BackBuffer = HeapReAlloc(GetProcessHeap(), 0, fb->BackBuffer, BufferSize); 933 } 934 if (!fb->BackBuffer) 935 { 936 ERR("Failed allocating back buffer !.\n"); 937 return; 938 } 939 } 940 941 fb->width = *width; 942 fb->height = *height; 943 } 944 } 945 946 /* Write a horizontal span of color pixels with a boolean mask. */ 947 #define WRITE_COLOR_SPAN_FRONTBUFFER(__bpp) \ 948 static void write_color_span_frontbuffer_##__bpp(struct sw_framebuffer* fb, \ 949 GLuint n, GLint x, GLint y, \ 950 const GLubyte red[], const GLubyte green[], \ 951 const GLubyte blue[], const GLubyte mask[] ) \ 952 { \ 953 TRACE("Writing color span at %u, %u (%u)\n", x, y, n); \ 954 \ 955 if (mask) \ 956 { \ 957 while (n--) \ 958 { \ 959 if (mask[n]) \ 960 { \ 961 SetPixel(fb->Hdc, x + n, fb->height - y, \ 962 PACK_COLORREF_##__bpp(red[n], green[n], blue[n])); \ 963 } \ 964 } \ 965 } \ 966 else \ 967 { \ 968 while (n--) \ 969 { \ 970 SetPixel(fb->Hdc, x + n, fb->height - y, \ 971 PACK_COLORREF_##__bpp(red[n], green[n], blue[n])); \ 972 } \ 973 } \ 974 } 975 WRITE_COLOR_SPAN_FRONTBUFFER(8) 976 WRITE_COLOR_SPAN_FRONTBUFFER(16) 977 WRITE_COLOR_SPAN_FRONTBUFFER(24) 978 WRITE_COLOR_SPAN_FRONTBUFFER(32) 979 #undef WRITE_COLOR_SPAN_FRONTBUFFER 980 981 #define WRITE_COLOR_SPAN(__bpp, __type, __pixel_size) \ 982 static void write_color_span_##__bpp(GLcontext* ctx, \ 983 GLuint n, GLint x, GLint y, \ 984 const GLubyte red[], const GLubyte green[], \ 985 const GLubyte blue[], const GLubyte alpha[], \ 986 const GLubyte mask[] ) \ 987 { \ 988 struct sw_context* sw_ctx = ctx->DriverCtx; \ 989 struct sw_framebuffer* fb = sw_ctx->fb; \ 990 BYTE* Buffer; \ 991 \ 992 if (sw_ctx->Mode == GL_FRONT) \ 993 { \ 994 write_color_span_frontbuffer_##__bpp(fb, n, x, y, red, green, blue, mask); \ 995 return; \ 996 } \ 997 \ 998 Buffer = fb->BackBuffer + y * WIDTH_BYTES_ALIGN32(fb->width, __bpp) \ 999 + (x + n) * __pixel_size; \ 1000 if (mask) \ 1001 { \ 1002 while (n--) \ 1003 { \ 1004 Buffer -= __pixel_size; \ 1005 if (mask[n]) \ 1006 { \ 1007 PUT_PIXEL_##__bpp((__type*)Buffer, \ 1008 PACK_COLOR_##__bpp(red[n], green[n], blue[n])); \ 1009 } \ 1010 } \ 1011 } \ 1012 else \ 1013 { \ 1014 while (n--) \ 1015 { \ 1016 Buffer -= __pixel_size; \ 1017 PUT_PIXEL_##__bpp((__type*)Buffer, \ 1018 PACK_COLOR_##__bpp(red[n], green[n], blue[n])); \ 1019 } \ 1020 } \ 1021 } 1022 WRITE_COLOR_SPAN(8, BYTE, 1) 1023 WRITE_COLOR_SPAN(16, USHORT, 2) 1024 WRITE_COLOR_SPAN(24, ULONG, 3) 1025 WRITE_COLOR_SPAN(32, ULONG, 4) 1026 #undef WRITE_COLOR_SPAN 1027 1028 static void write_monocolor_span_frontbuffer(struct sw_framebuffer* fb, GLuint n, GLint x, GLint y, 1029 const GLubyte mask[], COLORREF Color) 1030 { 1031 TRACE("Writing monocolor span at %u %u (%u), Color 0x%08x", x, y, n, Color); 1032 1033 if (mask) 1034 { 1035 while (n--) 1036 { 1037 if (mask[n]) 1038 SetPixel(fb->Hdc, x + n, y, Color); 1039 } 1040 } 1041 else 1042 { 1043 HBRUSH Brush = CreateSolidBrush(Color); 1044 Brush = SelectObject(fb->Hdc, Brush); 1045 1046 PatBlt(fb->Hdc, x, fb->height - y, n, 1, PATCOPY); 1047 1048 Brush = SelectObject(fb->Hdc, Brush); 1049 DeleteObject(Brush); 1050 } 1051 } 1052 1053 #define WRITE_MONOCOLOR_SPAN(__bpp, __type, __pixel_size) \ 1054 static void write_monocolor_span_##__bpp(GLcontext* ctx, \ 1055 GLuint n, GLint x, GLint y, \ 1056 const GLubyte mask[]) \ 1057 { \ 1058 struct sw_context* sw_ctx = ctx->DriverCtx; \ 1059 struct sw_framebuffer* fb = sw_ctx->fb; \ 1060 BYTE* Buffer; \ 1061 \ 1062 if (sw_ctx->Mode == GL_FRONT) \ 1063 { \ 1064 write_monocolor_span_frontbuffer(fb, n, x, y, mask, \ 1065 MAKE_COLORREF_##__bpp(fb->pixel_format, sw_ctx->u##__bpp.CurrentColor)); \ 1066 return; \ 1067 } \ 1068 \ 1069 Buffer = fb->BackBuffer + y * WIDTH_BYTES_ALIGN32(fb->width, __bpp) + (x + n) * __pixel_size; \ 1070 if (mask) \ 1071 { \ 1072 while (n--) \ 1073 { \ 1074 Buffer -= __pixel_size; \ 1075 if (mask[n]) \ 1076 PUT_PIXEL_##__bpp((__type*)Buffer, sw_ctx->u##__bpp.CurrentColor); \ 1077 } \ 1078 } \ 1079 else \ 1080 { \ 1081 while(n--) \ 1082 { \ 1083 Buffer -= __pixel_size; \ 1084 PUT_PIXEL_##__bpp((__type*)Buffer, sw_ctx->u##__bpp.CurrentColor); \ 1085 } \ 1086 } \ 1087 } 1088 WRITE_MONOCOLOR_SPAN(8, BYTE, 1) 1089 WRITE_MONOCOLOR_SPAN(16, USHORT, 2) 1090 WRITE_MONOCOLOR_SPAN(24, ULONG, 3) 1091 WRITE_MONOCOLOR_SPAN(32, ULONG, 4) 1092 #undef WRITE_MONOCOLOR_SPAN 1093 1094 /* Write an array of pixels with a boolean mask. */ 1095 #define WRITE_COLOR_PIXELS(__bpp, __type, __pixel_size) \ 1096 static void write_color_pixels_##__bpp(GLcontext* ctx, \ 1097 GLuint n, const GLint x[], const GLint y[], \ 1098 const GLubyte r[], const GLubyte g[], \ 1099 const GLubyte b[], const GLubyte a[], \ 1100 const GLubyte mask[]) \ 1101 { \ 1102 struct sw_context* sw_ctx = ctx->DriverCtx; \ 1103 struct sw_framebuffer* fb = sw_ctx->fb; \ 1104 \ 1105 TRACE("Writing color pixels\n"); \ 1106 \ 1107 if (sw_ctx->Mode == GL_FRONT) \ 1108 { \ 1109 while (n--) \ 1110 { \ 1111 if (mask[n]) \ 1112 { \ 1113 TRACE("Setting pixel %u, %u to 0x%08x.\n", x[n], fb->height - y[n], \ 1114 PACK_COLORREF_##__bpp(r[n], g[n], b[n])); \ 1115 SetPixel(fb->Hdc, x[n], fb->height - y[n], \ 1116 PACK_COLORREF_##__bpp(r[n], g[n], b[n])); \ 1117 } \ 1118 } \ 1119 \ 1120 return; \ 1121 } \ 1122 \ 1123 while (n--) \ 1124 { \ 1125 if (mask[n]) \ 1126 { \ 1127 BYTE* Buffer = fb->BackBuffer + y[n] * WIDTH_BYTES_ALIGN32(fb->width, __bpp) \ 1128 + x[n] * __pixel_size; \ 1129 PUT_PIXEL_##__bpp((__type*)Buffer, PACK_COLOR_##__bpp(r[n], g[n], b[n])); \ 1130 } \ 1131 } \ 1132 } 1133 WRITE_COLOR_PIXELS(8, BYTE, 1) 1134 WRITE_COLOR_PIXELS(16, USHORT, 2) 1135 WRITE_COLOR_PIXELS(24, ULONG, 3) 1136 WRITE_COLOR_PIXELS(32, ULONG, 4) 1137 #undef WRITE_COLOR_PIXELS 1138 1139 static void write_monocolor_pixels_frontbuffer( 1140 struct sw_framebuffer* fb, GLuint n, 1141 const GLint x[], const GLint y[], 1142 const GLubyte mask[], COLORREF Color) 1143 { 1144 TRACE("Writing monocolor pixels to front buffer.\n"); 1145 1146 while (n--) 1147 { 1148 if (mask[n]) 1149 { 1150 SetPixel(fb->Hdc, x[n], fb->height - y[n], Color); 1151 } 1152 } 1153 } 1154 1155 /* 1156 * Write an array of pixels with a boolean mask. The current color 1157 * is used for all pixels. 1158 */ 1159 #define WRITE_MONOCOLOR_PIXELS(__bpp, __type, __pixel_size) \ 1160 static void write_monocolor_pixels_##__bpp(GLcontext* ctx, GLuint n, \ 1161 const GLint x[], const GLint y[], \ 1162 const GLubyte mask[] ) \ 1163 { \ 1164 struct sw_context* sw_ctx = ctx->DriverCtx; \ 1165 struct sw_framebuffer* fb = sw_ctx->fb; \ 1166 \ 1167 if (sw_ctx->Mode == GL_FRONT) \ 1168 { \ 1169 write_monocolor_pixels_frontbuffer(fb, n, x, y, mask, \ 1170 MAKE_COLORREF_##__bpp(fb->pixel_format, sw_ctx->u##__bpp.CurrentColor)); \ 1171 \ 1172 return; \ 1173 } \ 1174 \ 1175 while (n--) \ 1176 { \ 1177 if (mask[n]) \ 1178 { \ 1179 BYTE* Buffer = fb->BackBuffer + y[n] * WIDTH_BYTES_ALIGN32(fb->width, 32) \ 1180 + x[n] * __pixel_size; \ 1181 PUT_PIXEL_##__bpp((__type*)Buffer, sw_ctx->u##__bpp.CurrentColor); \ 1182 } \ 1183 } \ 1184 } 1185 WRITE_MONOCOLOR_PIXELS(8, BYTE, 1) 1186 WRITE_MONOCOLOR_PIXELS(16, USHORT, 2) 1187 WRITE_MONOCOLOR_PIXELS(24, ULONG, 3) 1188 WRITE_MONOCOLOR_PIXELS(32, ULONG, 4) 1189 #undef WRITE_MONOCOLOR_PIXELS 1190 1191 /* Write a horizontal span of color pixels with a boolean mask. */ 1192 static void write_index_span( GLcontext* ctx, 1193 GLuint n, GLint x, GLint y, 1194 const GLuint index[], 1195 const GLubyte mask[] ) 1196 { 1197 ERR("Not implemented yet !\n"); 1198 } 1199 1200 /* Write an array of pixels with a boolean mask. */ 1201 static void write_index_pixels( GLcontext* ctx, 1202 GLuint n, const GLint x[], const GLint y[], 1203 const GLuint index[], const GLubyte mask[] ) 1204 { 1205 ERR("Not implemented yet !\n"); 1206 } 1207 1208 /* Read a horizontal span of color-index pixels. */ 1209 static void read_index_span( GLcontext* ctx, GLuint n, GLint x, GLint y, GLuint index[]) 1210 { 1211 ERR("Not implemented yet !\n"); 1212 } 1213 1214 /* Read a horizontal span of color pixels. */ 1215 #define READ_COLOR_SPAN(__bpp, __type, __pixel_size) \ 1216 static void read_color_span_##__bpp(GLcontext* ctx, \ 1217 GLuint n, GLint x, GLint y, \ 1218 GLubyte red[], GLubyte green[], \ 1219 GLubyte blue[], GLubyte alpha[] ) \ 1220 { \ 1221 struct sw_context* sw_ctx = ctx->DriverCtx; \ 1222 struct sw_framebuffer* fb = sw_ctx->fb; \ 1223 BYTE* Buffer; \ 1224 \ 1225 if (sw_ctx->Mode == GL_FRONT) \ 1226 { \ 1227 COLORREF Color; \ 1228 while (n--) \ 1229 { \ 1230 Color = GetPixel(fb->Hdc, x + n, fb->height - y); \ 1231 UNPACK_COLORREF_##__bpp(Color, &red[n], &green[n], &blue[n]); \ 1232 alpha[n] = 0; \ 1233 } \ 1234 \ 1235 return; \ 1236 } \ 1237 \ 1238 Buffer = fb->BackBuffer + y * WIDTH_BYTES_ALIGN32(fb->width, __bpp) \ 1239 + (x + n) * __pixel_size; \ 1240 while (n--) \ 1241 { \ 1242 Buffer -= __pixel_size; \ 1243 UNPACK_COLOR_##__bpp(GET_PIXEL_##__bpp((__type*)Buffer), \ 1244 &blue[n], &green[n], &red[n]); \ 1245 alpha[n] = 0; \ 1246 } \ 1247 } 1248 READ_COLOR_SPAN(8, BYTE, 1) 1249 READ_COLOR_SPAN(16, USHORT, 2) 1250 READ_COLOR_SPAN(24, ULONG, 3) 1251 READ_COLOR_SPAN(32, ULONG, 4) 1252 #undef READ_COLOR_SPAN 1253 1254 /* Read an array of color index pixels. */ 1255 static void read_index_pixels(GLcontext* ctx, 1256 GLuint n, const GLint x[], const GLint y[], 1257 GLuint index[], const GLubyte mask[]) 1258 { 1259 1260 ERR("Not implemented yet !\n"); 1261 } 1262 1263 /* Read an array of color pixels. */ 1264 #define READ_COLOR_PIXELS(__bpp, __type, __pixel_size) \ 1265 static void read_color_pixels_##__bpp(GLcontext* ctx, \ 1266 GLuint n, const GLint x[], const GLint y[], \ 1267 GLubyte red[], GLubyte green[], \ 1268 GLubyte blue[], GLubyte alpha[], \ 1269 const GLubyte mask[] ) \ 1270 { \ 1271 struct sw_context* sw_ctx = ctx->DriverCtx; \ 1272 struct sw_framebuffer* fb = sw_ctx->fb; \ 1273 \ 1274 if (sw_ctx->Mode == GL_FRONT) \ 1275 { \ 1276 COLORREF Color; \ 1277 while (n--) \ 1278 { \ 1279 if (mask[n]) \ 1280 { \ 1281 Color = GetPixel(fb->Hdc, x[n], fb->height - y[n]); \ 1282 UNPACK_COLORREF_##__bpp(Color, &red[n], &green[n], &blue[n]); \ 1283 alpha[n] = 0; \ 1284 } \ 1285 } \ 1286 \ 1287 return; \ 1288 } \ 1289 \ 1290 while (n--) \ 1291 { \ 1292 if (mask[n]) \ 1293 { \ 1294 BYTE *Buffer = fb->BackBuffer + y[n] * WIDTH_BYTES_ALIGN32(fb->width, __bpp) \ 1295 + x[n] * __pixel_size; \ 1296 UNPACK_COLOR_##__bpp(GET_PIXEL_##__bpp((__type*)Buffer), \ 1297 &blue[n], &green[n], &red[n]); \ 1298 alpha[n] = 0; \ 1299 } \ 1300 } \ 1301 } 1302 READ_COLOR_PIXELS(8, BYTE, 1) 1303 READ_COLOR_PIXELS(16, USHORT, 2) 1304 READ_COLOR_PIXELS(24, ULONG, 3) 1305 READ_COLOR_PIXELS(32, ULONG, 4) 1306 #undef READ_COLOR_PIXELS 1307 1308 static void setup_DD_pointers( GLcontext* ctx ) 1309 { 1310 struct sw_context* sw_ctx = ctx->DriverCtx; 1311 1312 ctx->Driver.RendererString = renderer_string; 1313 ctx->Driver.UpdateState = setup_DD_pointers; 1314 1315 switch (sw_ctx->fb->pixel_format->cColorBits) 1316 { 1317 #define HANDLE_BPP(__bpp) \ 1318 case __bpp: \ 1319 ctx->Driver.ClearIndex = clear_index_##__bpp; \ 1320 ctx->Driver.ClearColor = clear_color_##__bpp; \ 1321 ctx->Driver.Clear = clear_##__bpp; \ 1322 ctx->Driver.Index = set_index_##__bpp; \ 1323 ctx->Driver.Color = set_color_##__bpp; \ 1324 ctx->Driver.WriteColorSpan = write_color_span_##__bpp; \ 1325 ctx->Driver.WriteMonocolorSpan = write_monocolor_span_##__bpp; \ 1326 ctx->Driver.WriteMonoindexSpan = write_monocolor_span_##__bpp; \ 1327 ctx->Driver.WriteColorPixels = write_color_pixels_##__bpp; \ 1328 ctx->Driver.WriteMonocolorPixels = write_monocolor_pixels_##__bpp; \ 1329 ctx->Driver.WriteMonoindexPixels = write_monocolor_pixels_##__bpp; \ 1330 ctx->Driver.ReadColorSpan = read_color_span_##__bpp; \ 1331 ctx->Driver.ReadColorPixels = read_color_pixels_##__bpp; \ 1332 break 1333 HANDLE_BPP(8); 1334 HANDLE_BPP(16); 1335 HANDLE_BPP(24); 1336 HANDLE_BPP(32); 1337 #undef HANDLE_BPP 1338 default: 1339 ERR("Unhandled bit depth %u, defaulting to 32bpp.\n", sw_ctx->fb->pixel_format->cColorBits); 1340 ctx->Driver.ClearIndex = clear_index_32; 1341 ctx->Driver.ClearColor = clear_color_32; 1342 ctx->Driver.Clear = clear_32; 1343 ctx->Driver.Index = set_index_32; 1344 ctx->Driver.Color = set_color_32; 1345 ctx->Driver.WriteColorSpan = write_color_span_32; 1346 ctx->Driver.WriteMonocolorSpan = write_monocolor_span_32; 1347 ctx->Driver.WriteMonoindexSpan = write_monocolor_span_32; 1348 ctx->Driver.WriteColorPixels = write_color_pixels_32; 1349 ctx->Driver.WriteMonocolorPixels = write_monocolor_pixels_32; 1350 ctx->Driver.WriteMonoindexPixels = write_monocolor_pixels_32; 1351 ctx->Driver.ReadColorSpan = read_color_span_32; 1352 ctx->Driver.ReadColorPixels = read_color_pixels_32; 1353 break; 1354 } 1355 1356 ctx->Driver.SetBuffer = set_buffer; 1357 ctx->Driver.GetBufferSize = buffer_size; 1358 1359 /* Pixel/span writing functions: */ 1360 ctx->Driver.WriteIndexSpan = write_index_span; 1361 ctx->Driver.WriteIndexPixels = write_index_pixels; 1362 1363 /* Pixel/span reading functions: */ 1364 ctx->Driver.ReadIndexSpan = read_index_span; 1365 ctx->Driver.ReadIndexPixels = read_index_pixels; 1366 } 1367 1368 /* Declare API table */ 1369 #define USE_GL_FUNC(name, proto_args, call_args, offset, stack) extern void WINAPI _mesa_##name proto_args ; 1370 #define USE_GL_FUNC_RET(name, ret_type, proto_args, call_args, offset, stack) extern ret_type WINAPI _mesa_##name proto_args ; 1371 #include "glfuncs.h" 1372 1373 static GLCLTPROCTABLE sw_api_table = 1374 { 1375 OPENGL_VERSION_110_ENTRIES, 1376 { 1377 #define USE_GL_FUNC(name, proto_args, call_args, offset, stack) _mesa_##name, 1378 #include "glfuncs.h" 1379 } 1380 }; 1381 1382 /* Glue code */ 1383 GLcontext* gl_get_thread_context(void) 1384 { 1385 struct sw_context* sw_ctx = (struct sw_context*)IntGetCurrentDHGLRC(); 1386 return sw_ctx->gl_ctx; 1387 } 1388 1389 1390 BOOL sw_SetContext(struct wgl_dc_data* dc_data, DHGLRC dhglrc) 1391 { 1392 struct sw_context* sw_ctx = (struct sw_context*)dhglrc; 1393 struct sw_framebuffer* fb = dc_data->sw_data; 1394 UINT width, height; 1395 1396 /* Get framebuffer size */ 1397 if(dc_data->flags & WGL_DC_OBJ_DC) 1398 { 1399 HWND hwnd = dc_data->owner.hwnd; 1400 RECT client_rect; 1401 if(!hwnd) 1402 { 1403 ERR("Physical DC without a window!\n"); 1404 return FALSE; 1405 } 1406 if(!GetClientRect(hwnd, &client_rect)) 1407 { 1408 ERR("GetClientRect failed!\n"); 1409 return FALSE; 1410 } 1411 1412 /* This is a physical DC. Setup the hook */ 1413 sw_ctx->hook = SetWindowsHookEx(WH_CALLWNDPROC, 1414 sw_call_window_proc, 1415 NULL, 1416 GetCurrentThreadId()); 1417 1418 /* Calculate width & height */ 1419 width = client_rect.right - client_rect.left; 1420 height = client_rect.bottom - client_rect.top; 1421 } 1422 else /* OBJ_MEMDC */ 1423 { 1424 BITMAP bm; 1425 HBITMAP hbmp; 1426 HDC hdc = dc_data->owner.hdc; 1427 1428 if(fb->gl_visual->DBflag) 1429 { 1430 ERR("Memory DC called with a double buffered format.\n"); 1431 return FALSE; 1432 } 1433 1434 hbmp = GetCurrentObject( hdc, OBJ_BITMAP ); 1435 if(!hbmp) 1436 { 1437 ERR("No Bitmap!\n"); 1438 return FALSE; 1439 } 1440 if(GetObject(hbmp, sizeof(bm), &bm) == 0) 1441 { 1442 ERR("GetObject failed!\n"); 1443 return FALSE; 1444 } 1445 width = bm.bmWidth; 1446 height = bm.bmHeight; 1447 } 1448 1449 if(!width) width = 1; 1450 if(!height) height = 1; 1451 1452 /* Also make the mesa context current to mesa */ 1453 gl_make_current(sw_ctx->gl_ctx, fb->gl_buffer); 1454 1455 /* Setup our functions */ 1456 setup_DD_pointers(sw_ctx->gl_ctx); 1457 1458 /* Set the viewport if this is the first time we initialize this context */ 1459 if(sw_ctx->gl_ctx->Viewport.X == 0 && 1460 sw_ctx->gl_ctx->Viewport.Y == 0 && 1461 sw_ctx->gl_ctx->Viewport.Width == 0 && 1462 sw_ctx->gl_ctx->Viewport.Height == 0) 1463 { 1464 gl_Viewport(sw_ctx->gl_ctx, 0, 0, width, height); 1465 } 1466 1467 /* update the framebuffer size */ 1468 gl_ResizeBuffersMESA(sw_ctx->gl_ctx); 1469 1470 /* Use our API table */ 1471 IntSetCurrentDispatchTable(&sw_api_table.glDispatchTable); 1472 1473 /* We're good */ 1474 return TRUE; 1475 } 1476 1477 void sw_ReleaseContext(DHGLRC dhglrc) 1478 { 1479 struct sw_context* sw_ctx = (struct sw_context*)dhglrc; 1480 1481 /* Forward to mesa */ 1482 gl_make_current(NULL, NULL); 1483 1484 /* Unhook */ 1485 if(sw_ctx->hook) 1486 { 1487 UnhookWindowsHookEx(sw_ctx->hook); 1488 sw_ctx->hook = NULL; 1489 } 1490 } 1491 1492 BOOL sw_SwapBuffers(HDC hdc, struct wgl_dc_data* dc_data) 1493 { 1494 struct sw_framebuffer* fb = dc_data->sw_data; 1495 char Buffer[sizeof(BITMAPINFOHEADER) + 3 * sizeof(DWORD)]; 1496 BITMAPINFO *bmi = (BITMAPINFO*)Buffer; 1497 BYTE Bpp = fb->pixel_format->cColorBits; 1498 1499 if (!fb->gl_visual->DBflag) 1500 return TRUE; 1501 1502 if (!fb->BackBuffer) 1503 return FALSE; 1504 1505 bmi->bmiHeader.biSize = sizeof(bmi->bmiHeader); 1506 bmi->bmiHeader.biBitCount = Bpp; 1507 bmi->bmiHeader.biClrImportant = 0; 1508 bmi->bmiHeader.biClrUsed = 0; 1509 bmi->bmiHeader.biPlanes = 1; 1510 bmi->bmiHeader.biSizeImage = WIDTH_BYTES_ALIGN32(fb->width, Bpp) * fb->height; 1511 bmi->bmiHeader.biXPelsPerMeter = 0; 1512 bmi->bmiHeader.biYPelsPerMeter = 0; 1513 bmi->bmiHeader.biHeight = fb->height; 1514 bmi->bmiHeader.biWidth = fb->width; 1515 bmi->bmiHeader.biCompression = Bpp == 16 ? BI_BITFIELDS : BI_RGB; 1516 1517 if (Bpp == 16) 1518 { 1519 DWORD* BitMasks = (DWORD*)(&bmi->bmiColors[0]); 1520 BitMasks[0] = 0x0000F800; 1521 BitMasks[1] = 0x000007E0; 1522 BitMasks[2] = 0x0000001F; 1523 } 1524 1525 return SetDIBitsToDevice(fb->Hdc, 0, 0, fb->width, fb->height, 0, 0, 0, fb->height, fb->BackBuffer, bmi, 1526 fb->pixel_format->iPixelType == PFD_TYPE_COLORINDEX ? DIB_PAL_COLORS : DIB_RGB_COLORS) != 0; 1527 } 1528