root/drivers/media/pci/ivtv/ivtv-yuv.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. ivtv_yuv_prep_user_dma
  2. ivtv_yuv_filter_check
  3. ivtv_yuv_filter
  4. ivtv_yuv_handle_horizontal
  5. ivtv_yuv_handle_vertical
  6. ivtv_yuv_window_setup
  7. ivtv_yuv_work_handler
  8. ivtv_yuv_init
  9. ivtv_yuv_next_free
  10. ivtv_yuv_setup_frame
  11. ivtv_yuv_frame_complete
  12. ivtv_yuv_udma_frame
  13. ivtv_yuv_setup_stream_frame
  14. ivtv_yuv_udma_stream_frame
  15. ivtv_yuv_prep_frame
  16. ivtv_yuv_close

   1 // SPDX-License-Identifier: GPL-2.0-or-later
   2 /*
   3     yuv support
   4 
   5     Copyright (C) 2007  Ian Armstrong <ian@iarmst.demon.co.uk>
   6 
   7  */
   8 
   9 #include "ivtv-driver.h"
  10 #include "ivtv-udma.h"
  11 #include "ivtv-yuv.h"
  12 
  13 /* YUV buffer offsets */
  14 const u32 yuv_offset[IVTV_YUV_BUFFERS] = {
  15         0x001a8600,
  16         0x00240400,
  17         0x002d8200,
  18         0x00370000,
  19         0x00029000,
  20         0x000C0E00,
  21         0x006B0400,
  22         0x00748200
  23 };
  24 
  25 static int ivtv_yuv_prep_user_dma(struct ivtv *itv, struct ivtv_user_dma *dma,
  26                                   struct ivtv_dma_frame *args)
  27 {
  28         struct ivtv_dma_page_info y_dma;
  29         struct ivtv_dma_page_info uv_dma;
  30         struct yuv_playback_info *yi = &itv->yuv_info;
  31         u8 frame = yi->draw_frame;
  32         struct yuv_frame_info *f = &yi->new_frame_info[frame];
  33         int i;
  34         int y_pages, uv_pages;
  35         unsigned long y_buffer_offset, uv_buffer_offset;
  36         int y_decode_height, uv_decode_height, y_size;
  37 
  38         y_buffer_offset = IVTV_DECODER_OFFSET + yuv_offset[frame];
  39         uv_buffer_offset = y_buffer_offset + IVTV_YUV_BUFFER_UV_OFFSET;
  40 
  41         y_decode_height = uv_decode_height = f->src_h + f->src_y;
  42 
  43         if (f->offset_y)
  44                 y_buffer_offset += 720 * 16;
  45 
  46         if (y_decode_height & 15)
  47                 y_decode_height = (y_decode_height + 16) & ~15;
  48 
  49         if (uv_decode_height & 31)
  50                 uv_decode_height = (uv_decode_height + 32) & ~31;
  51 
  52         y_size = 720 * y_decode_height;
  53 
  54         /* Still in USE */
  55         if (dma->SG_length || dma->page_count) {
  56                 IVTV_DEBUG_WARN
  57                     ("prep_user_dma: SG_length %d page_count %d still full?\n",
  58                      dma->SG_length, dma->page_count);
  59                 return -EBUSY;
  60         }
  61 
  62         ivtv_udma_get_page_info (&y_dma, (unsigned long)args->y_source, 720 * y_decode_height);
  63         ivtv_udma_get_page_info (&uv_dma, (unsigned long)args->uv_source, 360 * uv_decode_height);
  64 
  65         /* Get user pages for DMA Xfer */
  66         y_pages = get_user_pages_unlocked(y_dma.uaddr,
  67                         y_dma.page_count, &dma->map[0], FOLL_FORCE);
  68         uv_pages = 0; /* silence gcc. value is set and consumed only if: */
  69         if (y_pages == y_dma.page_count) {
  70                 uv_pages = get_user_pages_unlocked(uv_dma.uaddr,
  71                                 uv_dma.page_count, &dma->map[y_pages],
  72                                 FOLL_FORCE);
  73         }
  74 
  75         if (y_pages != y_dma.page_count || uv_pages != uv_dma.page_count) {
  76                 int rc = -EFAULT;
  77 
  78                 if (y_pages == y_dma.page_count) {
  79                         IVTV_DEBUG_WARN
  80                                 ("failed to map uv user pages, returned %d expecting %d\n",
  81                                  uv_pages, uv_dma.page_count);
  82 
  83                         if (uv_pages >= 0) {
  84                                 for (i = 0; i < uv_pages; i++)
  85                                         put_page(dma->map[y_pages + i]);
  86                                 rc = -EFAULT;
  87                         } else {
  88                                 rc = uv_pages;
  89                         }
  90                 } else {
  91                         IVTV_DEBUG_WARN
  92                                 ("failed to map y user pages, returned %d expecting %d\n",
  93                                  y_pages, y_dma.page_count);
  94                 }
  95                 if (y_pages >= 0) {
  96                         for (i = 0; i < y_pages; i++)
  97                                 put_page(dma->map[i]);
  98                         /*
  99                          * Inherit the -EFAULT from rc's
 100                          * initialization, but allow it to be
 101                          * overridden by uv_pages above if it was an
 102                          * actual errno.
 103                          */
 104                 } else {
 105                         rc = y_pages;
 106                 }
 107                 return rc;
 108         }
 109 
 110         dma->page_count = y_pages + uv_pages;
 111 
 112         /* Fill & map SG List */
 113         if (ivtv_udma_fill_sg_list (dma, &uv_dma, ivtv_udma_fill_sg_list (dma, &y_dma, 0)) < 0) {
 114                 IVTV_DEBUG_WARN("could not allocate bounce buffers for highmem userspace buffers\n");
 115                 for (i = 0; i < dma->page_count; i++) {
 116                         put_page(dma->map[i]);
 117                 }
 118                 dma->page_count = 0;
 119                 return -ENOMEM;
 120         }
 121         dma->SG_length = pci_map_sg(itv->pdev, dma->SGlist, dma->page_count, PCI_DMA_TODEVICE);
 122 
 123         /* Fill SG Array with new values */
 124         ivtv_udma_fill_sg_array(dma, y_buffer_offset, uv_buffer_offset, y_size);
 125 
 126         /* If we've offset the y plane, ensure top area is blanked */
 127         if (f->offset_y && yi->blanking_dmaptr) {
 128                 dma->SGarray[dma->SG_length].size = cpu_to_le32(720*16);
 129                 dma->SGarray[dma->SG_length].src = cpu_to_le32(yi->blanking_dmaptr);
 130                 dma->SGarray[dma->SG_length].dst = cpu_to_le32(IVTV_DECODER_OFFSET + yuv_offset[frame]);
 131                 dma->SG_length++;
 132         }
 133 
 134         /* Tag SG Array with Interrupt Bit */
 135         dma->SGarray[dma->SG_length - 1].size |= cpu_to_le32(0x80000000);
 136 
 137         ivtv_udma_sync_for_device(itv);
 138         return 0;
 139 }
 140 
 141 /* We rely on a table held in the firmware - Quick check. */
 142 int ivtv_yuv_filter_check(struct ivtv *itv)
 143 {
 144         int i, y, uv;
 145 
 146         for (i = 0, y = 16, uv = 4; i < 16; i++, y += 24, uv += 12) {
 147                 if ((read_dec(IVTV_YUV_HORIZONTAL_FILTER_OFFSET + y) != i << 16) ||
 148                     (read_dec(IVTV_YUV_VERTICAL_FILTER_OFFSET + uv) != i << 16)) {
 149                         IVTV_WARN ("YUV filter table not found in firmware.\n");
 150                         return -1;
 151                 }
 152         }
 153         return 0;
 154 }
 155 
 156 static void ivtv_yuv_filter(struct ivtv *itv, int h_filter, int v_filter_1, int v_filter_2)
 157 {
 158         u32 i, line;
 159 
 160         /* If any filter is -1, then don't update it */
 161         if (h_filter > -1) {
 162                 if (h_filter > 4)
 163                         h_filter = 4;
 164                 i = IVTV_YUV_HORIZONTAL_FILTER_OFFSET + (h_filter * 384);
 165                 for (line = 0; line < 16; line++) {
 166                         write_reg(read_dec(i), 0x02804);
 167                         write_reg(read_dec(i), 0x0281c);
 168                         i += 4;
 169                         write_reg(read_dec(i), 0x02808);
 170                         write_reg(read_dec(i), 0x02820);
 171                         i += 4;
 172                         write_reg(read_dec(i), 0x0280c);
 173                         write_reg(read_dec(i), 0x02824);
 174                         i += 4;
 175                         write_reg(read_dec(i), 0x02810);
 176                         write_reg(read_dec(i), 0x02828);
 177                         i += 4;
 178                         write_reg(read_dec(i), 0x02814);
 179                         write_reg(read_dec(i), 0x0282c);
 180                         i += 8;
 181                         write_reg(0, 0x02818);
 182                         write_reg(0, 0x02830);
 183                 }
 184                 IVTV_DEBUG_YUV("h_filter -> %d\n", h_filter);
 185         }
 186 
 187         if (v_filter_1 > -1) {
 188                 if (v_filter_1 > 4)
 189                         v_filter_1 = 4;
 190                 i = IVTV_YUV_VERTICAL_FILTER_OFFSET + (v_filter_1 * 192);
 191                 for (line = 0; line < 16; line++) {
 192                         write_reg(read_dec(i), 0x02900);
 193                         i += 4;
 194                         write_reg(read_dec(i), 0x02904);
 195                         i += 8;
 196                         write_reg(0, 0x02908);
 197                 }
 198                 IVTV_DEBUG_YUV("v_filter_1 -> %d\n", v_filter_1);
 199         }
 200 
 201         if (v_filter_2 > -1) {
 202                 if (v_filter_2 > 4)
 203                         v_filter_2 = 4;
 204                 i = IVTV_YUV_VERTICAL_FILTER_OFFSET + (v_filter_2 * 192);
 205                 for (line = 0; line < 16; line++) {
 206                         write_reg(read_dec(i), 0x0290c);
 207                         i += 4;
 208                         write_reg(read_dec(i), 0x02910);
 209                         i += 8;
 210                         write_reg(0, 0x02914);
 211                 }
 212                 IVTV_DEBUG_YUV("v_filter_2 -> %d\n", v_filter_2);
 213         }
 214 }
 215 
 216 static void ivtv_yuv_handle_horizontal(struct ivtv *itv, struct yuv_frame_info *f)
 217 {
 218         struct yuv_playback_info *yi = &itv->yuv_info;
 219         u32 reg_2834, reg_2838, reg_283c;
 220         u32 reg_2844, reg_2854, reg_285c;
 221         u32 reg_2864, reg_2874, reg_2890;
 222         u32 reg_2870, reg_2870_base, reg_2870_offset;
 223         int x_cutoff;
 224         int h_filter;
 225         u32 master_width;
 226 
 227         IVTV_DEBUG_WARN
 228             ("Adjust to width %d src_w %d dst_w %d src_x %d dst_x %d\n",
 229              f->tru_w, f->src_w, f->dst_w, f->src_x, f->dst_x);
 230 
 231         /* How wide is the src image */
 232         x_cutoff = f->src_w + f->src_x;
 233 
 234         /* Set the display width */
 235         reg_2834 = f->dst_w;
 236         reg_2838 = reg_2834;
 237 
 238         /* Set the display position */
 239         reg_2890 = f->dst_x;
 240 
 241         /* Index into the image horizontally */
 242         reg_2870 = 0;
 243 
 244         /* 2870 is normally fudged to align video coords with osd coords.
 245            If running full screen, it causes an unwanted left shift
 246            Remove the fudge if we almost fill the screen.
 247            Gradually adjust the offset to avoid the video 'snapping'
 248            left/right if it gets dragged through this region.
 249            Only do this if osd is full width. */
 250         if (f->vis_w == 720) {
 251                 if ((f->tru_x - f->pan_x > -1) && (f->tru_x - f->pan_x <= 40) && (f->dst_w >= 680))
 252                         reg_2870 = 10 - (f->tru_x - f->pan_x) / 4;
 253                 else if ((f->tru_x - f->pan_x < 0) && (f->tru_x - f->pan_x >= -20) && (f->dst_w >= 660))
 254                         reg_2870 = (10 + (f->tru_x - f->pan_x) / 2);
 255 
 256                 if (f->dst_w >= f->src_w)
 257                         reg_2870 = reg_2870 << 16 | reg_2870;
 258                 else
 259                         reg_2870 = ((reg_2870 & ~1) << 15) | (reg_2870 & ~1);
 260         }
 261 
 262         if (f->dst_w < f->src_w)
 263                 reg_2870 = 0x000d000e - reg_2870;
 264         else
 265                 reg_2870 = 0x0012000e - reg_2870;
 266 
 267         /* We're also using 2870 to shift the image left (src_x & negative dst_x) */
 268         reg_2870_offset = (f->src_x * ((f->dst_w << 21) / f->src_w)) >> 19;
 269 
 270         if (f->dst_w >= f->src_w) {
 271                 x_cutoff &= ~1;
 272                 master_width = (f->src_w * 0x00200000) / (f->dst_w);
 273                 if (master_width * f->dst_w != f->src_w * 0x00200000)
 274                         master_width++;
 275                 reg_2834 = (reg_2834 << 16) | x_cutoff;
 276                 reg_2838 = (reg_2838 << 16) | x_cutoff;
 277                 reg_283c = master_width >> 2;
 278                 reg_2844 = master_width >> 2;
 279                 reg_2854 = master_width;
 280                 reg_285c = master_width >> 1;
 281                 reg_2864 = master_width >> 1;
 282 
 283                 /* We also need to factor in the scaling
 284                    (src_w - dst_w) / (src_w / 4) */
 285                 if (f->dst_w > f->src_w)
 286                         reg_2870_base = ((f->dst_w - f->src_w)<<16) / (f->src_w <<14);
 287                 else
 288                         reg_2870_base = 0;
 289 
 290                 reg_2870 += (((reg_2870_offset << 14) & 0xFFFF0000) | reg_2870_offset >> 2) + (reg_2870_base << 17 | reg_2870_base);
 291                 reg_2874 = 0;
 292         } else if (f->dst_w < f->src_w / 2) {
 293                 master_width = (f->src_w * 0x00080000) / f->dst_w;
 294                 if (master_width * f->dst_w != f->src_w * 0x00080000)
 295                         master_width++;
 296                 reg_2834 = (reg_2834 << 16) | x_cutoff;
 297                 reg_2838 = (reg_2838 << 16) | x_cutoff;
 298                 reg_283c = master_width >> 2;
 299                 reg_2844 = master_width >> 1;
 300                 reg_2854 = master_width;
 301                 reg_285c = master_width >> 1;
 302                 reg_2864 = master_width >> 1;
 303                 reg_2870 += ((reg_2870_offset << 15) & 0xFFFF0000) | reg_2870_offset;
 304                 reg_2870 += (5 - (((f->src_w + f->src_w / 2) - 1) / f->dst_w)) << 16;
 305                 reg_2874 = 0x00000012;
 306         } else {
 307                 master_width = (f->src_w * 0x00100000) / f->dst_w;
 308                 if (master_width * f->dst_w != f->src_w * 0x00100000)
 309                         master_width++;
 310                 reg_2834 = (reg_2834 << 16) | x_cutoff;
 311                 reg_2838 = (reg_2838 << 16) | x_cutoff;
 312                 reg_283c = master_width >> 2;
 313                 reg_2844 = master_width >> 1;
 314                 reg_2854 = master_width;
 315                 reg_285c = master_width >> 1;
 316                 reg_2864 = master_width >> 1;
 317                 reg_2870 += ((reg_2870_offset << 14) & 0xFFFF0000) | reg_2870_offset >> 1;
 318                 reg_2870 += (5 - (((f->src_w * 3) - 1) / f->dst_w)) << 16;
 319                 reg_2874 = 0x00000001;
 320         }
 321 
 322         /* Select the horizontal filter */
 323         if (f->src_w == f->dst_w) {
 324                 /* An exact size match uses filter 0 */
 325                 h_filter = 0;
 326         } else {
 327                 /* Figure out which filter to use */
 328                 h_filter = ((f->src_w << 16) / f->dst_w) >> 15;
 329                 h_filter = (h_filter >> 1) + (h_filter & 1);
 330                 /* Only an exact size match can use filter 0 */
 331                 h_filter += !h_filter;
 332         }
 333 
 334         write_reg(reg_2834, 0x02834);
 335         write_reg(reg_2838, 0x02838);
 336         IVTV_DEBUG_YUV("Update reg 0x2834 %08x->%08x 0x2838 %08x->%08x\n",
 337                        yi->reg_2834, reg_2834, yi->reg_2838, reg_2838);
 338 
 339         write_reg(reg_283c, 0x0283c);
 340         write_reg(reg_2844, 0x02844);
 341 
 342         IVTV_DEBUG_YUV("Update reg 0x283c %08x->%08x 0x2844 %08x->%08x\n",
 343                        yi->reg_283c, reg_283c, yi->reg_2844, reg_2844);
 344 
 345         write_reg(0x00080514, 0x02840);
 346         write_reg(0x00100514, 0x02848);
 347         IVTV_DEBUG_YUV("Update reg 0x2840 %08x->%08x 0x2848 %08x->%08x\n",
 348                        yi->reg_2840, 0x00080514, yi->reg_2848, 0x00100514);
 349 
 350         write_reg(reg_2854, 0x02854);
 351         IVTV_DEBUG_YUV("Update reg 0x2854 %08x->%08x \n",
 352                        yi->reg_2854, reg_2854);
 353 
 354         write_reg(reg_285c, 0x0285c);
 355         write_reg(reg_2864, 0x02864);
 356         IVTV_DEBUG_YUV("Update reg 0x285c %08x->%08x 0x2864 %08x->%08x\n",
 357                        yi->reg_285c, reg_285c, yi->reg_2864, reg_2864);
 358 
 359         write_reg(reg_2874, 0x02874);
 360         IVTV_DEBUG_YUV("Update reg 0x2874 %08x->%08x\n",
 361                        yi->reg_2874, reg_2874);
 362 
 363         write_reg(reg_2870, 0x02870);
 364         IVTV_DEBUG_YUV("Update reg 0x2870 %08x->%08x\n",
 365                        yi->reg_2870, reg_2870);
 366 
 367         write_reg(reg_2890, 0x02890);
 368         IVTV_DEBUG_YUV("Update reg 0x2890 %08x->%08x\n",
 369                        yi->reg_2890, reg_2890);
 370 
 371         /* Only update the filter if we really need to */
 372         if (h_filter != yi->h_filter) {
 373                 ivtv_yuv_filter(itv, h_filter, -1, -1);
 374                 yi->h_filter = h_filter;
 375         }
 376 }
 377 
 378 static void ivtv_yuv_handle_vertical(struct ivtv *itv, struct yuv_frame_info *f)
 379 {
 380         struct yuv_playback_info *yi = &itv->yuv_info;
 381         u32 master_height;
 382         u32 reg_2918, reg_291c, reg_2920, reg_2928;
 383         u32 reg_2930, reg_2934, reg_293c;
 384         u32 reg_2940, reg_2944, reg_294c;
 385         u32 reg_2950, reg_2954, reg_2958, reg_295c;
 386         u32 reg_2960, reg_2964, reg_2968, reg_296c;
 387         u32 reg_289c;
 388         u32 src_major_y, src_minor_y;
 389         u32 src_major_uv, src_minor_uv;
 390         u32 reg_2964_base, reg_2968_base;
 391         int v_filter_1, v_filter_2;
 392 
 393         IVTV_DEBUG_WARN
 394             ("Adjust to height %d src_h %d dst_h %d src_y %d dst_y %d\n",
 395              f->tru_h, f->src_h, f->dst_h, f->src_y, f->dst_y);
 396 
 397         /* What scaling mode is being used... */
 398         IVTV_DEBUG_YUV("Scaling mode Y: %s\n",
 399                        f->interlaced_y ? "Interlaced" : "Progressive");
 400 
 401         IVTV_DEBUG_YUV("Scaling mode UV: %s\n",
 402                        f->interlaced_uv ? "Interlaced" : "Progressive");
 403 
 404         /* What is the source video being treated as... */
 405         IVTV_DEBUG_WARN("Source video: %s\n",
 406                         f->interlaced ? "Interlaced" : "Progressive");
 407 
 408         /* We offset into the image using two different index methods, so split
 409            the y source coord into two parts. */
 410         if (f->src_y < 8) {
 411                 src_minor_uv = f->src_y;
 412                 src_major_uv = 0;
 413         } else {
 414                 src_minor_uv = 8;
 415                 src_major_uv = f->src_y - 8;
 416         }
 417 
 418         src_minor_y = src_minor_uv;
 419         src_major_y = src_major_uv;
 420 
 421         if (f->offset_y)
 422                 src_minor_y += 16;
 423 
 424         if (f->interlaced_y)
 425                 reg_2918 = (f->dst_h << 16) | (f->src_h + src_minor_y);
 426         else
 427                 reg_2918 = (f->dst_h << 16) | ((f->src_h + src_minor_y) << 1);
 428 
 429         if (f->interlaced_uv)
 430                 reg_291c = (f->dst_h << 16) | ((f->src_h + src_minor_uv) >> 1);
 431         else
 432                 reg_291c = (f->dst_h << 16) | (f->src_h + src_minor_uv);
 433 
 434         reg_2964_base = (src_minor_y * ((f->dst_h << 16) / f->src_h)) >> 14;
 435         reg_2968_base = (src_minor_uv * ((f->dst_h << 16) / f->src_h)) >> 14;
 436 
 437         if (f->dst_h / 2 >= f->src_h && !f->interlaced_y) {
 438                 master_height = (f->src_h * 0x00400000) / f->dst_h;
 439                 if ((f->src_h * 0x00400000) - (master_height * f->dst_h) >= f->dst_h / 2)
 440                         master_height++;
 441                 reg_2920 = master_height >> 2;
 442                 reg_2928 = master_height >> 3;
 443                 reg_2930 = master_height;
 444                 reg_2940 = master_height >> 1;
 445                 reg_2964_base >>= 3;
 446                 reg_2968_base >>= 3;
 447                 reg_296c = 0x00000000;
 448         } else if (f->dst_h >= f->src_h) {
 449                 master_height = (f->src_h * 0x00400000) / f->dst_h;
 450                 master_height = (master_height >> 1) + (master_height & 1);
 451                 reg_2920 = master_height >> 2;
 452                 reg_2928 = master_height >> 2;
 453                 reg_2930 = master_height;
 454                 reg_2940 = master_height >> 1;
 455                 reg_296c = 0x00000000;
 456                 if (f->interlaced_y) {
 457                         reg_2964_base >>= 3;
 458                 } else {
 459                         reg_296c++;
 460                         reg_2964_base >>= 2;
 461                 }
 462                 if (f->interlaced_uv)
 463                         reg_2928 >>= 1;
 464                 reg_2968_base >>= 3;
 465         } else if (f->dst_h >= f->src_h / 2) {
 466                 master_height = (f->src_h * 0x00200000) / f->dst_h;
 467                 master_height = (master_height >> 1) + (master_height & 1);
 468                 reg_2920 = master_height >> 2;
 469                 reg_2928 = master_height >> 2;
 470                 reg_2930 = master_height;
 471                 reg_2940 = master_height;
 472                 reg_296c = 0x00000101;
 473                 if (f->interlaced_y) {
 474                         reg_2964_base >>= 2;
 475                 } else {
 476                         reg_296c++;
 477                         reg_2964_base >>= 1;
 478                 }
 479                 if (f->interlaced_uv)
 480                         reg_2928 >>= 1;
 481                 reg_2968_base >>= 2;
 482         } else {
 483                 master_height = (f->src_h * 0x00100000) / f->dst_h;
 484                 master_height = (master_height >> 1) + (master_height & 1);
 485                 reg_2920 = master_height >> 2;
 486                 reg_2928 = master_height >> 2;
 487                 reg_2930 = master_height;
 488                 reg_2940 = master_height;
 489                 reg_2964_base >>= 1;
 490                 reg_2968_base >>= 2;
 491                 reg_296c = 0x00000102;
 492         }
 493 
 494         /* FIXME These registers change depending on scaled / unscaled output
 495            We really need to work out what they should be */
 496         if (f->src_h == f->dst_h) {
 497                 reg_2934 = 0x00020000;
 498                 reg_293c = 0x00100000;
 499                 reg_2944 = 0x00040000;
 500                 reg_294c = 0x000b0000;
 501         } else {
 502                 reg_2934 = 0x00000FF0;
 503                 reg_293c = 0x00000FF0;
 504                 reg_2944 = 0x00000FF0;
 505                 reg_294c = 0x00000FF0;
 506         }
 507 
 508         /* The first line to be displayed */
 509         reg_2950 = 0x00010000 + src_major_y;
 510         if (f->interlaced_y)
 511                 reg_2950 += 0x00010000;
 512         reg_2954 = reg_2950 + 1;
 513 
 514         reg_2958 = 0x00010000 + (src_major_y >> 1);
 515         if (f->interlaced_uv)
 516                 reg_2958 += 0x00010000;
 517         reg_295c = reg_2958 + 1;
 518 
 519         if (yi->decode_height == 480)
 520                 reg_289c = 0x011e0017;
 521         else
 522                 reg_289c = 0x01500017;
 523 
 524         if (f->dst_y < 0)
 525                 reg_289c = (reg_289c - ((f->dst_y & ~1)<<15))-(f->dst_y >>1);
 526         else
 527                 reg_289c = (reg_289c + ((f->dst_y & ~1)<<15))+(f->dst_y >>1);
 528 
 529         /* How much of the source to decode.
 530            Take into account the source offset */
 531         reg_2960 = ((src_minor_y + f->src_h + src_major_y) - 1) |
 532                 (((src_minor_uv + f->src_h + src_major_uv - 1) & ~1) << 15);
 533 
 534         /* Calculate correct value for register 2964 */
 535         if (f->src_h == f->dst_h) {
 536                 reg_2964 = 1;
 537         } else {
 538                 reg_2964 = 2 + ((f->dst_h << 1) / f->src_h);
 539                 reg_2964 = (reg_2964 >> 1) + (reg_2964 & 1);
 540         }
 541         reg_2968 = (reg_2964 << 16) + reg_2964 + (reg_2964 >> 1);
 542         reg_2964 = (reg_2964 << 16) + reg_2964 + (reg_2964 * 46 / 94);
 543 
 544         /* Okay, we've wasted time working out the correct value,
 545            but if we use it, it fouls the the window alignment.
 546            Fudge it to what we want... */
 547         reg_2964 = 0x00010001 + ((reg_2964 & 0x0000FFFF) - (reg_2964 >> 16));
 548         reg_2968 = 0x00010001 + ((reg_2968 & 0x0000FFFF) - (reg_2968 >> 16));
 549 
 550         /* Deviate further from what it should be. I find the flicker headache
 551            inducing so try to reduce it slightly. Leave 2968 as-is otherwise
 552            colours foul. */
 553         if ((reg_2964 != 0x00010001) && (f->dst_h / 2 <= f->src_h))
 554                 reg_2964 = (reg_2964 & 0xFFFF0000) + ((reg_2964 & 0x0000FFFF) / 2);
 555 
 556         if (!f->interlaced_y)
 557                 reg_2964 -= 0x00010001;
 558         if (!f->interlaced_uv)
 559                 reg_2968 -= 0x00010001;
 560 
 561         reg_2964 += ((reg_2964_base << 16) | reg_2964_base);
 562         reg_2968 += ((reg_2968_base << 16) | reg_2968_base);
 563 
 564         /* Select the vertical filter */
 565         if (f->src_h == f->dst_h) {
 566                 /* An exact size match uses filter 0/1 */
 567                 v_filter_1 = 0;
 568                 v_filter_2 = 1;
 569         } else {
 570                 /* Figure out which filter to use */
 571                 v_filter_1 = ((f->src_h << 16) / f->dst_h) >> 15;
 572                 v_filter_1 = (v_filter_1 >> 1) + (v_filter_1 & 1);
 573                 /* Only an exact size match can use filter 0 */
 574                 v_filter_1 += !v_filter_1;
 575                 v_filter_2 = v_filter_1;
 576         }
 577 
 578         write_reg(reg_2934, 0x02934);
 579         write_reg(reg_293c, 0x0293c);
 580         IVTV_DEBUG_YUV("Update reg 0x2934 %08x->%08x 0x293c %08x->%08x\n",
 581                        yi->reg_2934, reg_2934, yi->reg_293c, reg_293c);
 582         write_reg(reg_2944, 0x02944);
 583         write_reg(reg_294c, 0x0294c);
 584         IVTV_DEBUG_YUV("Update reg 0x2944 %08x->%08x 0x294c %08x->%08x\n",
 585                        yi->reg_2944, reg_2944, yi->reg_294c, reg_294c);
 586 
 587         /* Ensure 2970 is 0 (does it ever change ?) */
 588 /*      write_reg(0,0x02970); */
 589 /*      IVTV_DEBUG_YUV("Update reg 0x2970 %08x->%08x\n", yi->reg_2970, 0); */
 590 
 591         write_reg(reg_2930, 0x02938);
 592         write_reg(reg_2930, 0x02930);
 593         IVTV_DEBUG_YUV("Update reg 0x2930 %08x->%08x 0x2938 %08x->%08x\n",
 594                        yi->reg_2930, reg_2930, yi->reg_2938, reg_2930);
 595 
 596         write_reg(reg_2928, 0x02928);
 597         write_reg(reg_2928 + 0x514, 0x0292C);
 598         IVTV_DEBUG_YUV("Update reg 0x2928 %08x->%08x 0x292c %08x->%08x\n",
 599                        yi->reg_2928, reg_2928, yi->reg_292c, reg_2928 + 0x514);
 600 
 601         write_reg(reg_2920, 0x02920);
 602         write_reg(reg_2920 + 0x514, 0x02924);
 603         IVTV_DEBUG_YUV("Update reg 0x2920 %08x->%08x 0x2924 %08x->%08x\n",
 604                        yi->reg_2920, reg_2920, yi->reg_2924, reg_2920 + 0x514);
 605 
 606         write_reg(reg_2918, 0x02918);
 607         write_reg(reg_291c, 0x0291C);
 608         IVTV_DEBUG_YUV("Update reg 0x2918 %08x->%08x 0x291C %08x->%08x\n",
 609                        yi->reg_2918, reg_2918, yi->reg_291c, reg_291c);
 610 
 611         write_reg(reg_296c, 0x0296c);
 612         IVTV_DEBUG_YUV("Update reg 0x296c %08x->%08x\n",
 613                        yi->reg_296c, reg_296c);
 614 
 615         write_reg(reg_2940, 0x02948);
 616         write_reg(reg_2940, 0x02940);
 617         IVTV_DEBUG_YUV("Update reg 0x2940 %08x->%08x 0x2948 %08x->%08x\n",
 618                        yi->reg_2940, reg_2940, yi->reg_2948, reg_2940);
 619 
 620         write_reg(reg_2950, 0x02950);
 621         write_reg(reg_2954, 0x02954);
 622         IVTV_DEBUG_YUV("Update reg 0x2950 %08x->%08x 0x2954 %08x->%08x\n",
 623                        yi->reg_2950, reg_2950, yi->reg_2954, reg_2954);
 624 
 625         write_reg(reg_2958, 0x02958);
 626         write_reg(reg_295c, 0x0295C);
 627         IVTV_DEBUG_YUV("Update reg 0x2958 %08x->%08x 0x295C %08x->%08x\n",
 628                        yi->reg_2958, reg_2958, yi->reg_295c, reg_295c);
 629 
 630         write_reg(reg_2960, 0x02960);
 631         IVTV_DEBUG_YUV("Update reg 0x2960 %08x->%08x \n",
 632                        yi->reg_2960, reg_2960);
 633 
 634         write_reg(reg_2964, 0x02964);
 635         write_reg(reg_2968, 0x02968);
 636         IVTV_DEBUG_YUV("Update reg 0x2964 %08x->%08x 0x2968 %08x->%08x\n",
 637                        yi->reg_2964, reg_2964, yi->reg_2968, reg_2968);
 638 
 639         write_reg(reg_289c, 0x0289c);
 640         IVTV_DEBUG_YUV("Update reg 0x289c %08x->%08x\n",
 641                        yi->reg_289c, reg_289c);
 642 
 643         /* Only update filter 1 if we really need to */
 644         if (v_filter_1 != yi->v_filter_1) {
 645                 ivtv_yuv_filter(itv, -1, v_filter_1, -1);
 646                 yi->v_filter_1 = v_filter_1;
 647         }
 648 
 649         /* Only update filter 2 if we really need to */
 650         if (v_filter_2 != yi->v_filter_2) {
 651                 ivtv_yuv_filter(itv, -1, -1, v_filter_2);
 652                 yi->v_filter_2 = v_filter_2;
 653         }
 654 }
 655 
 656 /* Modify the supplied coordinate information to fit the visible osd area */
 657 static u32 ivtv_yuv_window_setup(struct ivtv *itv, struct yuv_frame_info *f)
 658 {
 659         struct yuv_frame_info *of = &itv->yuv_info.old_frame_info;
 660         int osd_crop;
 661         u32 osd_scale;
 662         u32 yuv_update = 0;
 663 
 664         /* Sorry, but no negative coords for src */
 665         if (f->src_x < 0)
 666                 f->src_x = 0;
 667         if (f->src_y < 0)
 668                 f->src_y = 0;
 669 
 670         /* Can only reduce width down to 1/4 original size */
 671         if ((osd_crop = f->src_w - 4 * f->dst_w) > 0) {
 672                 f->src_x += osd_crop / 2;
 673                 f->src_w = (f->src_w - osd_crop) & ~3;
 674                 f->dst_w = f->src_w / 4;
 675                 f->dst_w += f->dst_w & 1;
 676         }
 677 
 678         /* Can only reduce height down to 1/4 original size */
 679         if (f->src_h / f->dst_h >= 2) {
 680                 /* Overflow may be because we're running progressive,
 681                    so force mode switch */
 682                 f->interlaced_y = 1;
 683                 /* Make sure we're still within limits for interlace */
 684                 if ((osd_crop = f->src_h - 4 * f->dst_h) > 0) {
 685                         /* If we reach here we'll have to force the height. */
 686                         f->src_y += osd_crop / 2;
 687                         f->src_h = (f->src_h - osd_crop) & ~3;
 688                         f->dst_h = f->src_h / 4;
 689                         f->dst_h += f->dst_h & 1;
 690                 }
 691         }
 692 
 693         /* If there's nothing to safe to display, we may as well stop now */
 694         if ((int)f->dst_w <= 2 || (int)f->dst_h <= 2 ||
 695             (int)f->src_w <= 2 || (int)f->src_h <= 2) {
 696                 return IVTV_YUV_UPDATE_INVALID;
 697         }
 698 
 699         /* Ensure video remains inside OSD area */
 700         osd_scale = (f->src_h << 16) / f->dst_h;
 701 
 702         if ((osd_crop = f->pan_y - f->dst_y) > 0) {
 703                 /* Falls off the upper edge - crop */
 704                 f->src_y += (osd_scale * osd_crop) >> 16;
 705                 f->src_h -= (osd_scale * osd_crop) >> 16;
 706                 f->dst_h -= osd_crop;
 707                 f->dst_y = 0;
 708         } else {
 709                 f->dst_y -= f->pan_y;
 710         }
 711 
 712         if ((osd_crop = f->dst_h + f->dst_y - f->vis_h) > 0) {
 713                 /* Falls off the lower edge - crop */
 714                 f->dst_h -= osd_crop;
 715                 f->src_h -= (osd_scale * osd_crop) >> 16;
 716         }
 717 
 718         osd_scale = (f->src_w << 16) / f->dst_w;
 719 
 720         if ((osd_crop = f->pan_x - f->dst_x) > 0) {
 721                 /* Fall off the left edge - crop */
 722                 f->src_x += (osd_scale * osd_crop) >> 16;
 723                 f->src_w -= (osd_scale * osd_crop) >> 16;
 724                 f->dst_w -= osd_crop;
 725                 f->dst_x = 0;
 726         } else {
 727                 f->dst_x -= f->pan_x;
 728         }
 729 
 730         if ((osd_crop = f->dst_w + f->dst_x - f->vis_w) > 0) {
 731                 /* Falls off the right edge - crop */
 732                 f->dst_w -= osd_crop;
 733                 f->src_w -= (osd_scale * osd_crop) >> 16;
 734         }
 735 
 736         if (itv->yuv_info.track_osd) {
 737                 /* The OSD can be moved. Track to it */
 738                 f->dst_x += itv->yuv_info.osd_x_offset;
 739                 f->dst_y += itv->yuv_info.osd_y_offset;
 740         }
 741 
 742         /* Width & height for both src & dst must be even.
 743            Same for coordinates. */
 744         f->dst_w &= ~1;
 745         f->dst_x &= ~1;
 746 
 747         f->src_w += f->src_x & 1;
 748         f->src_x &= ~1;
 749 
 750         f->src_w &= ~1;
 751         f->dst_w &= ~1;
 752 
 753         f->dst_h &= ~1;
 754         f->dst_y &= ~1;
 755 
 756         f->src_h += f->src_y & 1;
 757         f->src_y &= ~1;
 758 
 759         f->src_h &= ~1;
 760         f->dst_h &= ~1;
 761 
 762         /* Due to rounding, we may have reduced the output size to <1/4 of
 763            the source. Check again, but this time just resize. Don't change
 764            source coordinates */
 765         if (f->dst_w < f->src_w / 4) {
 766                 f->src_w &= ~3;
 767                 f->dst_w = f->src_w / 4;
 768                 f->dst_w += f->dst_w & 1;
 769         }
 770         if (f->dst_h < f->src_h / 4) {
 771                 f->src_h &= ~3;
 772                 f->dst_h = f->src_h / 4;
 773                 f->dst_h += f->dst_h & 1;
 774         }
 775 
 776         /* Check again. If there's nothing to safe to display, stop now */
 777         if ((int)f->dst_w <= 2 || (int)f->dst_h <= 2 ||
 778             (int)f->src_w <= 2 || (int)f->src_h <= 2) {
 779                 return IVTV_YUV_UPDATE_INVALID;
 780         }
 781 
 782         /* Both x offset & width are linked, so they have to be done together */
 783         if ((of->dst_w != f->dst_w) || (of->src_w != f->src_w) ||
 784             (of->dst_x != f->dst_x) || (of->src_x != f->src_x) ||
 785             (of->pan_x != f->pan_x) || (of->vis_w != f->vis_w)) {
 786                 yuv_update |= IVTV_YUV_UPDATE_HORIZONTAL;
 787         }
 788 
 789         if ((of->src_h != f->src_h) || (of->dst_h != f->dst_h) ||
 790             (of->dst_y != f->dst_y) || (of->src_y != f->src_y) ||
 791             (of->pan_y != f->pan_y) || (of->vis_h != f->vis_h) ||
 792             (of->lace_mode != f->lace_mode) ||
 793             (of->interlaced_y != f->interlaced_y) ||
 794             (of->interlaced_uv != f->interlaced_uv)) {
 795                 yuv_update |= IVTV_YUV_UPDATE_VERTICAL;
 796         }
 797 
 798         return yuv_update;
 799 }
 800 
 801 /* Update the scaling register to the requested value */
 802 void ivtv_yuv_work_handler(struct ivtv *itv)
 803 {
 804         struct yuv_playback_info *yi = &itv->yuv_info;
 805         struct yuv_frame_info f;
 806         int frame = yi->update_frame;
 807         u32 yuv_update;
 808 
 809         IVTV_DEBUG_YUV("Update yuv registers for frame %d\n", frame);
 810         f = yi->new_frame_info[frame];
 811 
 812         if (yi->track_osd) {
 813                 /* Snapshot the osd pan info */
 814                 f.pan_x = yi->osd_x_pan;
 815                 f.pan_y = yi->osd_y_pan;
 816                 f.vis_w = yi->osd_vis_w;
 817                 f.vis_h = yi->osd_vis_h;
 818         } else {
 819                 /* Not tracking the osd, so assume full screen */
 820                 f.pan_x = 0;
 821                 f.pan_y = 0;
 822                 f.vis_w = 720;
 823                 f.vis_h = yi->decode_height;
 824         }
 825 
 826         /* Calculate the display window coordinates. Exit if nothing left */
 827         if (!(yuv_update = ivtv_yuv_window_setup(itv, &f)))
 828                 return;
 829 
 830         if (yuv_update & IVTV_YUV_UPDATE_INVALID) {
 831                 write_reg(0x01008080, 0x2898);
 832         } else if (yuv_update) {
 833                 write_reg(0x00108080, 0x2898);
 834 
 835                 if (yuv_update & IVTV_YUV_UPDATE_HORIZONTAL)
 836                         ivtv_yuv_handle_horizontal(itv, &f);
 837 
 838                 if (yuv_update & IVTV_YUV_UPDATE_VERTICAL)
 839                         ivtv_yuv_handle_vertical(itv, &f);
 840         }
 841         yi->old_frame_info = f;
 842 }
 843 
 844 static void ivtv_yuv_init(struct ivtv *itv)
 845 {
 846         struct yuv_playback_info *yi = &itv->yuv_info;
 847 
 848         IVTV_DEBUG_YUV("ivtv_yuv_init\n");
 849 
 850         /* Take a snapshot of the current register settings */
 851         yi->reg_2834 = read_reg(0x02834);
 852         yi->reg_2838 = read_reg(0x02838);
 853         yi->reg_283c = read_reg(0x0283c);
 854         yi->reg_2840 = read_reg(0x02840);
 855         yi->reg_2844 = read_reg(0x02844);
 856         yi->reg_2848 = read_reg(0x02848);
 857         yi->reg_2854 = read_reg(0x02854);
 858         yi->reg_285c = read_reg(0x0285c);
 859         yi->reg_2864 = read_reg(0x02864);
 860         yi->reg_2870 = read_reg(0x02870);
 861         yi->reg_2874 = read_reg(0x02874);
 862         yi->reg_2898 = read_reg(0x02898);
 863         yi->reg_2890 = read_reg(0x02890);
 864 
 865         yi->reg_289c = read_reg(0x0289c);
 866         yi->reg_2918 = read_reg(0x02918);
 867         yi->reg_291c = read_reg(0x0291c);
 868         yi->reg_2920 = read_reg(0x02920);
 869         yi->reg_2924 = read_reg(0x02924);
 870         yi->reg_2928 = read_reg(0x02928);
 871         yi->reg_292c = read_reg(0x0292c);
 872         yi->reg_2930 = read_reg(0x02930);
 873         yi->reg_2934 = read_reg(0x02934);
 874         yi->reg_2938 = read_reg(0x02938);
 875         yi->reg_293c = read_reg(0x0293c);
 876         yi->reg_2940 = read_reg(0x02940);
 877         yi->reg_2944 = read_reg(0x02944);
 878         yi->reg_2948 = read_reg(0x02948);
 879         yi->reg_294c = read_reg(0x0294c);
 880         yi->reg_2950 = read_reg(0x02950);
 881         yi->reg_2954 = read_reg(0x02954);
 882         yi->reg_2958 = read_reg(0x02958);
 883         yi->reg_295c = read_reg(0x0295c);
 884         yi->reg_2960 = read_reg(0x02960);
 885         yi->reg_2964 = read_reg(0x02964);
 886         yi->reg_2968 = read_reg(0x02968);
 887         yi->reg_296c = read_reg(0x0296c);
 888         yi->reg_2970 = read_reg(0x02970);
 889 
 890         yi->v_filter_1 = -1;
 891         yi->v_filter_2 = -1;
 892         yi->h_filter = -1;
 893 
 894         /* Set some valid size info */
 895         yi->osd_x_offset = read_reg(0x02a04) & 0x00000FFF;
 896         yi->osd_y_offset = (read_reg(0x02a04) >> 16) & 0x00000FFF;
 897 
 898         /* Bit 2 of reg 2878 indicates current decoder output format
 899            0 : NTSC    1 : PAL */
 900         if (read_reg(0x2878) & 4)
 901                 yi->decode_height = 576;
 902         else
 903                 yi->decode_height = 480;
 904 
 905         if (!itv->osd_info) {
 906                 yi->osd_vis_w = 720 - yi->osd_x_offset;
 907                 yi->osd_vis_h = yi->decode_height - yi->osd_y_offset;
 908         } else {
 909                 /* If no visible size set, assume full size */
 910                 if (!yi->osd_vis_w)
 911                         yi->osd_vis_w = 720 - yi->osd_x_offset;
 912 
 913                 if (!yi->osd_vis_h) {
 914                         yi->osd_vis_h = yi->decode_height - yi->osd_y_offset;
 915                 } else if (yi->osd_vis_h + yi->osd_y_offset > yi->decode_height) {
 916                         /* If output video standard has changed, requested height may
 917                            not be legal */
 918                         IVTV_DEBUG_WARN("Clipping yuv output - fb size (%d) exceeds video standard limit (%d)\n",
 919                                         yi->osd_vis_h + yi->osd_y_offset,
 920                                         yi->decode_height);
 921                         yi->osd_vis_h = yi->decode_height - yi->osd_y_offset;
 922                 }
 923         }
 924 
 925         /* We need a buffer for blanking when Y plane is offset - non-fatal if we can't get one */
 926         yi->blanking_ptr = kzalloc(720 * 16, GFP_ATOMIC|__GFP_NOWARN);
 927         if (yi->blanking_ptr) {
 928                 yi->blanking_dmaptr = pci_map_single(itv->pdev, yi->blanking_ptr, 720*16, PCI_DMA_TODEVICE);
 929         } else {
 930                 yi->blanking_dmaptr = 0;
 931                 IVTV_DEBUG_WARN("Failed to allocate yuv blanking buffer\n");
 932         }
 933 
 934         /* Enable YUV decoder output */
 935         write_reg_sync(0x01, IVTV_REG_VDM);
 936 
 937         set_bit(IVTV_F_I_DECODING_YUV, &itv->i_flags);
 938         atomic_set(&yi->next_dma_frame, 0);
 939 }
 940 
 941 /* Get next available yuv buffer on PVR350 */
 942 static void ivtv_yuv_next_free(struct ivtv *itv)
 943 {
 944         int draw, display;
 945         struct yuv_playback_info *yi = &itv->yuv_info;
 946 
 947         if (atomic_read(&yi->next_dma_frame) == -1)
 948                 ivtv_yuv_init(itv);
 949 
 950         draw = atomic_read(&yi->next_fill_frame);
 951         display = atomic_read(&yi->next_dma_frame);
 952 
 953         if (display > draw)
 954                 display -= IVTV_YUV_BUFFERS;
 955 
 956         if (draw - display >= yi->max_frames_buffered)
 957                 draw = (u8)(draw - 1) % IVTV_YUV_BUFFERS;
 958         else
 959                 yi->new_frame_info[draw].update = 0;
 960 
 961         yi->draw_frame = draw;
 962 }
 963 
 964 /* Set up frame according to ivtv_dma_frame parameters */
 965 static void ivtv_yuv_setup_frame(struct ivtv *itv, struct ivtv_dma_frame *args)
 966 {
 967         struct yuv_playback_info *yi = &itv->yuv_info;
 968         u8 frame = yi->draw_frame;
 969         u8 last_frame = (u8)(frame - 1) % IVTV_YUV_BUFFERS;
 970         struct yuv_frame_info *nf = &yi->new_frame_info[frame];
 971         struct yuv_frame_info *of = &yi->new_frame_info[last_frame];
 972         int lace_threshold = yi->lace_threshold;
 973 
 974         /* Preserve old update flag in case we're overwriting a queued frame */
 975         int update = nf->update;
 976 
 977         /* Take a snapshot of the yuv coordinate information */
 978         nf->src_x = args->src.left;
 979         nf->src_y = args->src.top;
 980         nf->src_w = args->src.width;
 981         nf->src_h = args->src.height;
 982         nf->dst_x = args->dst.left;
 983         nf->dst_y = args->dst.top;
 984         nf->dst_w = args->dst.width;
 985         nf->dst_h = args->dst.height;
 986         nf->tru_x = args->dst.left;
 987         nf->tru_w = args->src_width;
 988         nf->tru_h = args->src_height;
 989 
 990         /* Are we going to offset the Y plane */
 991         nf->offset_y = (nf->tru_h + nf->src_x < 512 - 16) ? 1 : 0;
 992 
 993         nf->update = 0;
 994         nf->interlaced_y = 0;
 995         nf->interlaced_uv = 0;
 996         nf->delay = 0;
 997         nf->sync_field = 0;
 998         nf->lace_mode = yi->lace_mode & IVTV_YUV_MODE_MASK;
 999 
1000         if (lace_threshold < 0)
1001                 lace_threshold = yi->decode_height - 1;
1002 
1003         /* Work out the lace settings */
1004         switch (nf->lace_mode) {
1005         case IVTV_YUV_MODE_PROGRESSIVE: /* Progressive mode */
1006                 nf->interlaced = 0;
1007                 if (nf->tru_h < 512 || (nf->tru_h > 576 && nf->tru_h < 1021))
1008                         nf->interlaced_y = 0;
1009                 else
1010                         nf->interlaced_y = 1;
1011 
1012                 if (nf->tru_h < 1021 && (nf->dst_h >= nf->src_h / 2))
1013                         nf->interlaced_uv = 0;
1014                 else
1015                         nf->interlaced_uv = 1;
1016                 break;
1017 
1018         case IVTV_YUV_MODE_AUTO:
1019                 if (nf->tru_h <= lace_threshold || nf->tru_h > 576 || nf->tru_w > 720) {
1020                         nf->interlaced = 0;
1021                         if ((nf->tru_h < 512) ||
1022                             (nf->tru_h > 576 && nf->tru_h < 1021) ||
1023                             (nf->tru_w > 720 && nf->tru_h < 1021))
1024                                 nf->interlaced_y = 0;
1025                         else
1026                                 nf->interlaced_y = 1;
1027                         if (nf->tru_h < 1021 && (nf->dst_h >= nf->src_h / 2))
1028                                 nf->interlaced_uv = 0;
1029                         else
1030                                 nf->interlaced_uv = 1;
1031                 } else {
1032                         nf->interlaced = 1;
1033                         nf->interlaced_y = 1;
1034                         nf->interlaced_uv = 1;
1035                 }
1036                 break;
1037 
1038         case IVTV_YUV_MODE_INTERLACED: /* Interlace mode */
1039         default:
1040                 nf->interlaced = 1;
1041                 nf->interlaced_y = 1;
1042                 nf->interlaced_uv = 1;
1043                 break;
1044         }
1045 
1046         if (memcmp(&yi->old_frame_info_args, nf, sizeof(*nf))) {
1047                 yi->old_frame_info_args = *nf;
1048                 nf->update = 1;
1049                 IVTV_DEBUG_YUV("Requesting reg update for frame %d\n", frame);
1050         }
1051 
1052         nf->update |= update;
1053         nf->sync_field = yi->lace_sync_field;
1054         nf->delay = nf->sync_field != of->sync_field;
1055 }
1056 
1057 /* Frame is complete & ready for display */
1058 void ivtv_yuv_frame_complete(struct ivtv *itv)
1059 {
1060         atomic_set(&itv->yuv_info.next_fill_frame,
1061                         (itv->yuv_info.draw_frame + 1) % IVTV_YUV_BUFFERS);
1062 }
1063 
1064 static int ivtv_yuv_udma_frame(struct ivtv *itv, struct ivtv_dma_frame *args)
1065 {
1066         DEFINE_WAIT(wait);
1067         int rc = 0;
1068         int got_sig = 0;
1069         /* DMA the frame */
1070         mutex_lock(&itv->udma.lock);
1071 
1072         if ((rc = ivtv_yuv_prep_user_dma(itv, &itv->udma, args)) != 0) {
1073                 mutex_unlock(&itv->udma.lock);
1074                 return rc;
1075         }
1076 
1077         ivtv_udma_prepare(itv);
1078         prepare_to_wait(&itv->dma_waitq, &wait, TASK_INTERRUPTIBLE);
1079         /* if no UDMA is pending and no UDMA is in progress, then the DMA
1080            is finished */
1081         while (test_bit(IVTV_F_I_UDMA_PENDING, &itv->i_flags) ||
1082                test_bit(IVTV_F_I_UDMA, &itv->i_flags)) {
1083                 /* don't interrupt if the DMA is in progress but break off
1084                    a still pending DMA. */
1085                 got_sig = signal_pending(current);
1086                 if (got_sig && test_and_clear_bit(IVTV_F_I_UDMA_PENDING, &itv->i_flags))
1087                         break;
1088                 got_sig = 0;
1089                 schedule();
1090         }
1091         finish_wait(&itv->dma_waitq, &wait);
1092 
1093         /* Unmap Last DMA Xfer */
1094         ivtv_udma_unmap(itv);
1095 
1096         if (got_sig) {
1097                 IVTV_DEBUG_INFO("User stopped YUV UDMA\n");
1098                 mutex_unlock(&itv->udma.lock);
1099                 return -EINTR;
1100         }
1101 
1102         ivtv_yuv_frame_complete(itv);
1103 
1104         mutex_unlock(&itv->udma.lock);
1105         return rc;
1106 }
1107 
1108 /* Setup frame according to V4L2 parameters */
1109 void ivtv_yuv_setup_stream_frame(struct ivtv *itv)
1110 {
1111         struct yuv_playback_info *yi = &itv->yuv_info;
1112         struct ivtv_dma_frame dma_args;
1113 
1114         ivtv_yuv_next_free(itv);
1115 
1116         /* Copy V4L2 parameters to an ivtv_dma_frame struct... */
1117         dma_args.y_source = NULL;
1118         dma_args.uv_source = NULL;
1119         dma_args.src.left = 0;
1120         dma_args.src.top = 0;
1121         dma_args.src.width = yi->v4l2_src_w;
1122         dma_args.src.height = yi->v4l2_src_h;
1123         dma_args.dst = yi->main_rect;
1124         dma_args.src_width = yi->v4l2_src_w;
1125         dma_args.src_height = yi->v4l2_src_h;
1126 
1127         /* ... and use the same setup routine as ivtv_yuv_prep_frame */
1128         ivtv_yuv_setup_frame(itv, &dma_args);
1129 
1130         if (!itv->dma_data_req_offset)
1131                 itv->dma_data_req_offset = yuv_offset[yi->draw_frame];
1132 }
1133 
1134 /* Attempt to dma a frame from a user buffer */
1135 int ivtv_yuv_udma_stream_frame(struct ivtv *itv, void __user *src)
1136 {
1137         struct yuv_playback_info *yi = &itv->yuv_info;
1138         struct ivtv_dma_frame dma_args;
1139         int res;
1140 
1141         ivtv_yuv_setup_stream_frame(itv);
1142 
1143         /* We only need to supply source addresses for this */
1144         dma_args.y_source = src;
1145         dma_args.uv_source = src + 720 * ((yi->v4l2_src_h + 31) & ~31);
1146         /* Wait for frame DMA. Note that serialize_lock is locked,
1147            so to allow other processes to access the driver while
1148            we are waiting unlock first and later lock again. */
1149         mutex_unlock(&itv->serialize_lock);
1150         res = ivtv_yuv_udma_frame(itv, &dma_args);
1151         mutex_lock(&itv->serialize_lock);
1152         return res;
1153 }
1154 
1155 /* IVTV_IOC_DMA_FRAME ioctl handler */
1156 int ivtv_yuv_prep_frame(struct ivtv *itv, struct ivtv_dma_frame *args)
1157 {
1158         int res;
1159 
1160 /*      IVTV_DEBUG_INFO("yuv_prep_frame\n"); */
1161         ivtv_yuv_next_free(itv);
1162         ivtv_yuv_setup_frame(itv, args);
1163         /* Wait for frame DMA. Note that serialize_lock is locked,
1164            so to allow other processes to access the driver while
1165            we are waiting unlock first and later lock again. */
1166         mutex_unlock(&itv->serialize_lock);
1167         res = ivtv_yuv_udma_frame(itv, args);
1168         mutex_lock(&itv->serialize_lock);
1169         return res;
1170 }
1171 
1172 void ivtv_yuv_close(struct ivtv *itv)
1173 {
1174         struct yuv_playback_info *yi = &itv->yuv_info;
1175         int h_filter, v_filter_1, v_filter_2;
1176 
1177         IVTV_DEBUG_YUV("ivtv_yuv_close\n");
1178         mutex_unlock(&itv->serialize_lock);
1179         ivtv_waitq(&itv->vsync_waitq);
1180         mutex_lock(&itv->serialize_lock);
1181 
1182         yi->running = 0;
1183         atomic_set(&yi->next_dma_frame, -1);
1184         atomic_set(&yi->next_fill_frame, 0);
1185 
1186         /* Reset registers we have changed so mpeg playback works */
1187 
1188         /* If we fully restore this register, the display may remain active.
1189            Restore, but set one bit to blank the video. Firmware will always
1190            clear this bit when needed, so not a problem. */
1191         write_reg(yi->reg_2898 | 0x01000000, 0x2898);
1192 
1193         write_reg(yi->reg_2834, 0x02834);
1194         write_reg(yi->reg_2838, 0x02838);
1195         write_reg(yi->reg_283c, 0x0283c);
1196         write_reg(yi->reg_2840, 0x02840);
1197         write_reg(yi->reg_2844, 0x02844);
1198         write_reg(yi->reg_2848, 0x02848);
1199         write_reg(yi->reg_2854, 0x02854);
1200         write_reg(yi->reg_285c, 0x0285c);
1201         write_reg(yi->reg_2864, 0x02864);
1202         write_reg(yi->reg_2870, 0x02870);
1203         write_reg(yi->reg_2874, 0x02874);
1204         write_reg(yi->reg_2890, 0x02890);
1205         write_reg(yi->reg_289c, 0x0289c);
1206 
1207         write_reg(yi->reg_2918, 0x02918);
1208         write_reg(yi->reg_291c, 0x0291c);
1209         write_reg(yi->reg_2920, 0x02920);
1210         write_reg(yi->reg_2924, 0x02924);
1211         write_reg(yi->reg_2928, 0x02928);
1212         write_reg(yi->reg_292c, 0x0292c);
1213         write_reg(yi->reg_2930, 0x02930);
1214         write_reg(yi->reg_2934, 0x02934);
1215         write_reg(yi->reg_2938, 0x02938);
1216         write_reg(yi->reg_293c, 0x0293c);
1217         write_reg(yi->reg_2940, 0x02940);
1218         write_reg(yi->reg_2944, 0x02944);
1219         write_reg(yi->reg_2948, 0x02948);
1220         write_reg(yi->reg_294c, 0x0294c);
1221         write_reg(yi->reg_2950, 0x02950);
1222         write_reg(yi->reg_2954, 0x02954);
1223         write_reg(yi->reg_2958, 0x02958);
1224         write_reg(yi->reg_295c, 0x0295c);
1225         write_reg(yi->reg_2960, 0x02960);
1226         write_reg(yi->reg_2964, 0x02964);
1227         write_reg(yi->reg_2968, 0x02968);
1228         write_reg(yi->reg_296c, 0x0296c);
1229         write_reg(yi->reg_2970, 0x02970);
1230 
1231         /* Prepare to restore filters */
1232 
1233         /* First the horizontal filter */
1234         if ((yi->reg_2834 & 0x0000FFFF) == (yi->reg_2834 >> 16)) {
1235                 /* An exact size match uses filter 0 */
1236                 h_filter = 0;
1237         } else {
1238                 /* Figure out which filter to use */
1239                 h_filter = ((yi->reg_2834 << 16) / (yi->reg_2834 >> 16)) >> 15;
1240                 h_filter = (h_filter >> 1) + (h_filter & 1);
1241                 /* Only an exact size match can use filter 0. */
1242                 h_filter += !h_filter;
1243         }
1244 
1245         /* Now the vertical filter */
1246         if ((yi->reg_2918 & 0x0000FFFF) == (yi->reg_2918 >> 16)) {
1247                 /* An exact size match uses filter 0/1 */
1248                 v_filter_1 = 0;
1249                 v_filter_2 = 1;
1250         } else {
1251                 /* Figure out which filter to use */
1252                 v_filter_1 = ((yi->reg_2918 << 16) / (yi->reg_2918 >> 16)) >> 15;
1253                 v_filter_1 = (v_filter_1 >> 1) + (v_filter_1 & 1);
1254                 /* Only an exact size match can use filter 0 */
1255                 v_filter_1 += !v_filter_1;
1256                 v_filter_2 = v_filter_1;
1257         }
1258 
1259         /* Now restore the filters */
1260         ivtv_yuv_filter(itv, h_filter, v_filter_1, v_filter_2);
1261 
1262         /* and clear a few registers */
1263         write_reg(0, 0x02814);
1264         write_reg(0, 0x0282c);
1265         write_reg(0, 0x02904);
1266         write_reg(0, 0x02910);
1267 
1268         /* Release the blanking buffer */
1269         if (yi->blanking_ptr) {
1270                 kfree(yi->blanking_ptr);
1271                 yi->blanking_ptr = NULL;
1272                 pci_unmap_single(itv->pdev, yi->blanking_dmaptr, 720*16, PCI_DMA_TODEVICE);
1273         }
1274 
1275         /* Invalidate the old dimension information */
1276         yi->old_frame_info.src_w = 0;
1277         yi->old_frame_info.src_h = 0;
1278         yi->old_frame_info_args.src_w = 0;
1279         yi->old_frame_info_args.src_h = 0;
1280 
1281         /* All done. */
1282         clear_bit(IVTV_F_I_DECODING_YUV, &itv->i_flags);
1283 }

/* [<][>][^][v][top][bottom][index][help] */