hc
2023-12-09 b22da3d8526a935aa31e086e63f60ff3246cb61c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
// SPDX-License-Identifier: GPL-2.0+
//
// Copyright (C) 2011 Samsung Electronics Co.Ltd
// Author: Joonyoung Shim <jy0922.shim@samsung.com>
 
#include <linux/clk.h>
#include <linux/delay.h>
#include <linux/err.h>
#include <linux/io.h>
#include <linux/platform_device.h>
#include "map.h"
#include "cpu.h"
#include "usb-phy.h"
 
#include "regs-sys-s3c64xx.h"
#include "regs-usb-hsotg-phy-s3c64xx.h"
 
enum samsung_usb_phy_type {
   USB_PHY_TYPE_DEVICE,
   USB_PHY_TYPE_HOST,
};
 
static int s3c_usb_otgphy_init(struct platform_device *pdev)
{
   struct clk *xusbxti;
   u32 phyclk;
 
   writel(readl(S3C64XX_OTHERS) | S3C64XX_OTHERS_USBMASK, S3C64XX_OTHERS);
 
   /* set clock frequency for PLL */
   phyclk = readl(S3C_PHYCLK) & ~S3C_PHYCLK_CLKSEL_MASK;
 
   xusbxti = clk_get(&pdev->dev, "xusbxti");
   if (!IS_ERR(xusbxti)) {
       switch (clk_get_rate(xusbxti)) {
       case 12 * MHZ:
           phyclk |= S3C_PHYCLK_CLKSEL_12M;
           break;
       case 24 * MHZ:
           phyclk |= S3C_PHYCLK_CLKSEL_24M;
           break;
       default:
       case 48 * MHZ:
           /* default reference clock */
           break;
       }
       clk_put(xusbxti);
   }
 
   /* TODO: select external clock/oscillator */
   writel(phyclk | S3C_PHYCLK_CLK_FORCE, S3C_PHYCLK);
 
   /* set to normal OTG PHY */
   writel((readl(S3C_PHYPWR) & ~S3C_PHYPWR_NORMAL_MASK), S3C_PHYPWR);
   mdelay(1);
 
   /* reset OTG PHY and Link */
   writel(S3C_RSTCON_PHY | S3C_RSTCON_HCLK | S3C_RSTCON_PHYCLK,
           S3C_RSTCON);
   udelay(20);    /* at-least 10uS */
   writel(0, S3C_RSTCON);
 
   return 0;
}
 
static int s3c_usb_otgphy_exit(struct platform_device *pdev)
{
   writel((readl(S3C_PHYPWR) | S3C_PHYPWR_ANALOG_POWERDOWN |
               S3C_PHYPWR_OTG_DISABLE), S3C_PHYPWR);
 
   writel(readl(S3C64XX_OTHERS) & ~S3C64XX_OTHERS_USBMASK, S3C64XX_OTHERS);
 
   return 0;
}
 
int s3c_usb_phy_init(struct platform_device *pdev, int type)
{
   if (type == USB_PHY_TYPE_DEVICE)
       return s3c_usb_otgphy_init(pdev);
 
   return -EINVAL;
}
 
int s3c_usb_phy_exit(struct platform_device *pdev, int type)
{
   if (type == USB_PHY_TYPE_DEVICE)
       return s3c_usb_otgphy_exit(pdev);
 
   return -EINVAL;
}