litex_setup: add automatic update of litex_setup (because it also changes) and be...
[litex.git] / litex_setup.py
1 #!/usr/bin/env python3
2
3 import os
4 import sys
5 import subprocess
6 import shutil
7 import hashlib
8 from collections import OrderedDict
9
10 import requests
11 import urllib.request
12
13 current_path = os.path.abspath(os.curdir)
14
15 # Repositories -------------------------------------------------------------------------------------
16
17 # name, (url, recursive clone, develop, sha1)
18 repos = [
19 # HDL
20 ("migen", ("https://github.com/m-labs/", True, True, None)),
21 ("nmigen", ("https://github.com/nmigen/", True, True, None)),
22
23 # LiteX SoC builder
24 ("pythondata-software-compiler_rt", ("https://github.com/litex-hub/", False, True, None)),
25 ("litex", ("https://github.com/enjoy-digital/", False, True, None)),
26
27 # LiteX cores ecosystem
28 ("liteeth", ("https://github.com/enjoy-digital/", False, True, None)),
29 ("litedram", ("https://github.com/enjoy-digital/", False, True, None)),
30 ("litepcie", ("https://github.com/enjoy-digital/", False, True, None)),
31 ("litesata", ("https://github.com/enjoy-digital/", False, True, None)),
32 ("litesdcard", ("https://github.com/enjoy-digital/", False, True, None)),
33 ("liteiclink", ("https://github.com/enjoy-digital/", False, True, None)),
34 ("litevideo", ("https://github.com/enjoy-digital/", False, True, None)),
35 ("litescope", ("https://github.com/enjoy-digital/", False, True, None)),
36 ("litejesd204b", ("https://github.com/enjoy-digital/", False, True, None)),
37 ("litespi", ("https://github.com/litex-hub/", False, True, None)),
38 ("litehyperbus", ("https://github.com/litex-hub/", False, True, None)),
39
40 # LiteX boards support
41 ("litex-boards", ("https://github.com/litex-hub/", False, True, None)),
42
43 # Optional LiteX data
44 ("pythondata-misc-tapcfg", ("https://github.com/litex-hub/", False, True, None)),
45 ("pythondata-cpu-lm32", ("https://github.com/litex-hub/", False, True, None)),
46 ("pythondata-cpu-mor1kx", ("https://github.com/litex-hub/", False, True, None)),
47 ("pythondata-cpu-picorv32", ("https://github.com/litex-hub/", False, True, None)),
48 ("pythondata-cpu-serv", ("https://github.com/litex-hub/", False, True, None)),
49 ("pythondata-cpu-vexriscv", ("https://github.com/litex-hub/", False, True, None)),
50 ("pythondata-cpu-rocket", ("https://github.com/litex-hub/", False, True, None)),
51 ("pythondata-cpu-minerva", ("https://github.com/litex-hub/", False, True, None)),
52 ("pythondata-cpu-microwatt", ("https://github.com/litex-hub/", False, True, 0xa7859fb)),
53 ("pythondata-cpu-blackparrot", ("https://github.com/litex-hub/", False, True, None)),
54 ]
55
56 repos = OrderedDict(repos)
57
58 # RISC-V toolchain download ------------------------------------------------------------------------
59
60 def sifive_riscv_download():
61 base_url = "https://static.dev.sifive.com/dev-tools/"
62 base_file = "riscv64-unknown-elf-gcc-8.3.0-2019.08.0-x86_64-"
63
64 # Windows
65 if (sys.platform.startswith("win") or sys.platform.startswith("cygwin")):
66 end_file = "w64-mingw32.zip"
67 # Linux
68 elif sys.platform.startswith("linux"):
69 end_file = "linux-ubuntu14.tar.gz"
70 # Mac OS
71 elif sys.platform.startswith("darwin"):
72 end_file = "apple-darwin.tar.gz"
73 else:
74 raise NotImplementedError(sys.platform)
75 fn = base_file + end_file
76
77 if not os.path.exists(fn):
78 url = base_url + fn
79 print("Downloading", url, "to", fn)
80 urllib.request.urlretrieve(url, fn)
81 else:
82 print("Using existing file", fn)
83
84 print("Extracting", fn)
85 shutil.unpack_archive(fn)
86
87 # Setup --------------------------------------------------------------------------------------------
88
89 if os.environ.get("TRAVIS", "") == "true":
90 # Ignore `ssl.SSLCertVerificationError` on CI.
91 import ssl
92 ssl._create_default_https_context = ssl._create_unverified_context
93
94 if len(sys.argv) < 2:
95 print("Available commands:")
96 print("- init")
97 print("- update")
98 print("- install (add --user to install to user directory)")
99 print("- gcc")
100 print("- dev (dev mode, disable automatic litex_setup.py update)")
101 exit()
102
103 # Check/Update litex_setup.py
104
105 litex_setup_url = "https://raw.githubusercontent.com/enjoy-digital/litex/master/litex_setup.py"
106 current_sha1 = hashlib.sha1(open(os.path.realpath(__file__)).read().encode("utf-8")).hexdigest()
107 print("[checking litex_setup.py]...")
108 try:
109 r = requests.get(litex_setup_url)
110 if r.status_code != 404:
111 upstream_sha1 = hashlib.sha1(r.content).hexdigest()
112 if current_sha1 != upstream_sha1:
113 if "dev" not in sys.argv[1:]:
114 print("[updating litex_setup.py]...")
115 with open(os.path.realpath(__file__), "wb") as f:
116 f.write(r.content)
117 os.execl(sys.executable, sys.executable, *sys.argv)
118 except:
119 pass
120
121 # Repositories cloning
122 if "init" in sys.argv[1:]:
123 for name in repos.keys():
124 os.chdir(os.path.join(current_path))
125 if not os.path.exists(name):
126 url, need_recursive, need_develop, sha1 = repos[name]
127 # clone repo (recursive if needed)
128 print("[cloning " + name + "]...")
129 full_url = url + name
130 opts = "--recursive" if need_recursive else ""
131 subprocess.check_call("git clone " + full_url + " " + opts, shell=True)
132 if sha1 is not None:
133 os.chdir(os.path.join(current_path, name))
134 os.system("git checkout {:7x}".format(sha1))
135
136 # Repositories update
137 if "update" in sys.argv[1:]:
138 for name in repos.keys():
139 os.chdir(os.path.join(current_path))
140 url, need_recursive, need_develop, sha1 = repos[name]
141 print(url)
142 if not os.path.exists(name):
143 raise Exception("{} not initialized, please (re)-run init and install first.".format(name))
144 # update
145 print("[updating " + name + "]...")
146 os.chdir(os.path.join(current_path, name))
147 subprocess.check_call("git checkout master", shell=True)
148 subprocess.check_call("git pull --ff-only", shell=True)
149 if sha1 is not None:
150 os.chdir(os.path.join(current_path, name))
151 os.system("git checkout {:7x}".format(sha1))
152
153 # Repositories installation
154 if "install" in sys.argv[1:]:
155 for name in repos.keys():
156 os.chdir(os.path.join(current_path))
157 url, need_recursive, need_develop, sha1 = repos[name]
158 # develop if needed
159 print("[installing " + name + "]...")
160 if need_develop:
161 os.chdir(os.path.join(current_path, name))
162 if "--user" in sys.argv[1:]:
163 subprocess.check_call("python3 setup.py develop --user", shell=True)
164 else:
165 subprocess.check_call("python3 setup.py develop", shell=True)
166
167 if "--user" in sys.argv[1:]:
168 if ".local/bin" not in os.environ.get("PATH", ""):
169 print("Make sure that ~/.local/bin is in your PATH")
170 print("export PATH=$PATH:~/.local/bin")
171
172 # RISC-V GCC installation
173 if "gcc" in sys.argv[1:]:
174 os.chdir(os.path.join(current_path))
175 sifive_riscv_download()
176 if "riscv64" not in os.environ.get("PATH", ""):
177 print("Make sure that the downloaded RISC-V compiler is in your $PATH.")
178 print("export PATH=$PATH:$(echo $PWD/riscv64-*/bin/)")