-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathfrequency_assignment.py
More file actions
192 lines (167 loc) · 8.48 KB
/
frequency_assignment.py
File metadata and controls
192 lines (167 loc) · 8.48 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
from typing import Any, Iterable, Union
import logging
from timeit import default_timer as timer
import numpy as np
from scipy import stats
import qmkphp
from single_frequency import min_rec_power_single_freq
from two_frequencies import min_rec_power_two_freq
from util import to_decibel
LOGGER = logging.getLogger(__name__)
def generate_random_assignment(num_users, num_freq, num_assign_freq=2):
choices = np.random.choice(np.arange(num_freq), size=(num_users, num_assign_freq),
replace=False)
assignments = np.zeros((num_users, num_freq))
for _user, _assignment in enumerate(choices):
assignments[_user, _assignment] = 1
assignments = assignments.T
return assignments
def generate_round_robin_assignment(num_users, num_freq, num_assign_freq=2,
in_blocks=False):
if not in_blocks:
assignments = np.zeros((num_freq, num_users))
for _i in range(num_assign_freq):
assignments = assignments + np.eye(num_freq, num_users, k=-_i*num_users)
else:
assignments = np.eye(num_users)
assignments = np.repeat(assignments, num_assign_freq, axis=0)
assignments = np.pad(assignments, ([0, num_freq-len(assignments)], [0, 0]))
return assignments
def round_robin_profit_based(profits: np.array, num_assign_freq=2):
num_users, num_freq, _ = np.shape(profits)
_assignments = [[] for _ in range(num_users)]
temp_prof = np.copy(profits)
for user in range(num_users):
f1 = np.argmax(np.diag(temp_prof[user]))
_assignments[user].append(f1)
temp_prof[:, :, f1] = -np.inf
for user in range(num_users):
f2 = np.argmax(temp_prof[user][_assignments[user][0]])
_assignments[user].append(f2)
temp_prof[:, :, f2] = -np.inf
assignments = np.zeros((num_freq, num_users))
for user, user_assignemnts in enumerate(_assignments):
assignments[user_assignemnts, user] = 1
return assignments
def create_profit_matrix(d_min: float, d_max: float, frequencies: Iterable,
h_tx: float, h_rx: float):
main_diag = [min_rec_power_single_freq(d_min, d_max, f, h_tx, h_rx)
for f in frequencies]
idx_triu = np.triu_indices(len(frequencies), k=1)
profits = np.zeros((len(frequencies), len(frequencies)))
for i, j in zip(*idx_triu):
_power_sum = min_rec_power_two_freq(d_min, d_max, frequencies[i],
frequencies[j], h_tx, h_rx)
profits[i, j] = profits[j, i] = _power_sum - main_diag[i] - main_diag[j]
#profits[i, j] = profits[j, i] = 1
profits[np.diag_indices_from(profits)] = main_diag
#profits = to_decibel(profits)
return profits
def generate_user_parameters(num_users: int, h_tx: float):
h_rx = 2*np.random.rand(num_users)+1.
d_min = 20*np.random.rand(num_users)+20
_width = 90*np.random.rand(num_users) + 10
#d_min = 20*np.random.rand(num_users)+20
#_width = 150*np.random.rand(num_users) + 50
d_max = d_min + _width
users = [{"h_tx": h_tx, "h_rx": _h_rx, "d_min": _d_min, "d_max": _d_max}
for _h_rx, _d_min, _d_max in zip(h_rx, d_min, d_max)]
return users
def main(num_users: int = 3, num_frequencies: int = 10, h_tx: float = 10.,
frequencies: Iterable = None,
users: Union[Iterable[dict], None] = None, num_trials: int = 100):
if users is not None:
num_users = len(users)
num_trials = 1
else:
users = generate_user_parameters(num_users, h_tx)
if frequencies is None:
#frequencies = np.linspace(0, 300e6, num_frequencies) + 2.4e9 #np.arange(0, 100)*1e6 + 2.4e9
frequencies = np.linspace(0, 100e6, num_frequencies) + 2.4e9 #np.arange(0, 100)*1e6 + 2.4e9
num_frequencies = len(frequencies)
LOGGER.info(f"Number of users: {num_users:d}")
LOGGER.info(f"Number of frequencies: {num_frequencies:d}")
capacities = [2]*num_users
weights = np.ones(num_frequencies)
results_qmkp = {"time": [], "profit": [], "time_total": []}
comparison = {"random": generate_random_assignment,
"robin": generate_round_robin_assignment,
"robinblocks": lambda *x: generate_round_robin_assignment(*x, in_blocks=True)}
results_comparison = {k: {"time": [], "profit": []} for k in comparison}
results_rr = []
for trial in range(num_trials):
LOGGER.info(f"Working on trial {trial+1:d}/{num_trials:d}")
start_profit = timer()
profits = [create_profit_matrix(frequencies=frequencies, **user_info)
for user_info in users]
profits = np.array(profits)
LOGGER.debug(f"Profit shape: {np.shape(profits)}")
start = timer()
assignments = qmkphp.constructive_procedure(capacities=capacities,
weights=weights,
profits=profits)
end = timer()
total_profit = qmkphp.total_profit_qmkp(profits, assignments)
results_qmkp["time_total"].append(end - start_profit)
results_qmkp["time"].append(end - start)
results_qmkp["profit"].append(total_profit/num_users)
LOGGER.debug(f"QMKP Time: {end-start:.3f}")
LOGGER.debug(f"QMKP Profit: {to_decibel(total_profit/num_users):.1f}")
assignments = round_robin_profit_based(profits)
total_profit = qmkphp.total_profit_qmkp(profits, assignments)
results_rr.append(total_profit/num_users)
for _comp, assign_func in comparison.items():
LOGGER.debug("Running comparison algorithm: {}".format(_comp))
start = timer()
assignments = assign_func(num_users, len(frequencies))
end = timer()
total_profit = qmkphp.total_profit_qmkp(profits, assignments)
results_comparison[_comp]["time"].append(end - start)
results_comparison[_comp]["profit"].append(total_profit/num_users)
LOGGER.debug(f"Comparison Time: {end-start:.3f}")
LOGGER.debug(f"Comparison Profit: {to_decibel(total_profit/num_users):.1f}")
users = generate_user_parameters(num_users, h_tx)
profit_results = {"QMKP": to_decibel(np.mean(results_qmkp['profit']))}
profit_results.update({k: to_decibel(np.mean(_results['profit']))
for k, _results in results_comparison.items()})
LOGGER.info("QMKP Times: {}".format(stats.describe(results_qmkp['time'])))
LOGGER.info("QMKP Times Total: {}".format(stats.describe(results_qmkp['time_total'])))
#LOGGER.info("Random Times: {}".format(stats.describe(results_random['time'])))
#LOGGER.debug("QMKP Profit: {}".format(stats.describe(results_qmkp['profit'])))
#LOGGER.debug("Random Profit: {}".format(stats.describe(results_random['profit'])))
LOGGER.info("QMKP Avg Profit: {} dB".format(profit_results["QMKP"]))
LOGGER.info("RR Best Avg Profit: {} dB".format(to_decibel(np.mean(results_rr))))
LOGGER.info("Random Avg Profit: {} dB".format(profit_results))
return results_qmkp['profit'], results_rr
if __name__ == "__main__":
import argparse
parser = argparse.ArgumentParser()
parser.add_argument("-u", "--num_users", type=int, default=3)
parser.add_argument("-f", "--num_frequencies", type=int, default=10)
parser.add_argument("-t", "--h_tx", type=float, default=10)
parser.add_argument("-c", "--config",
help="Config file that contains the user definitions")
parser.add_argument("-n", "--num_trials", type=int, default=100)
parser.add_argument("-v", "--verbosity", action="count", default=0,
help="Increase output verbosity")
args = vars(parser.parse_args())
_config_file = args.pop("config")
if _config_file is not None:
import ast
with open(_config_file, 'r') as _config:
_config_string = _config.read()
args['users'] = ast.literal_eval(_config_string)
verb = args.pop("verbosity")
logging.basicConfig(format="%(asctime)s - [%(levelname)8s]: %(message)s",
handlers=[
logging.FileHandler("main.log", encoding="utf-8"),
logging.StreamHandler()
])
loglevel = logging.WARNING - verb*10
LOGGER.setLevel(loglevel)
results_qmkp, results_rr = main(**args)
#import matplotlib.pyplot as plt
#plt.hist(to_decibel(results_qmkp), bins=100, label="QMKP")
#plt.hist(to_decibel(results_rr), bins=100, label="RR")
#plt.legend()
#plt.show()